Skip to content

Commit

Permalink
Merge pull request #555 from parallaxsecond/get-time-7
Browse files Browse the repository at this point in the history
Porting GetTime to v7
  • Loading branch information
ionut-arm authored Dec 3, 2024
2 parents 22b5862 + 1a8501a commit a7df622
Show file tree
Hide file tree
Showing 2 changed files with 157 additions and 2 deletions.
118 changes: 116 additions & 2 deletions tss-esapi/src/context/tpm_commands/attestation_commands.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
use crate::{
handles::{KeyHandle, ObjectHandle},
structures::{Attest, AttestBuffer, Data, PcrSelectionList, Signature, SignatureScheme},
tss2_esys::{Esys_Certify, Esys_Quote},
tss2_esys::{Esys_Certify, Esys_GetTime, Esys_Quote},
Context, Error, Result,
};
use log::error;
Expand Down Expand Up @@ -197,8 +197,122 @@ impl Context {
}
}

/// Get the current time and clock from the TPM
///
/// # Arguments
/// * `signing_key_handle` - Handle of the key used to sign the attestation buffer
/// * `qualifying_data` - Qualifying data
/// * `signing_scheme` - Signing scheme to use if the scheme for `signing_key_handle` is `Null`.
///
/// The `signing_key_handle` must be usable for signing.
///
/// If `signing_key_handle` has the Restricted attribute set to `true` then `signing_scheme` must be
/// [SignatureScheme::Null].
///
/// # Returns
/// The command returns a tuple consisting of:
/// * `attest_data` - TPM-generated attestation data.
/// * `signature` - Signature for the attestation data.
///
/// # Errors
/// * if the qualifying data provided is too long, a `WrongParamSize` wrapper error will be returned
///
/// # Examples
///
/// ```rust
/// # use tss_esapi::{Context, TctiNameConf};
/// # use std::convert::TryFrom;
/// # use tss_esapi::{
/// # abstraction::cipher::Cipher,
/// # interface_types::{
/// # algorithm::{HashingAlgorithm, RsaSchemeAlgorithm},
/// # key_bits::RsaKeyBits,
/// # resource_handles::Hierarchy,
/// # },
/// # structures::{
/// # RsaExponent, RsaScheme,
/// # },
/// # utils::{create_unrestricted_signing_rsa_public, create_restricted_decryption_rsa_public},
/// # };
/// use std::convert::TryInto;
/// use tss_esapi::{
/// structures::{Data, SignatureScheme},
/// interface_types::session_handles::AuthSession,
/// };
/// # let mut context =
/// # Context::new(
/// # TctiNameConf::from_environment_variable().expect("Failed to get TCTI"),
/// # ).expect("Failed to create Context");
/// let qualifying_data = vec![0xff; 16];
/// # let signing_key_pub = create_unrestricted_signing_rsa_public(
/// # RsaScheme::create(RsaSchemeAlgorithm::RsaSsa, Some(HashingAlgorithm::Sha256))
/// # .expect("Failed to create RSA scheme"),
/// # RsaKeyBits::Rsa2048,
/// # RsaExponent::default(),
/// # )
/// # .expect("Failed to create an unrestricted signing rsa public structure");
/// # let sign_key_handle = context
/// # .execute_with_nullauth_session(|ctx| {
/// # ctx.create_primary(Hierarchy::Owner, signing_key_pub, None, None, None, None)
/// # })
/// # .unwrap()
/// # .key_handle;
/// let (attest, signature) = context
/// .execute_with_sessions(
/// (
/// Some(AuthSession::Password),
/// Some(AuthSession::Password),
/// None,
/// ),
/// |ctx| {
/// ctx.get_time(
/// sign_key_handle,
/// Data::try_from(qualifying_data).unwrap(),
/// SignatureScheme::Null,
/// )
/// },
/// )
/// .expect("Failed to get tpm time");
/// ```
pub fn get_time(
&mut self,
signing_key_handle: KeyHandle,
qualifying_data: Data,
signing_scheme: SignatureScheme,
) -> Result<(Attest, Signature)> {
let mut timeinfo_ptr = null_mut();
let mut signature_ptr = null_mut();
let ret = unsafe {
Esys_GetTime(
self.mut_context(),
ObjectHandle::Endorsement.into(),
signing_key_handle.into(),
self.required_session_1()?,
self.required_session_2()?,
self.optional_session_3(),
&qualifying_data.into(),
&signing_scheme.into(),
&mut timeinfo_ptr,
&mut signature_ptr,
)
};

let ret = Error::from_tss_rc(ret);

if ret.is_success() {
let timeinfo = Context::ffi_data_to_owned(timeinfo_ptr);
let signature = Context::ffi_data_to_owned(signature_ptr);
Ok((
Attest::try_from(AttestBuffer::try_from(timeinfo)?)?,
Signature::try_from(signature)?,
))
} else {
error!("Error in Get Time: {}", ret);
Err(ret)
}
}

// Missing function: GetSessionAuditDigest
// Missing function: GestCommandAuditDigest
// Missing function: GetTime
// Missing function: CertifyX509
}
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,47 @@ mod test_quote {
}
}

#[test]
fn time() {
let mut context = create_ctx_with_session();
// No qualifying data
let qualifying_data = vec![0xff; 16];

let key_handle = context
.create_primary(Hierarchy::Owner, signing_key_pub(), None, None, None, None)
.unwrap()
.key_handle;
let (attest, _signature) = context
.execute_with_sessions(
(
Some(AuthSession::Password),
Some(AuthSession::Password),
None,
),
|ctx| {
ctx.get_time(
key_handle,
Data::try_from(qualifying_data).unwrap(),
SignatureScheme::Null,
)
},
)
.expect("Failed to get time");

assert_eq!(
AttestationType::Time,
attest.attestation_type(),
"Attestation type of the returned value is not indicating Time"
);

match attest.attested() {
AttestInfo::Time { info: _ } => {}
_ => {
panic!("Attested did not contain the expected variant.")
}
}
}

#[test]
fn certify() {
let mut context = create_ctx_with_session();
Expand Down

0 comments on commit a7df622

Please sign in to comment.