1
+ // SPDX-License-Identifier: MIT OR Apache-2.0
2
+
3
+ use uefi:: proto:: scsi:: pass_thru:: ExtScsiPassThru ;
4
+ use uefi:: proto:: scsi:: ScsiRequestBuilder ;
5
+
6
+ pub fn test ( ) {
7
+ info ! ( "Running extended SCSI Pass Thru tests" ) ;
8
+ test_allocating_api ( ) ;
9
+ test_reusing_buffer_api ( ) ;
10
+ }
11
+
12
+ fn test_allocating_api ( ) {
13
+ let scsi_ctrl_handles = uefi:: boot:: find_handles :: < ExtScsiPassThru > ( ) . unwrap ( ) ;
14
+ assert_eq ! ( scsi_ctrl_handles. len( ) , 2 ) ;
15
+
16
+ let mut found_drive = false ;
17
+
18
+ for handle in scsi_ctrl_handles {
19
+ let scsi_pt = uefi:: boot:: open_protocol_exclusive :: < ExtScsiPassThru > ( handle) . unwrap ( ) ;
20
+ for device in scsi_pt. iter_devices ( ) {
21
+ // see: https://www.seagate.com/files/staticfiles/support/docs/manual/Interface%20manuals/100293068j.pdf
22
+ // 3.6 INQUIRY command
23
+ let request = ScsiRequestBuilder :: read ( scsi_pt. io_align ( ) )
24
+ . with_timeout ( core:: time:: Duration :: from_millis ( 500 ) )
25
+ . with_command_data ( & [ 0x12 , 0x00 , 0x00 , 0x00 , 0xFF , 0x00 ] ) . unwrap ( )
26
+ . with_read_buffer ( 255 ) . unwrap ( )
27
+ . build ( ) ;
28
+ let Ok ( response) = device. execute_command ( request) else {
29
+ continue ; // no device
30
+ } ;
31
+ let bfr = response. read_buffer ( ) . unwrap ( ) ;
32
+ // more no device checks
33
+ if bfr. len ( ) < 32 { continue ; }
34
+ if bfr[ 0 ] & 0b00011111 == 0x1F { continue ; }
35
+
36
+ // found device
37
+ let vendor_id = core:: str:: from_utf8 ( & bfr[ 8 ..16 ] ) . unwrap ( ) . trim ( ) ;
38
+ let product_id = core:: str:: from_utf8 ( & bfr[ 16 ..32 ] ) . unwrap ( ) . trim ( ) ;
39
+ if vendor_id == "uefi-rs" && product_id == "ExtScsiPassThru" {
40
+ info ! ( "Found Testdisk at: {:?} | {}" , device. target( ) , device. lun( ) ) ;
41
+ found_drive = true ;
42
+ }
43
+ }
44
+ }
45
+
46
+ assert ! ( found_drive) ;
47
+ }
48
+
49
+ fn test_reusing_buffer_api ( ) {
50
+ let scsi_ctrl_handles = uefi:: boot:: find_handles :: < ExtScsiPassThru > ( ) . unwrap ( ) ;
51
+ assert_eq ! ( scsi_ctrl_handles. len( ) , 2 ) ;
52
+
53
+ let mut found_drive = false ;
54
+
55
+ for handle in scsi_ctrl_handles {
56
+ let scsi_pt = uefi:: boot:: open_protocol_exclusive :: < ExtScsiPassThru > ( handle) . unwrap ( ) ;
57
+ let mut cmd_bfr = scsi_pt. alloc_io_buffer ( 6 ) . unwrap ( ) ;
58
+ cmd_bfr. copy_from ( & [ 0x12 , 0x00 , 0x00 , 0x00 , 0xFF , 0x00 ] ) ;
59
+ let mut read_bfr = scsi_pt. alloc_io_buffer ( 255 ) . unwrap ( ) ;
60
+
61
+ for device in scsi_pt. iter_devices ( ) {
62
+ // see: https://www.seagate.com/files/staticfiles/support/docs/manual/Interface%20manuals/100293068j.pdf
63
+ // 3.6 INQUIRY command
64
+ let request = ScsiRequestBuilder :: read ( scsi_pt. io_align ( ) )
65
+ . with_timeout ( core:: time:: Duration :: from_millis ( 500 ) )
66
+ . use_command_buffer ( & mut cmd_bfr) . unwrap ( )
67
+ . use_read_buffer ( & mut read_bfr) . unwrap ( )
68
+ . build ( ) ;
69
+ let Ok ( response) = device. execute_command ( request) else {
70
+ continue ; // no device
71
+ } ;
72
+ let bfr = response. read_buffer ( ) . unwrap ( ) ;
73
+ // more no device checks
74
+ if bfr. len ( ) < 32 { continue ; }
75
+ if bfr[ 0 ] & 0b00011111 == 0x1F { continue ; }
76
+
77
+ // found device
78
+ let vendor_id = core:: str:: from_utf8 ( & bfr[ 8 ..16 ] ) . unwrap ( ) . trim ( ) ;
79
+ let product_id = core:: str:: from_utf8 ( & bfr[ 16 ..32 ] ) . unwrap ( ) . trim ( ) ;
80
+ if vendor_id == "uefi-rs" && product_id == "ExtScsiPassThru" {
81
+ info ! ( "Found Testdisk at: {:?} | {}" , device. target( ) , device. lun( ) ) ;
82
+ found_drive = true ;
83
+ }
84
+ }
85
+ }
86
+
87
+ assert ! ( found_drive) ;
88
+ }
0 commit comments