1use crate::common::CommonArch;
7use crate::common::CommonTriple;
8use flowey::node::prelude::*;
9use flowey_lib_common::run_cargo_build::CargoFeatureSet;
10use std::collections::BTreeMap;
11use std::collections::BTreeSet;
12
13#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
14pub enum OpenvmmHclFeature {
15 Gdb,
16 MiSecure,
17 Tpm,
18 LocalOnlyCustom(String),
19}
20
21#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
22pub enum OpenvmmHclBuildProfile {
23 Debug,
24 Release,
25 OpenvmmHclShip,
26}
27
28#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
29pub enum MaxTraceLevel {
30 Trace,
31 Debug,
32 Info,
33 Warn,
34 Error,
35 Off,
36}
37
38impl MaxTraceLevel {
39 pub fn features(&self) -> Vec<String> {
40 let name = match self {
41 MaxTraceLevel::Trace => return Vec::new(),
42 MaxTraceLevel::Debug => "debug",
43 MaxTraceLevel::Info => "info",
44 MaxTraceLevel::Warn => "warn",
45 MaxTraceLevel::Error => "error",
46 MaxTraceLevel::Off => "off",
47 };
48 vec![
51 format!("tracing/max_level_{}", name),
52 format!("tracing/release_max_level_{}", name),
53 ]
54 }
55}
56
57#[derive(Serialize, Deserialize)]
58pub struct OpenvmmHclOutput {
59 pub bin: PathBuf,
60 pub dbg: Option<PathBuf>,
61}
62
63#[derive(Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)]
64pub struct OpenvmmHclBuildParams {
65 pub target: CommonTriple,
66 pub profile: OpenvmmHclBuildProfile,
67 pub features: BTreeSet<OpenvmmHclFeature>,
68 pub no_split_dbg_info: bool,
69 pub max_trace_level: MaxTraceLevel,
70}
71
72flowey_request! {
73 pub struct Request {
74 pub build_params: OpenvmmHclBuildParams,
75 pub openvmm_hcl_output: WriteVar<OpenvmmHclOutput>,
76 }
77}
78
79new_flow_node!(struct Node);
80
81impl FlowNode for Node {
82 type Request = Request;
83
84 fn imports(ctx: &mut ImportCtx<'_>) {
85 ctx.import::<crate::run_cargo_build::Node>();
86 ctx.import::<crate::init_openvmm_magicpath_openhcl_sysroot::Node>();
87 }
88
89 fn emit(requests: Vec<Self::Request>, ctx: &mut NodeCtx<'_>) -> anyhow::Result<()> {
90 let requests = requests
92 .into_iter()
93 .fold(BTreeMap::<_, Vec<_>>::new(), |mut m, r| {
94 let Request {
95 build_params,
96 openvmm_hcl_output,
97 } = r;
98 m.entry(build_params).or_default().push(openvmm_hcl_output);
99 m
100 });
101
102 for (
105 OpenvmmHclBuildParams {
106 target,
107 profile,
108 features,
109 no_split_dbg_info,
110 max_trace_level,
111 },
112 outvars,
113 ) in requests
114 {
115 let mut pre_build_deps = Vec::new();
116
117 let target = target.as_triple();
118
119 let arch = CommonArch::from_triple(&target)
120 .with_context(|| format!("cannot build openvmm_hcl on {}", target.architecture))?;
121
122 let openhcl_deps_path = ctx
123 .reqv(|v| crate::init_openvmm_magicpath_openhcl_sysroot::Request { arch, path: v });
124
125 pre_build_deps.push(openhcl_deps_path.into_side_effect());
127
128 let mut features = features
129 .into_iter()
130 .map(|f| match f {
131 OpenvmmHclFeature::Gdb => "gdb".into(),
132 OpenvmmHclFeature::MiSecure => "mi-secure".into(),
133 OpenvmmHclFeature::Tpm => "tpm".into(),
134 OpenvmmHclFeature::LocalOnlyCustom(s) => s,
135 })
136 .collect::<Vec<String>>();
137
138 features.extend(max_trace_level.features());
139
140 let output = ctx.reqv(|v| crate::run_cargo_build::Request {
141 crate_name: "openvmm_hcl".into(),
142 out_name: "openvmm_hcl".into(),
143 crate_type: flowey_lib_common::run_cargo_build::CargoCrateType::Bin,
144 profile: match profile {
145 OpenvmmHclBuildProfile::Debug => crate::run_cargo_build::BuildProfile::Debug,
146 OpenvmmHclBuildProfile::Release => {
147 crate::run_cargo_build::BuildProfile::Release
148 }
149 OpenvmmHclBuildProfile::OpenvmmHclShip => {
150 crate::run_cargo_build::BuildProfile::UnderhillShip
151 }
152 },
153 features: CargoFeatureSet::Specific(features),
154 target,
155 no_split_dbg_info,
156 extra_env: None,
157 pre_build_deps,
158 output: v,
159 });
160
161 ctx.emit_minor_rust_step("report built openvmm_hcl", |ctx| {
162 let outvars = outvars.claim(ctx);
163 let output = output.claim(ctx);
164 move |rt| {
165 let output = match rt.read(output) {
166 crate::run_cargo_build::CargoBuildOutput::ElfBin { bin, dbg } => {
167 OpenvmmHclOutput { bin, dbg }
168 }
169 _ => unreachable!(),
170 };
171
172 for var in outvars {
173 rt.write(var, &output);
174 }
175 }
176 });
177 }
178
179 Ok(())
180 }
181}