Эх сурвалжийг харах

rsdp: add the validate_checksum to rsdp

Add the validate_checksum method to the implementation of the RsdpV1Tag
and RsdpV2Tag.
Dan Robertson 4 жил өмнө
parent
commit
536e262f0f
2 өөрчлөгдсөн 22 нэмэгдсэн , 11 устгасан
  1. 1 1
      src/lib.rs
  2. 21 10
      src/rsdp.rs

+ 1 - 1
src/lib.rs

@@ -1078,7 +1078,7 @@ mod tests {
         // Test the RSDP tag
         let rsdp_old = bi.rsdp_v1_tag().unwrap();
         assert_eq!("RSD PTR ", rsdp_old.signature().unwrap());
-        assert_eq!(89, rsdp_old.checksum());
+        assert!(rsdp_old.checksum_is_valid());
         assert_eq!("BOCHS ", rsdp_old.oem_id().unwrap());
         assert_eq!(0, rsdp_old.revision());
         assert_eq!(0x7FE18DC, rsdp_old.rsdt_address());

+ 21 - 10
src/rsdp.rs

@@ -6,8 +6,11 @@
 /// Even though the bootloader should give the address of the real RSDP/XSDT, the checksum and
 /// signature should be manually verified.
 
+use core::slice;
 use core::str;
 
+const RSDPV1_LENGTH: usize = 20;
+
 /// This tag contains a copy of RSDP as defined per ACPI 1.0 specification. 
 #[derive(Clone, Copy, Debug)]
 #[repr(C, packed)]
@@ -29,11 +32,15 @@ impl RsdpV1Tag {
         str::from_utf8(&self.signature).ok()
     }
 
-    /// The value to add to all the other bytes (of the Version 1.0 table) to calculate the Checksum of the table.
-    ///
-    /// If this value added to all the others and casted to byte isn't equal to 0, the table must be ignored. 
-    pub fn checksum(&self) -> u8 {
-        self.checksum
+    /// Validation of the RSDPv1 checksum
+    pub fn checksum_is_valid(&self) -> bool {
+        let bytes = unsafe {
+            slice::from_raw_parts(
+                self as *const _ as *const u8,
+                RSDPV1_LENGTH + 8
+            )
+        };
+        bytes[8..].iter().fold(0u8, |acc, val| acc.wrapping_add(*val)) == 0
     }
 
     /// An OEM-supplied string that identifies the OEM. 
@@ -77,11 +84,15 @@ impl RsdpV2Tag {
         str::from_utf8(&self.signature).ok()
     }
 
-    /// The value to add to all the other bytes (of the Version 1.0 table) to calculate the Checksum of the table.
-    ///
-    /// If this value added to all the others and casted to byte isn't equal to 0, the table must be ignored. 
-    pub fn checksum(&self) -> u8 {
-        self.checksum
+    /// Validation of the RSDPv2 extended checksum
+    pub fn checksum_is_valid(&self) -> bool {
+        let bytes = unsafe {
+            slice::from_raw_parts(
+                self as *const _ as *const u8,
+                self.length as usize + 8
+            )
+        };
+        bytes[8..].iter().fold(0u8, |acc, val| acc.wrapping_add(*val)) == 0
     }
 
     /// An OEM-supplied string that identifies the OEM.