mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2024-11-09 21:30:07 +00:00
dd539df633
The generic version does not work in this case as it the size of the the &[u8] is not the size of the slice's contents but how much memory the slice object itself takes up. Signed-off-by: Rob Bradford <robert.bradford@intel.com>
138 lines
3.5 KiB
Rust
138 lines
3.5 KiB
Rust
// Copyright © 2019 Intel Corporation
|
|
//
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
//
|
|
|
|
#[repr(packed)]
|
|
pub struct GenericAddress {
|
|
pub address_space_id: u8,
|
|
pub register_bit_width: u8,
|
|
pub register_bit_offset: u8,
|
|
pub access_size: u8,
|
|
pub address: u64,
|
|
}
|
|
|
|
impl GenericAddress {
|
|
pub fn io_port_address(address: u16) -> Self {
|
|
GenericAddress {
|
|
address_space_id: 1,
|
|
register_bit_width: 8,
|
|
register_bit_offset: 0,
|
|
access_size: 1,
|
|
address: u64::from(address),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct SDT {
|
|
data: Vec<u8>,
|
|
}
|
|
|
|
#[allow(clippy::len_without_is_empty)]
|
|
impl SDT {
|
|
pub fn new(
|
|
signature: [u8; 4],
|
|
length: u32,
|
|
revision: u8,
|
|
oem_id: [u8; 6],
|
|
oem_table: [u8; 8],
|
|
oem_revision: u32,
|
|
) -> Self {
|
|
assert!(length >= 36);
|
|
|
|
let mut data = Vec::with_capacity(length as usize);
|
|
data.extend_from_slice(&signature);
|
|
data.extend_from_slice(&length.to_le_bytes());
|
|
data.push(revision);
|
|
data.push(0); // checksum
|
|
data.extend_from_slice(&oem_id);
|
|
data.extend_from_slice(&oem_table);
|
|
data.extend_from_slice(&oem_revision.to_le_bytes());
|
|
data.extend_from_slice(b"CLDH");
|
|
data.extend_from_slice(&0u32.to_le_bytes());
|
|
assert_eq!(data.len(), 36);
|
|
|
|
data.resize(length as usize, 0);
|
|
let mut sdt = SDT { data };
|
|
|
|
sdt.update_checksum();
|
|
sdt
|
|
}
|
|
|
|
pub fn update_checksum(&mut self) {
|
|
self.data[9] = 0;
|
|
let checksum = super::generate_checksum(self.data.as_slice());
|
|
self.data[9] = checksum
|
|
}
|
|
|
|
pub fn as_slice(&self) -> &[u8] {
|
|
&self.data.as_slice()
|
|
}
|
|
|
|
pub fn append<T>(&mut self, value: T) {
|
|
let orig_length = self.data.len();
|
|
let new_length = orig_length + std::mem::size_of::<T>();
|
|
self.data.resize(new_length, 0);
|
|
self.write_u32(4, new_length as u32);
|
|
self.write(orig_length, value);
|
|
}
|
|
|
|
pub fn append_slice(&mut self, data: &[u8]) {
|
|
let orig_length = self.data.len();
|
|
let new_length = orig_length + data.len();
|
|
self.write_u32(4, new_length as u32);
|
|
self.data.extend_from_slice(data);
|
|
self.update_checksum();
|
|
}
|
|
|
|
/// Write a value at the given offset
|
|
pub fn write<T>(&mut self, offset: usize, value: T) {
|
|
assert!((offset + (std::mem::size_of::<T>() - 1)) < self.data.len());
|
|
unsafe {
|
|
*(((self.data.as_mut_ptr() as usize) + offset) as *mut T) = value;
|
|
}
|
|
self.update_checksum();
|
|
}
|
|
|
|
pub fn write_u8(&mut self, offset: usize, val: u8) {
|
|
self.write(offset, val);
|
|
}
|
|
|
|
pub fn write_u16(&mut self, offset: usize, val: u16) {
|
|
self.write(offset, val);
|
|
}
|
|
|
|
pub fn write_u32(&mut self, offset: usize, val: u32) {
|
|
self.write(offset, val);
|
|
}
|
|
|
|
pub fn write_u64(&mut self, offset: usize, val: u64) {
|
|
self.write(offset, val);
|
|
}
|
|
|
|
pub fn len(&self) -> usize {
|
|
self.data.len()
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::SDT;
|
|
|
|
#[test]
|
|
fn test_sdt() {
|
|
let mut sdt = SDT::new(*b"TEST", 40, 1, *b"CLOUDH", *b"TESTTEST", 1);
|
|
let sum: u8 = sdt
|
|
.as_slice()
|
|
.iter()
|
|
.fold(0u8, |acc, x| acc.wrapping_add(*x));
|
|
assert_eq!(sum, 0);
|
|
sdt.write_u32(36, 0x12345678);
|
|
let sum: u8 = sdt
|
|
.as_slice()
|
|
.iter()
|
|
.fold(0u8, |acc, x| acc.wrapping_add(*x));
|
|
assert_eq!(sum, 0);
|
|
}
|
|
}
|