flowey_lib_hvlite/
build_openhcl_boot.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! Build `openhcl_boot` binaries

use crate::run_cargo_build::BuildProfile;
use crate::run_cargo_build::common::CommonArch;
use flowey::node::prelude::*;
use std::collections::BTreeMap;

#[derive(Serialize, Deserialize)]
pub struct OpenhclBootOutput {
    pub bin: PathBuf,
    pub dbg: PathBuf,
}

#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize)]
pub enum OpenhclBootBuildProfile {
    Debug,
    Release,
}

#[derive(Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)]
pub struct OpenhclBootBuildParams {
    pub arch: CommonArch,
    pub profile: OpenhclBootBuildProfile,
}

flowey_request! {
    pub struct Request {
        pub build_params: OpenhclBootBuildParams,
        pub openhcl_boot: WriteVar<OpenhclBootOutput>,
    }
}

new_flow_node!(struct Node);

impl FlowNode for Node {
    type Request = Request;

    fn imports(ctx: &mut ImportCtx<'_>) {
        ctx.import::<crate::run_cargo_build::Node>();
    }

    fn emit(requests: Vec<Self::Request>, ctx: &mut NodeCtx<'_>) -> anyhow::Result<()> {
        // de-dupe incoming requests
        let requests = requests
            .into_iter()
            .fold(BTreeMap::<_, Vec<_>>::new(), |mut m, r| {
                let Request {
                    build_params,
                    openhcl_boot,
                } = r;
                m.entry(build_params).or_default().push(openhcl_boot);
                m
            });

        for (OpenhclBootBuildParams { arch, profile }, openhcl_boot) in requests {
            let target = match arch {
                CommonArch::X86_64 => target_lexicon::Triple {
                    architecture: arch.as_arch(),
                    operating_system: target_lexicon::OperatingSystem::None_,
                    environment: target_lexicon::Environment::Unknown,
                    vendor: target_lexicon::Vendor::Unknown,
                    binary_format: target_lexicon::BinaryFormat::Unknown,
                },
                CommonArch::Aarch64 => target_lexicon::Triple {
                    architecture: arch.as_arch(),
                    operating_system: target_lexicon::OperatingSystem::Linux,
                    environment: target_lexicon::Environment::Musl,
                    vendor: target_lexicon::Vendor::Unknown,
                    binary_format: target_lexicon::BinaryFormat::Elf,
                },
            };

            // We use special profiles for boot, convert from the standard ones:
            let profile = match profile {
                OpenhclBootBuildProfile::Debug => BuildProfile::BootDev,
                OpenhclBootBuildProfile::Release => BuildProfile::BootRelease,
            };

            let output = ctx.reqv(|v| crate::run_cargo_build::Request {
                crate_name: "openhcl_boot".into(),
                out_name: "openhcl_boot".into(),
                crate_type: flowey_lib_common::run_cargo_build::CargoCrateType::Bin,
                profile,
                features: Default::default(),
                target,
                no_split_dbg_info: false,
                extra_env: Some(ReadVar::from_static(
                    [("MINIMAL_RT_BUILD".to_string(), "1".to_string())]
                        .into_iter()
                        .collect(),
                )),
                pre_build_deps: Vec::new(),
                output: v,
            });

            ctx.emit_minor_rust_step("report built openhcl_boot", |ctx| {
                let openhcl_boot = openhcl_boot.claim(ctx);
                let output = output.claim(ctx);
                move |rt| {
                    let output = match rt.read(output) {
                        crate::run_cargo_build::CargoBuildOutput::ElfBin { bin, dbg } => {
                            OpenhclBootOutput {
                                bin,
                                dbg: dbg.unwrap(),
                            }
                        }
                        _ => unreachable!(),
                    };

                    for var in openhcl_boot {
                        rt.write(var, &output);
                    }
                }
            });
        }

        Ok(())
    }
}