xtask/tasks/guest_test/uefi/
mod.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
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

use crate::Xtask;
use clap::Parser;
use std::path::Path;
use std::path::PathBuf;
use xshell::cmd;

mod gpt_efi_disk;

/// Build a UEFI test image.
#[derive(Parser)]
pub struct Uefi {
    // Output disk image. If left blank, outputs disk at `<bootx64/bootaa64>.img`
    // if only one EFI boot file is provided.
    //
    // Extension determines disk type.
    //
    // Only `.img` disk image files currently supported.
    #[clap(long)]
    output: Option<PathBuf>,

    /// File to set as `bootx64.efi`. Builds `guest_test_uefi` for x64 if no file is provided (default).
    #[clap(long)]
    #[expect(clippy::option_option)]
    bootx64: Option<Option<PathBuf>>,

    /// File to set as `bootaa64.efi`. Builds `guest_test_uefi` for ARM64 if no file is provided.
    #[clap(long)]
    #[expect(clippy::option_option)]
    bootaa64: Option<Option<PathBuf>>,
}

impl Xtask for Uefi {
    fn run(self, _ctx: crate::XtaskCtx) -> anyhow::Result<()> {
        let mut files = Vec::new();

        // Default case: build x64
        let bootx64 = if self.bootx64.is_none() && self.bootaa64.is_none() {
            Some(None)
        } else {
            self.bootx64
        };

        if let Some(bootx64) = bootx64.as_ref() {
            if let Some(bootx64) = bootx64 {
                files.push((Path::new("efi/boot/bootx64.efi"), bootx64.as_path()));
            } else {
                let sh = xshell::Shell::new()?;
                cmd!(
                    sh,
                    "cargo build -p guest_test_uefi --target x86_64-unknown-uefi"
                )
                .run()?;

                files.push((
                    Path::new("efi/boot/bootx64.efi"),
                    Path::new("./target/x86_64-unknown-uefi/debug/guest_test_uefi.efi"),
                ));
            }
        }

        if let Some(bootaa64) = self.bootaa64.as_ref() {
            if let Some(bootaa64) = bootaa64 {
                files.push((Path::new("efi/boot/bootaa64.efi"), bootaa64.as_path()))
            } else {
                let sh = xshell::Shell::new()?;
                cmd!(
                    sh,
                    "cargo build -p guest_test_uefi --target aarch64-unknown-uefi"
                )
                .run()?;

                files.push((
                    Path::new("efi/boot/bootaa64.efi"),
                    Path::new("./target/aarch64-unknown-uefi/debug/guest_test_uefi.efi"),
                ));
            }
        }

        let out_img = match self.output {
            Some(path) => path,
            None => {
                if files.len() != 1 {
                    anyhow::bail!(
                        "Multiple EFI files specified. Please provide an explicit output path."
                    )
                }
                files[0].1.with_extension("img")
            }
        };

        gpt_efi_disk::create_gpt_efi_disk(&out_img, &files)?;

        Ok(())
    }
}