1use anyhow::{bail, Context, Error};
9
10#[derive(Clone, Copy, PartialEq)]
11pub enum Policy {
12 Null,
13 TeeRequired,
14 TeeTransitional,
15 TeeOpportunistic,
16}
17
18impl TryFrom<String> for Policy {
19 type Error = Error;
20
21 fn try_from(value: String) -> Result<Self, Self::Error> {
22 match value.as_ref() {
23 "null" => Ok(Policy::Null),
24 "tee" => Ok(Policy::TeeRequired),
25 "tee-transitional" => Ok(Policy::TeeTransitional),
26 "tee-opportunistic" => Ok(Policy::TeeOpportunistic),
27 p => bail!("unrecognized key source policy: '{p}'"),
28 }
29 }
30}
31
32pub async fn get_policy() -> Result<Policy, Error> {
34 let policy = fuchsia_fs::file::read_in_namespace_to_string("/boot/config/zxcrypt").await;
35 let policy = match policy {
36 Ok(policy) => policy,
37 Err(_) => fuchsia_fs::file::read_in_namespace_to_string("/pkg/config/zxcrypt").await?,
39 };
40 policy.try_into()
41}
42
43#[derive(Debug)]
44pub enum KeySource {
45 Null,
46 Tee,
47}
48
49pub enum KeyConsumer {
52 Fxfs,
57 Zxcrypt,
59}
60
61impl KeySource {
62 pub async fn get_key(&self, consumer: KeyConsumer) -> Result<Vec<u8>, Error> {
63 match self {
64 KeySource::Null => match consumer {
65 KeyConsumer::Fxfs => {
66 let mut key = b"zxcrypt".to_vec();
67 key.resize(16, 0);
68 Ok(key)
69 }
70 KeyConsumer::Zxcrypt => Ok(vec![0u8; 32]),
71 },
72 KeySource::Tee => {
73 kms_stateless::get_hardware_derived_key(kms_stateless::KeyInfo::new_zxcrypt())
77 .await
78 .context("failed to get hardware key")
79 }
80 }
81 }
82}
83
84pub fn format_sources(policy: Policy) -> Vec<KeySource> {
86 match policy {
87 Policy::Null => vec![KeySource::Null],
88 Policy::TeeRequired => vec![KeySource::Tee],
89 Policy::TeeTransitional => vec![KeySource::Tee],
90 Policy::TeeOpportunistic => vec![KeySource::Tee, KeySource::Null],
91 }
92}
93
94pub fn unseal_sources(policy: Policy) -> Vec<KeySource> {
96 match policy {
97 Policy::Null => vec![KeySource::Null],
98 Policy::TeeRequired => vec![KeySource::Tee],
99 Policy::TeeTransitional => vec![KeySource::Tee, KeySource::Null],
100 Policy::TeeOpportunistic => vec![KeySource::Tee, KeySource::Null],
101 }
102}