Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Use cpu_endian to check local endianness #637

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 10 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions keylime/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ edition = "2021"

[dependencies]
base64 = "0.21"
cpu-endian = "0.1.1"
hex = "0.4"
log = "0.4"
openssl = "0.10.15"
Expand Down
38 changes: 38 additions & 0 deletions keylime/src/endian.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
// SPDX-License-Identifier: Apache-2.0
// Copyright 2023 Keylime Authors

use cpu_endian;

pub trait LocalEndianness<T, const N: usize> {
fn local_endianness(t: T) -> [u8; N];
}

impl LocalEndianness<u32, 4> for u32 {
fn local_endianness(input: u32) -> [u8; 4] {
match cpu_endian::working() {
cpu_endian::Endian::Little => input.to_le_bytes(),
cpu_endian::Endian::Big => input.to_be_bytes(),
_ => input.to_le_bytes(),
}
}
}

impl LocalEndianness<u16, 2> for u16 {
fn local_endianness(input: u16) -> [u8; 2] {
match cpu_endian::working() {
cpu_endian::Endian::Little => input.to_le_bytes(),
cpu_endian::Endian::Big => input.to_be_bytes(),
_ => input.to_le_bytes(),
}
}
}

impl LocalEndianness<u8, 1> for u8 {
fn local_endianness(input: u8) -> [u8; 1] {
match cpu_endian::working() {
cpu_endian::Endian::Little => input.to_le_bytes(),
cpu_endian::Endian::Big => input.to_be_bytes(),
_ => input.to_le_bytes(),
}
}
}
14 changes: 8 additions & 6 deletions keylime/src/ima/entry.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
// https://www.kernel.org/doc/html/v5.12/security/IMA-templates.html

use crate::algorithms::HashAlgorithm;
use crate::endian::LocalEndianness;
use openssl::hash::MessageDigest;
use std::convert::{TryFrom, TryInto};
use std::io::{Error, ErrorKind, Result, Write};
Expand Down Expand Up @@ -107,12 +108,13 @@ impl TryFrom<&str> for Digest {
impl Encode for Digest {
fn encode(&self, writer: &mut dyn Write) -> Result<()> {
if self.algorithm == HashAlgorithm::Sha1 {
writer.write_all(&(self.value.len() as u32).to_le_bytes())?;
writer
.write_all(&u32::local_endianness(self.value.len() as u32))?;
writer.write_all(&self.value)?;
} else {
let algorithm = format!("{}", self.algorithm);
let total_len = algorithm.len() + 2 + self.value.len();
writer.write_all(&(total_len as u32).to_le_bytes())?;
writer.write_all(&u32::local_endianness(total_len as u32))?;
writer.write_all(algorithm.as_bytes())?;
writer.write_all(&[58u8, 0u8])?;
writer.write_all(&self.value)?;
Expand Down Expand Up @@ -147,7 +149,7 @@ const TCG_EVENT_NAME_LEN_MAX: usize = 255;
impl Encode for Name {
fn encode(&self, writer: &mut dyn Write) -> Result<()> {
let bytes = self.name.as_bytes();
writer.write_all(&((bytes.len() + 1) as u32).to_le_bytes())?;
writer.write_all(&u32::local_endianness((bytes.len() + 1) as u32))?;
writer.write_all(bytes)?;
writer.write_all(&[0u8])?; // NUL
Ok(())
Expand Down Expand Up @@ -194,7 +196,7 @@ impl TryFrom<&str> for Signature {

impl Encode for Signature {
fn encode(&self, writer: &mut dyn Write) -> Result<()> {
writer.write_all(&(self.value.len() as u32).to_le_bytes())?;
writer.write_all(&u32::local_endianness(self.value.len() as u32))?;
writer.write_all(&self.value)?;
Ok(())
}
Expand All @@ -218,7 +220,7 @@ impl TryFrom<&str> for Buffer {

impl Encode for Buffer {
fn encode(&self, writer: &mut dyn Write) -> Result<()> {
writer.write_all(&(self.value.len() as u32).to_le_bytes())?;
writer.write_all(&u32::local_endianness(self.value.len() as u32))?;
writer.write_all(&self.value)?;
Ok(())
}
Expand Down Expand Up @@ -347,7 +349,7 @@ impl Encode for ImaSig {
if let Some(signature) = &self.signature {
signature.encode(writer)?;
} else {
writer.write_all(&0u32.to_le_bytes())?;
writer.write_all(&u32::local_endianness(0u32))?;
}
Ok(())
}
Expand Down
1 change: 1 addition & 0 deletions keylime/src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
pub mod algorithms;
pub mod endian;
pub mod ima;
pub mod list_parser;
pub mod tpm;
Expand Down
13 changes: 7 additions & 6 deletions keylime/src/tpm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
// Copyright 2021 Keylime Authors

use crate::algorithms::{EncryptionAlgorithm, HashAlgorithm, SignAlgorithm};
use crate::endian::LocalEndianness;
use base64::{engine::general_purpose, Engine as _};
use log::*;
use std::convert::{TryFrom, TryInto};
Expand Down Expand Up @@ -357,10 +358,10 @@ assert_eq_size!(TPML_DIGEST, [u8; 532]);
// memory aligment.
fn serialize_pcrsel(pcr_selection: &TPML_PCR_SELECTION) -> Vec<u8> {
let mut output = Vec::with_capacity(TPML_PCR_SELECTION_SIZE);
output.extend(u32::to_le_bytes(pcr_selection.count));
output.extend(u32::local_endianness(pcr_selection.count));
for selection in pcr_selection.pcrSelections.iter() {
output.extend(selection.hash.to_le_bytes());
output.extend(selection.sizeofSelect.to_le_bytes());
output.extend(u16::local_endianness(selection.hash));
output.extend(u8::local_endianness(selection.sizeofSelect));
output.extend(selection.pcrSelect);
output.extend([0u8; 1]); // padding to keep the memory alignment
}
Expand All @@ -371,9 +372,9 @@ fn serialize_pcrsel(pcr_selection: &TPML_PCR_SELECTION) -> Vec<u8> {
// The serialization will adjust the data endianness as necessary.
fn serialize_digest(digest_list: &TPML_DIGEST) -> Vec<u8> {
let mut output = Vec::with_capacity(TPML_DIGEST_SIZE);
output.extend(u32::to_le_bytes(digest_list.count));
output.extend(u32::local_endianness(digest_list.count));
for digest in digest_list.digests.iter() {
output.extend(digest.size.to_le_bytes());
output.extend(u16::local_endianness(digest.size));
output.extend(digest.buffer);
}
output
Expand Down Expand Up @@ -410,7 +411,7 @@ fn pcrdata_to_vec(
Vec::with_capacity(pcrsel_vec.len() + 4 + digest_vec.len());

data_vec.extend(&pcrsel_vec);
data_vec.extend(num_tpml_digests.to_le_bytes());
data_vec.extend(u32::local_endianness(num_tpml_digests));
data_vec.extend(&digest_vec);

data_vec
Expand Down