Skip to content

Commit

Permalink
Verifier: IBM SE to fix review comments
Browse files Browse the repository at this point in the history
Signed-off-by: Qi Feng Huo <[email protected]>
  • Loading branch information
Qi Feng Huo committed May 31, 2024
1 parent 7a82caf commit a889145
Show file tree
Hide file tree
Showing 9 changed files with 162 additions and 140 deletions.
5 changes: 2 additions & 3 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -37,8 +37,8 @@ prost = "0.11.0"
regorus = { version = "0.1.5", default-features = false, features = ["regex", "base64", "time"] }
rstest = "0.18.1"
serde = { version = "1.0", features = ["derive"] }
serde_with = { version = "1.11.0", features = ["base64"] }
serde_json = "1.0.89"
serde_with = { version = "1.11.0", features = ["base64"] }
serial_test = "0.9.0"
sha2 = "0.10"
shadow-rs = "0.19.0"
Expand All @@ -47,5 +47,4 @@ thiserror = "1.0"
tokio = { version = "1.23.0", features = ["full"] }
tempfile = "3.4.0"
tonic = "0.8.1"
tonic-build = "0.8.0"
s390_pv = "0.10.0"
tonic-build = "0.8.0"
21 changes: 18 additions & 3 deletions attestation-service/attestation-service/src/bin/grpc/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -205,14 +205,29 @@ impl AttestationService for Arc<RwLock<AttestationServer>> {
request: Request<ChallengeRequest>,
) -> Result<Response<ChallengeResponse>, Status> {
let request: ChallengeRequest = request.into_inner();
let tee = to_kbs_tee(&request.tee)
.map_err(|e| Status::aborted(format!("parse TEE type: {e}")))?;
info!("get_attestation_challenge API called.");
debug!("get_attestation_challenge: {request:#?}");

let inner_tee = request
.inner
.get("tee")
.map_or(Err(Status::aborted("Error parse inner_tee tee")), Ok)?;
let tee_params = request
.inner
.get("tee_params")
.map_or(Err(Status::aborted("Error parse inner_tee tee_params")), Ok)?;
let tee = to_kbs_tee(&inner_tee).map_or(
Err(Status::aborted(format!(
"Error parse TEE type: {inner_tee}"
))),
Ok,
)?;

let attestation_challenge = self
.read()
.await
.attestation_service
.generate_supplemental_challenge(tee, request.tee_params.clone())
.generate_supplemental_challenge(tee, tee_params.clone())
.await
.map_err(|e| Status::aborted(format!("Challenge: {e:?}")))?;

Expand Down
29 changes: 23 additions & 6 deletions attestation-service/attestation-service/src/bin/restful/mod.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
use std::sync::Arc;
use std::{collections::HashMap, sync::Arc};

use actix_web::{body::BoxBody, web, HttpRequest, HttpResponse, ResponseError};
use anyhow::{bail, Context};
use anyhow::{anyhow, bail, Context};
use attestation_service::{AttestationService, HashAlgorithm};
use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine};
use kbs_types::Tee;
Expand Down Expand Up @@ -46,8 +46,10 @@ pub struct AttestationRequest {

#[derive(Debug, Serialize, Deserialize)]
pub struct ChallengeRequest {
tee: String,
tee_params: String,
// tee: String,
// tee_params: String,
#[serde(flatten)]
inner: HashMap<String, String>,
}

#[derive(Debug, Serialize, Deserialize)]
Expand Down Expand Up @@ -191,11 +193,26 @@ pub async fn get_challenge(
request: web::Json<ChallengeRequest>,
cocoas: web::Data<Arc<RwLock<AttestationService>>>,
) -> Result<HttpResponse> {
let tee = to_tee(&request.tee)?;
info!("get_challenge API called.");
let request: ChallengeRequest = request.into_inner();

debug!("get_challenge: {request:#?}");
let inner_tee = request
.inner
.get("tee")
.as_ref()
.map(|s| s.as_str())
.ok_or(anyhow!("Failed to get inner tee"))?;
let tee_params = request
.inner
.get("tee_params")
.ok_or(anyhow!("Failed to get inner tee_params"))?;

let tee = to_tee(inner_tee)?;
let challenge = cocoas
.read()
.await
.generate_supplemental_challenge(tee, request.tee_params.clone())
.generate_supplemental_challenge(tee, tee_params.to_string())
.await
.context("generate challenge")?;
Ok(HttpResponse::Ok().body(challenge))
Expand Down
7 changes: 4 additions & 3 deletions attestation-service/docs/parsed_claims.md
Original file line number Diff line number Diff line change
Expand Up @@ -96,10 +96,11 @@ Note: The TD Report and TD Quote are fetched during early boot in this TEE. Kern
## IBM Secure Execution (SE)
- `se.version`: The version this quote structure.
- `se.cuid`: The config uid.
- `se.hdr.tag`: SE Header Tag (seht)
- `se.image.phkh`: SE image Public host key hash
- `se.attestation.phkh`: SE attestation Public host key hash
- `se.hdr.tag`: SE header tag (seht)
- `se.image.phkh`: SE image public host key hash
- `se.attestation.phkh`: SE attestation public host key hash
- `se.user_data`: Custom attestation key owner data.

## AMD SEV-SNP

- `snp.measurement` Launch Digest covering initial guest memory
Expand Down
19 changes: 3 additions & 16 deletions attestation-service/protos/attestation.proto
Original file line number Diff line number Diff line change
Expand Up @@ -2,19 +2,6 @@ syntax = "proto3";

package attestation;

enum Tee {
SEV = 0;
SGX = 1;
SNP = 2;
TDX = 3;
Sample = 4;
AzSnpVtpm = 5;
CSV = 6;
CCA = 7;
AzTdxVtpm = 8;
Se = 9;
}

message AttestationRequest {
// TEE enum. Specify the evidence type
string tee = 1;
Expand Down Expand Up @@ -91,9 +78,9 @@ message SetPolicyRequest {
message SetPolicyResponse {}

message ChallengeRequest {
// TEE enum. Specify the evidence type
string tee = 1;
string tee_params = 2;
// string tee = 1;
// string tee_params = 2;
map<string, string> inner = 1;
}
message ChallengeResponse {
string attestation_challenge = 1;
Expand Down
1 change: 0 additions & 1 deletion attestation-service/verifier/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -174,7 +174,6 @@ pub trait Verifier {
/// generate the evidence
///
/// A optional `tee_parameters` comes from the attester side as the input.
async fn generate_supplemental_challenge(
&self,
_tee_parameters: String,
Expand Down
108 changes: 37 additions & 71 deletions attestation-service/verifier/src/se/ibmse.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@
// SPDX-License-Identifier: Apache-2.0
//

use anyhow::{anyhow, Result};
use log::{debug, warn};
use anyhow::{anyhow, bail, Result};
use log::{debug, info, warn};
use openssl::{
encrypt::{Decrypter, Encrypter},
pkey::{PKey, Private, Public},
Expand All @@ -24,28 +24,28 @@ use serde_json;
use serde_with::{base64::Base64, serde_as};
use std::{fs::File, io::Read};

fn encrypt_measurement_key(key: &[u8], rsa_public_key: &PKey<Public>) -> Vec<u8> {
let mut encrypter = Encrypter::new(rsa_public_key).unwrap();
encrypter.set_rsa_padding(Padding::PKCS1).unwrap();
fn encrypt_measurement_key(key: &[u8], rsa_public_key: &PKey<Public>) -> Result<Vec<u8>> {
let mut encrypter = Encrypter::new(rsa_public_key)?;
encrypter.set_rsa_padding(Padding::PKCS1)?;

let buffer_len = encrypter.encrypt_len(key).unwrap();
let buffer_len = encrypter.encrypt_len(key)?;
let mut encrypted_hmac_key = vec![0; buffer_len];
let len = encrypter.encrypt(key, &mut encrypted_hmac_key).unwrap();
let len = encrypter.encrypt(key, &mut encrypted_hmac_key)?;
encrypted_hmac_key.truncate(len);

encrypted_hmac_key
Ok(encrypted_hmac_key)
}

fn decrypt_measurement_key(key: &[u8], rsa_private_key: &PKey<Private>) -> Vec<u8> {
let mut decrypter = Decrypter::new(rsa_private_key).unwrap();
decrypter.set_rsa_padding(Padding::PKCS1).unwrap();
fn decrypt_measurement_key(key: &[u8], rsa_private_key: &PKey<Private>) -> Result<Vec<u8>> {
let mut decrypter = Decrypter::new(rsa_private_key)?;
decrypter.set_rsa_padding(Padding::PKCS1)?;

let buffer_len = decrypter.decrypt_len(key).unwrap();
let buffer_len = decrypter.decrypt_len(key)?;
let mut decrypted_hmac_key = vec![0; buffer_len];
let decrypted_len = decrypter.decrypt(key, &mut decrypted_hmac_key).unwrap();
let decrypted_len = decrypter.decrypt(key, &mut decrypted_hmac_key)?;
decrypted_hmac_key.truncate(decrypted_len);

decrypted_hmac_key
Ok(decrypted_hmac_key)
}

#[serde_as]
Expand Down Expand Up @@ -87,14 +87,6 @@ pub struct SeAttestationRequest {
}

impl SeAttestationRequest {
pub fn from_slice(request: &[u8]) -> Result<Self> {
Ok(serde_json::from_slice(request).unwrap())
}

pub fn from_string(request: &str) -> Result<Self> {
Ok(serde_json::from_str(request).unwrap())
}

pub fn create(
hkds: &Vec<String>,
certs: &Vec<String>,
Expand All @@ -121,8 +113,9 @@ impl SeAttestationRequest {
if certs.len() != 1 {
warn!("The host key document in '{hkd}' contains more than one certificate!")
}
// Panic: len is == 1 -> unwrap will succeed/not panic
let c = certs.first().unwrap();
let c = certs
.first()
.ok_or(anyhow!("File does not contain a X509 certificate"))?;
verifier.verify(c)?;
arcb.add_hostkey(c.public_key()?);
}
Expand All @@ -137,9 +130,13 @@ impl SeAttestationRequest {

let conf_data = arcb.confidential_data();
let encr_measurement_key =
encrypt_measurement_key(conf_data.measurement_key(), rsa_public_key);
let encr_request_nonce =
encrypt_measurement_key(conf_data.nonce().clone().unwrap().value(), rsa_public_key);
encrypt_measurement_key(conf_data.measurement_key(), rsa_public_key)?;
let binding = conf_data
.nonce()
.clone()
.ok_or(anyhow!("Failed to get nonce binding"))?;
let nonce = binding.value();
let encr_request_nonce = encrypt_measurement_key(nonce, rsa_public_key)?;

Ok(Self {
request_blob,
Expand Down Expand Up @@ -172,34 +169,6 @@ pub struct SeAttestationResponse {
}

impl SeAttestationResponse {
pub fn from_slice(response: &[u8]) -> Result<Self> {
Ok(serde_json::from_slice(response).unwrap())
}

pub fn from_string(request: &str) -> Result<Self> {
Ok(serde_json::from_str(request).unwrap())
}

pub fn create(
measurement: &[u8],
additional_data: &[u8],
user_data: &[u8],
cuid: &ConfigUid,
encr_measurement_key: &[u8],
encr_request_nonce: &[u8],
image_hdr_tags: &BootHdrTags,
) -> Result<Self> {
Ok(Self {
measurement: measurement.to_vec(),
additional_data: additional_data.to_vec(),
user_data: user_data.to_vec(),
cuid: *cuid,
encr_measurement_key: encr_measurement_key.to_vec(),
encr_request_nonce: encr_request_nonce.to_vec(),
image_hdr_tags: *image_hdr_tags,
})
}

pub fn verify(&self, priv_key_file: &str) -> Result<SeAttestationClaims> {
let mut file = File::open(priv_key_file)?;
let mut contents = vec![];
Expand All @@ -208,19 +177,16 @@ impl SeAttestationResponse {
let rsa = Rsa::private_key_from_pem(&contents)?;
let rsa_private_key = &PKey::from_rsa(rsa)?;

let meas_key = decrypt_measurement_key(&self.encr_measurement_key, rsa_private_key);
let nonce = decrypt_measurement_key(&self.encr_request_nonce, rsa_private_key);
let meas_key = decrypt_measurement_key(&self.encr_measurement_key, rsa_private_key)?;
let nonce = decrypt_measurement_key(&self.encr_request_nonce, rsa_private_key)?;

if nonce.len() != 16 {
return Err(anyhow!("The nonce vector must have exactly 16 elements."));
bail!("The nonce vector must have exactly 16 elements.");
}
let boxed_slice: Box<[u8]> = nonce.into_boxed_slice();
let boxed_array: Box<[u8; 16]> = match boxed_slice.try_into() {
Ok(ba) => ba,
Err(_) => return Err(anyhow!("Failed to convert nonce from Vec<u8> to [u8; 16].")),
};
let nonce_array: [u8; 16] = *boxed_array;

let nonce_array: [u8; 16] = nonce
.try_into()
.map_err(|_| anyhow!("Failed to convert nonce from Vec<u8> to [u8; 16]."))?;
let meas_key = &PKey::hmac(&meas_key)?;
let items = AttestationItems::new(
&self.image_hdr_tags,
Expand All @@ -231,14 +197,13 @@ impl SeAttestationResponse {
);

let measurement =
AttestationMeasurement::calculate(items, AttestationMeasAlg::HmacSha512, meas_key)
.unwrap();
AttestationMeasurement::calculate(items, AttestationMeasAlg::HmacSha512, meas_key)?;

if !measurement.eq_secure(&self.measurement) {
debug!("Recieved: {:?}", self.measurement);
debug!("Calculated: {:?}", measurement.as_ref());
warn!("Attestation measurement verification failed. Calculated and received attestation measurement are not equal.");
return Err(anyhow!("Failed to verify the measurement!"));
bail!("Failed to verify the measurement!");
}

// let userdata = serde_json::from_slice(&self.user_data)?;
Expand Down Expand Up @@ -279,14 +244,15 @@ pub fn create(
se_img_hdr: &str,
pub_key_file: &str,
) -> Result<String> {
info!("IBM SE create API called.");
debug!("hkds: {:#?}", hkds);
debug!("certs: {:#?}", certs);
debug!("ca: {:#?}", ca);
debug!("se_img_hdr: {:#?}", se_img_hdr);
debug!("pub_key_file: {:#?}", pub_key_file);

let mut hdr_file = open_file(se_img_hdr)?;
let mut image_hdr_tags = BootHdrTags::from_se_image(&mut hdr_file).unwrap();
let mut image_hdr_tags = BootHdrTags::from_se_image(&mut hdr_file)?;
let root_ca = Some(ca);

let se_request = SeAttestationRequest::create(
Expand All @@ -296,8 +262,7 @@ pub fn create(
root_ca,
&mut image_hdr_tags,
pub_key_file,
)
.unwrap();
)?;

let challenge = serde_json::to_string(&se_request)?;
debug!("challenge json: {:#?}", challenge);
Expand All @@ -306,10 +271,11 @@ pub fn create(
}

pub fn verify(response: &[u8], priv_key_file: &str) -> Result<SeAttestationClaims> {
info!("IBM SE verify API called.");
// response is serialized SeAttestationResponse String bytes
let response_str = std::str::from_utf8(response)?;
debug!("SeAttestationResponse json: {:#?}", response_str);
let se_response = SeAttestationResponse::from_string(response_str)?;
let se_response: SeAttestationResponse = serde_json::from_str(response_str)?;

let claims = se_response.verify(priv_key_file)?;
debug!("claims json: {:#?}", claims);
Expand Down
Loading

0 comments on commit a889145

Please sign in to comment.