Skip to content

Commit 27f4a05

Browse files
committed
uefi: Add safe protocol bindings for EFI_EXT_SCSI_PASS_THRU_PROTOCOL
1 parent d34c353 commit 27f4a05

File tree

9 files changed

+709
-2
lines changed

9 files changed

+709
-2
lines changed

uefi-raw/src/protocol/scsi.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ pub struct ScsiIoScsiRequestPacket {
8484
///
8585
/// A `timeout` value of 0 indicates that the function will wait indefinitely for
8686
/// the execution to complete. If the execution time exceeds the specified `timeout`
87-
/// (greater than 0), the function will return `EFI_TIMEOUT`.
87+
/// (greater than 0), the function will return [`Status::TIMEOUT`].
8888
pub timeout: u64,
8989

9090
/// A pointer to the data buffer for reading from the device in read and bidirectional commands.

uefi-test-runner/src/proto/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ pub fn test() {
2525
shell_params::test();
2626
string::test();
2727
misc::test();
28+
scsi::test();
2829

2930
#[cfg(any(
3031
target_arch = "x86",
@@ -81,5 +82,6 @@ mod shell_params;
8182
target_arch = "aarch64"
8283
))]
8384
mod shim;
85+
mod scsi;
8486
mod string;
8587
mod tcg;
+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
// SPDX-License-Identifier: MIT OR Apache-2.0
2+
3+
mod pass_thru;
4+
5+
pub fn test() {
6+
pass_thru::test();
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
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+
}

uefi/CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
- Added `boot::signal_event`.
55
- Added conversions between `proto::network::IpAddress` and `core::net` types.
66
- Added conversions between `proto::network::MacAddress` and the `[u8; 6]` type that's more commonly used to represent MAC addresses.
7+
- Added `proto::scsi::pass_thru::ExtScsiPassThru`.
78

89
## Changed
910
- **Breaking:** Removed `BootPolicyError` as `BootPolicy` construction is no

uefi/src/proto/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,8 @@ pub mod misc;
2020
pub mod network;
2121
pub mod pi;
2222
pub mod rng;
23+
#[cfg(feature = "alloc")]
24+
pub mod scsi;
2325
pub mod security;
2426
pub mod shell_params;
2527
pub mod shim;

0 commit comments

Comments
 (0)