use super::ScsiError;
use super::SimpleScsiDisk;
use crate::UNMAP_RANGE_DESCRIPTOR_COUNT_MAX;
use crate::VHDMP_MAX_WRITE_SAME_LENGTH_BYTES;
use crate::scsi;
use guestmem::MemoryWrite;
use guid::Guid;
use scsi::AdditionalSenseCode;
use scsi_buffers::RequestBuffers;
use scsi_core::Request;
use zerocopy::FromBytes;
use zerocopy::FromZeros;
use zerocopy::Immutable;
use zerocopy::IntoBytes;
use zerocopy::KnownLayout;
type U16BE = zerocopy::byteorder::U16<zerocopy::byteorder::BigEndian>;
type U32BE = zerocopy::byteorder::U32<zerocopy::byteorder::BigEndian>;
type U64BE = zerocopy::byteorder::U64<zerocopy::byteorder::BigEndian>;
#[repr(C)]
#[derive(Debug, IntoBytes, Immutable, KnownLayout, FromBytes)]
struct VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptorHeader {
pub ecop_descriptor_type: U16BE,
pub ecop_descriptor_length: U16BE,
}
#[repr(C)]
#[derive(Debug, IntoBytes, Immutable, KnownLayout, FromBytes)]
struct VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptor {
pub header: VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptorHeader,
pub microsoft_signature: u8, pub microsoft_command_id_and_versionsion: u8, pub populate_token_and_write_using_token_command_op_code: u8, pub receive_rod_token_information_command_op_code: u8, pub reserved1: [u8; 2],
pub maximum_range_descriptors: U16BE,
pub maximum_inactivity_timer: U32BE,
pub default_inactivity_timer: U32BE,
pub maximum_token_transfer_size: U64BE,
pub optimal_transfer_count: U64BE,
}
pub const INQUIRY_DATA_TEMPLATE: scsi::InquiryData = scsi::InquiryData {
header: scsi::InquiryDataHeader {
device_type: scsi::DIRECT_ACCESS_DEVICE,
flags2: scsi::InquiryDataFlag2::new()
.with_device_type_modifier(0x00)
.with_removable_media(false),
versions: scsi::T10_VERSION_SPC3,
flags3: scsi::InquiryDataFlag3::new()
.with_response_data_format(scsi::T10_RESPONSE_DATA_SPC3)
.with_aerc(false)
.with_hi_support(false)
.with_norm_aca(false)
.with_reserved_bit(false),
additional_length: (size_of::<scsi::InquiryData>() - size_of::<scsi::InquiryDataHeader>())
as u8,
},
reserved: [0; 2],
misc: 0,
vendor_id: *b"Msft ",
product_id: *b"Virtual Disk ",
product_revision_level: *b"1.0 ",
vendor_specific: [0; 20],
reserved3: [0; 2],
version_descriptors: [0; 8],
reserved4: [0; 30],
};
fn write_vpd_page<T: ?Sized + IntoBytes + Immutable + KnownLayout>(
external_data: &RequestBuffers<'_>,
allocation_length: usize,
page_code: u8,
page_data: &T,
) -> Result<usize, ScsiError> {
let header = scsi::VpdPageHeader {
device_type: scsi::DIRECT_ACCESS_DEVICE,
page_code,
reserved: 0,
page_length: size_of_val(page_data).try_into().unwrap(),
};
let tx = std::cmp::min(
allocation_length,
size_of_val(&header) + size_of_val(page_data),
);
let mut writer = external_data.writer();
writer
.write(header.as_bytes())
.map_err(ScsiError::MemoryAccess)?;
writer
.write(&page_data.as_bytes()[..tx - size_of_val(&header)])
.map_err(ScsiError::MemoryAccess)?;
Ok(tx)
}
impl SimpleScsiDisk {
fn handle_vpd_supported_pages(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
let mut supported_pages = vec![
scsi::VPD_SUPPORTED_PAGES, scsi::VPD_DEVICE_IDENTIFIERS, scsi::VPD_MSFT_PAGING_EXTENT_PROPERTIES, scsi::VPD_MSFT_VIRTUAL_DEVICE_PROPERTIES, ];
if self.scsi_parameters.support_odx {
supported_pages.push(scsi::VPD_THIRD_PARTY_COPY);
}
if self.scsi_parameters.support_unmap {
supported_pages.extend([
scsi::VPD_BLOCK_DEVICE_CHARACTERISTICS,
scsi::VPD_BLOCK_LIMITS,
scsi::VPD_LOGICAL_BLOCK_PROVISIONING,
]);
}
if !self.scsi_parameters.serial_number.is_empty() {
supported_pages.push(scsi::VPD_SERIAL_NUMBER);
}
supported_pages.sort_unstable();
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_SUPPORTED_PAGES,
supported_pages.as_slice(),
)
}
fn handle_vpd_serial_number(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_SERIAL_NUMBER,
self.scsi_parameters.serial_number.as_slice(),
)
}
fn handle_vpd_device_identifiers(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
#[repr(C)]
#[derive(IntoBytes, Immutable, KnownLayout)]
struct Ids {
t10_id: scsi::VpdT10Id,
naa_id: scsi::VpdNaaId,
}
let mut page = Ids {
t10_id: scsi::VpdT10Id {
header: scsi::VpdIdentificationDescriptor {
code_set: scsi::VPD_CODE_SET_BINARY,
identifiertype: scsi::VPD_IDENTIFIER_TYPE_VENDOR_ID, reserved3: 0x00,
identifier_length: (size_of::<scsi::VpdT10Id>()
- size_of::<scsi::VpdIdentificationDescriptor>())
as u8,
},
vendor_id: self.scsi_parameters.identity.vendor_id.into(),
context_guid: self.scsi_parameters.disk_id,
},
naa_id: scsi::VpdNaaId {
header: scsi::VpdIdentificationDescriptor {
code_set: scsi::VPD_CODE_SET_BINARY,
identifiertype: scsi::VPD_IDENTIFIER_TYPE_FCPH_NAME, reserved3: 0x00,
identifier_length: (size_of::<scsi::VpdNaaId>()
- size_of::<scsi::VpdIdentificationDescriptor>())
as u8,
},
ouid_msb: 0x60, ouid_middle: [0x02, 0x24],
ouid_lsb: 0x80,
vendor_specific_id: [0; 12],
},
};
let id_split_size = page.naa_id.vendor_specific_id.len() / 2;
page.naa_id.vendor_specific_id[..id_split_size]
.copy_from_slice(&self.scsi_parameters.disk_id.as_bytes()[..id_split_size]);
page.naa_id.vendor_specific_id[id_split_size..]
.copy_from_slice(&page.t10_id.context_guid[10..]);
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_DEVICE_IDENTIFIERS,
&page,
)
}
fn handle_vpd_block_limits(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
let max_write_same_length = (self.scsi_parameters.maximum_transfer_length as u64)
.min(VHDMP_MAX_WRITE_SAME_LENGTH_BYTES)
>> self.sector_shift;
let optimal_unmap_granularity = self.scsi_parameters.optimal_unmap_sectors.max(1);
let page = scsi::VpdBlockLimitsDescriptor {
reserved0: 0x00,
max_compare_and_write_length: 0, max_unmap_lba_count: u32::MAX.into(),
max_unmap_block_descriptor_count: u32::from(UNMAP_RANGE_DESCRIPTOR_COUNT_MAX).into(),
optimal_unmap_granularity: optimal_unmap_granularity.into(),
unmap_granularity_alignment: [0x80, 0x00, 0x00, 0x00], max_write_same_length: max_write_same_length.into(),
..FromZeros::new_zeroed()
};
tracing::debug!(
max_lba_count = page.max_unmap_lba_count.get(), max_block_descriptor_count = page.max_unmap_block_descriptor_count.get(),
optimal_unmap_granularity,
"handle_vpd_block_limits unmap properties",
);
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_BLOCK_LIMITS,
&page,
)
}
fn handle_vpd_block_device_characteristics(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
let page = scsi::VpdBlockDeviceCharacteristicsPage {
medium_rotation_rate: self.scsi_parameters.medium_rotation_rate.into(),
..FromZeros::new_zeroed()
};
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_BLOCK_DEVICE_CHARACTERISTICS,
&page,
)
}
fn handle_vpd_third_party_copy(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
const VHDMP_MAX_BYTES_PER_OFFLOAD: u64 = 64 * 1024 * 1024;
let page = VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptor {
header: VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptorHeader {
ecop_descriptor_type: 0.into(),
ecop_descriptor_length: ((size_of::<
VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptor,
>() - size_of::<
VhdmpVpdWindowsBlockDeviceRodLimitsEcopDescriptorHeader,
>()) as u16)
.into(),
},
microsoft_signature: 0x4D, microsoft_command_id_and_versionsion: 0x1F, populate_token_and_write_using_token_command_op_code: 0x83, receive_rod_token_information_command_op_code: 0x84, reserved1: [0; 2],
maximum_range_descriptors: 8.into(),
maximum_inactivity_timer: 0.into(), default_inactivity_timer: 0.into(),
maximum_token_transfer_size: (VHDMP_MAX_BYTES_PER_OFFLOAD >> self.sector_shift).into(),
optimal_transfer_count: (VHDMP_MAX_BYTES_PER_OFFLOAD >> self.sector_shift).into(),
};
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_THIRD_PARTY_COPY,
&page,
)
}
fn handle_vpd_msft_virtual_device_properties(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
const VPD_MSFT_VIRTUAL_DEVICE_PROPERTIES_PAGE_SIGNATURE: Guid =
guid::guid!("89a98f15-c928-4d8b-94cd-ef51faa99d33");
let page = scsi::VpdMsftVirtualDevicePropertiesPage {
version: 1, flags: 0, reserved: [0; 2],
signature: VPD_MSFT_VIRTUAL_DEVICE_PROPERTIES_PAGE_SIGNATURE.into(),
};
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_MSFT_VIRTUAL_DEVICE_PROPERTIES,
&page,
)
}
fn handle_vpd_msft_paging_extent_properties(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
let page = scsi::VpdMsftPagingExtentPropertiesPage {
version: 1,
mode_select_extension: 1,
reserved: [0; 2],
};
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_MSFT_PAGING_EXTENT_PROPERTIES,
&page,
)
}
fn handle_vpd_logical_block_provisioning(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
sector_count: u64,
) -> Result<usize, ScsiError> {
let threshold_exponent = (sector_count >> 32)
.next_power_of_two()
.trailing_zeros()
.max(1) as u8;
let mut page = scsi::VpdLogicalBlockProvisioningPage {
threshold_exponent,
flags: 0,
provisioning_type: scsi::PROVISIONING_TYPE_THIN,
reserved2: 0,
};
if self.scsi_parameters.support_unmap {
page.flags = 0x80;
}
write_vpd_page(
external_data,
allocation_length,
scsi::VPD_LOGICAL_BLOCK_PROVISIONING,
&page,
)
}
fn handle_no_vpd_page(
&self,
external_data: &RequestBuffers<'_>,
allocation_length: usize,
) -> Result<usize, ScsiError> {
if allocation_length < size_of::<scsi::InquiryDataHeader>() {
return Err(ScsiError::SrbError);
}
let page = scsi::InquiryData {
misc: 0x02, vendor_id: self.scsi_parameters.identity.vendor_id.into(),
product_id: self.scsi_parameters.identity.product_id.into(),
product_revision_level: self.scsi_parameters.identity.product_revision_level.into(),
..INQUIRY_DATA_TEMPLATE
};
let tx = std::cmp::min(allocation_length, size_of_val(&page));
external_data
.writer()
.write(&page.as_bytes()[..tx])
.map_err(ScsiError::MemoryAccess)?;
Ok(tx)
}
pub(crate) fn handle_inquiry(
&self,
external_data: &RequestBuffers<'_>,
request: &Request,
sector_count: u64,
) -> Result<usize, ScsiError> {
let cdb = scsi::CdbInquiry::read_from_prefix(&request.cdb[..])
.unwrap()
.0; let allocation_length = cdb.allocation_length.get() as usize;
if external_data.len() < allocation_length {
return Err(ScsiError::SrbError);
}
if cdb.flags.csd() {
return Err(ScsiError::IllegalRequest(AdditionalSenseCode::INVALID_CDB));
}
let enable_vpd = cdb.flags.vpd();
if cdb.page_code != 0 && !enable_vpd {
return Err(ScsiError::IllegalRequest(AdditionalSenseCode::INVALID_CDB));
}
if enable_vpd {
if allocation_length < size_of::<scsi::VpdPageHeader>() {
return Err(ScsiError::SrbError);
}
match cdb.page_code {
scsi::VPD_SUPPORTED_PAGES => {
self.handle_vpd_supported_pages(external_data, allocation_length)
}
scsi::VPD_SERIAL_NUMBER if !self.scsi_parameters.serial_number.is_empty() => {
self.handle_vpd_serial_number(external_data, allocation_length)
}
scsi::VPD_DEVICE_IDENTIFIERS => {
self.handle_vpd_device_identifiers(external_data, allocation_length)
}
scsi::VPD_BLOCK_LIMITS if self.scsi_parameters.support_unmap => {
self.handle_vpd_block_limits(external_data, allocation_length)
}
scsi::VPD_BLOCK_DEVICE_CHARACTERISTICS if self.scsi_parameters.support_unmap => {
self.handle_vpd_block_device_characteristics(external_data, allocation_length)
}
scsi::VPD_LOGICAL_BLOCK_PROVISIONING if self.scsi_parameters.support_unmap => self
.handle_vpd_logical_block_provisioning(
external_data,
allocation_length,
sector_count,
),
scsi::VPD_THIRD_PARTY_COPY if self.scsi_parameters.support_odx => {
self.handle_vpd_third_party_copy(external_data, allocation_length)
}
scsi::VPD_MSFT_VIRTUAL_DEVICE_PROPERTIES => {
self.handle_vpd_msft_virtual_device_properties(external_data, allocation_length)
}
scsi::VPD_MSFT_PAGING_EXTENT_PROPERTIES => {
self.handle_vpd_msft_paging_extent_properties(external_data, allocation_length)
}
n => Err(ScsiError::UnsupportedVpdPageCode(n)),
}
} else {
self.handle_no_vpd_page(external_data, allocation_length)
}
}
}