创建可启动的UEFI ISO

时间:2018-06-08 02:57:33

标签: hyper-v uefi

我正在尝试创建可启动的UEFI ISO,但Hyper-V并未将其视为有效的UEFI文件系统。我做错了什么?

ISO创作:

xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles

isoFiles只有一个文件boot/hello.efi。在存档管理器中安装或打开时,该文件存在。

Hyper-V输出: Hyper-V screenshot

编辑: 我们还有更多的背景吗?

这是EFI代码,它使用gnu-efi v3.0.8:

#include <efi.h>
#include <efilib.h>

EFI_STATUS efi_main(EFI_HANDLE ImageHandle, EFI_SYSTEM_TABLE *SystemTable){
    EFI_STATUS Status;
    EFI_INPUT_KEY Key;

    ST = SystemTable;

    Status = ST->ConOut->OutputString(ST->ConOut, L"Hello world!\n\r");
    if(EFI_ERROR(Status)){
        return Status;
    }

    Status = ST->ConIn->Reset(ST->ConIn, FALSE);
    if(EFI_ERROR(Status)){
        return Status;
    }

    while((Status = ST->ConIn->ReadKeyStroke(ST->ConIn, &Key)) == EFI_NOT_READY){}

    return Status;
}

这是它的Makefile:

DIR_OUT     := ..
TARGET      := uefi
OUT         := $(DIR_OUT)/out/boot

DIR_SRC     := src
DIR_BLD     := build

EFI_DIR     := gnu-efi-3.0.8
EFI_LIB     := $(EFI_DIR)/x86_64/gnuefi/ $(EFI_DIR)/x86_64/lib
EFI_INC     := $(EFI_DIR)/inc $(EFI_DIR)/inc/x86_64 $(EFI_DIR)/inc/protocol
EFI_CRT     := $(EFI_DIR)/x86_64/gnuefi/crt0-efi-x86_64.o
EFI_LDS     := $(EFI_DIR)/gnuefi/elf_x86_64_efi.lds

CFILES      := $(wildcard $(DIR_SRC)/*.c)
OFILES      := $(patsubst $(DIR_SRC)/%.c,$(DIR_BLD)/%.o,$(CFILES))

PREFIX      := 
CC          := $(PREFIX)gcc
LD          := $(PREFIX)ld
OBJCPY      := $(PREFIX)objcopy

INCLUDES    := $(addprefix -I,$(EFI_INC))
CFLAGS      := -fno-stack-protector -fpic -fshort-wchar -mno-red-zone -Wall -ffreestanding -DEFI_FUNCTION_WRAPPER -c $(INCLUDES)
LDFLAGS     := -nostdlib -znocombreloc -T $(EFI_LDS) -shared -Bsymbolic $(addprefix -L,$(EFI_LIB)) $(EFI_CRT)

all: builddir bootdir $(OUT)/$(TARGET).efi

.PRECIOUS: $(OFILES)

clean:
    rm -vrf $(OFILES) $(OUT)/$(TARGET).efi

$(OUT)/$(TARGET).efi: $(DIR_BLD)/$(TARGET).efi
    cp $< $@

%.efi: %.so
    $(OBJCPY) -j .text -j .sdata -j .data -j .dynamic -j .dynsym  -j .rel -j .rela -j .reloc --target=efi-app-x86_64 $^ $@

$(DIR_BLD)/$(TARGET).so: $(OFILES)
    $(LD) $(LDFLAGS) $< -o $@ -lefi -lgnuefi

$(DIR_BLD)/%.o: $(DIR_SRC)/%.c
    $(CC) $(CFLAGS) $< -o $@

bootdir:
    mkdir -p $(OUT)
builddir:
    mkdir -p $(DIR_BLD)

这是创建ISO的Makefile:

ISO_DIR     := ..
ISO_FILE    := uefi.iso
ISO_LABEL   := UEFI

DIR_BLD     := build

MKFSISO     := xorriso -as mkisofs

FATSIZE     := 8M
MKFSVFAT    := /sbin/mkfs.vfat
MOUNT       := sudo mount
UMOUNT      := sudo umount

UID         := $(shell id -u gudenau)
GID         := $(shell id -g gudenau)

all: buildDir $(ISO_DIR)/$(ISO_FILE)

.PRECIOUS: $(ISO_DIR)/$(ISO_FILE) $(DIR_BLD)/efi.img

$(ISO_DIR)/%.iso: $(DIR_BLD)/efi.img
    $(MKFSISO) -o $@ -iso-level 3 -V "$(ISO_LABEL)" $(DIR_BLD)/efi.img -e /efi.img -no-emul-boot

%.img:
    dd if=/dev/zero of=$@ bs=1024 count=1024
    $(MKFSVFAT) $@
    mkdir -p $(DIR_BLD)/image
    $(MOUNT) -o gid=$(GID),uid=$(UID) $@ $(DIR_BLD)/image
    mkdir -p $(DIR_BLD)/image/EFI/BOOT/
    cp ../bootloader/build/uefi.efi $(DIR_BLD)/image/EFI/BOOT/BOOTX64.EFI
    $(UMOUNT) $(DIR_BLD)/image
    rm -vfr $(DIR_BLD)/image

clean:
    rm -vf $(DIR_BLD)/efi.img $(ISO_DIR)/$(ISO_FILE)

buildDir:
    mkdir -p $(DIR_BLD)

我知道Makefiles目前真的很糟糕,但我现在只是想让它工作。我会在以后更好。

编辑: 经过更多的研究,我现在有了这个Makefile:

ISO_DIR     := ..
ISO_FILE    := uefi.iso
ISO_LABEL   := UEFI

DIR_BLD     := build

MKFSISO     := xorriso -as mkisofs

FATHEADS    := 32
FATTRACKS   := 32
FATSECTOR   := 512
FATTHING    := 128
FATFORMAT   := mformat

MMD         := mmd
MCOPY       := mcopy

all: buildDir $(ISO_DIR)/$(ISO_FILE)

.PRECIOUS: $(ISO_DIR)/$(ISO_FILE) $(DIR_BLD)/efi.img

$(ISO_DIR)/%.iso: $(DIR_BLD)/efi.img
    $(MKFSISO) -o $@ -iso-level 3 -V "$(ISO_LABEL)" $(DIR_BLD)/efi.img -e /efi.img -no-emul-boot

%.img:
    $(FATFORMAT) -i $@ -F -h $(FATHEADS) -t $(FATTRACKS) -n $(FATTHING) -c 1 -C
    $(MDD) -i $@ ::/EFI
    $(MDD) -i $@ ::/EFI/BOOT
    $(MCOPY) -i $@ ../bootloader/build/uefi.efi ::/EFI/BOOT/BOOTX64.EFI

clean:
    rm -vf $(DIR_BLD)/efi.img $(ISO_DIR)/$(ISO_FILE)

buildDir:
    mkdir -p $(DIR_BLD)

但我无法弄清楚如何用它制作GPT图像,因为mkgpt似乎不存在。

1 个答案:

答案 0 :(得分:0)

根据(虚拟)介质类型,您需要分区表或 标记EFI系统分区的El Torito引导目录。 在该分区内必须是带有启动程序的FAT文件系统。 该程序的名称取决于CPU类型。例如。 \ EFI \ BOOT \ BOOTX64.EFI 适用于64位Intel / AMD。对于32位Intel / AMD,它是BOOTIA32.EFI。

首先,您需要创建FAT文件系统映像 具有规定名称的程序。我们将图像文件命名为“efi.img” 并将其放入当前的工作目录。 然后你把FAT图像文件 作为数据文件进入ISO并将其标记为EFI El Torito启动映像 这样EFI就可以在(虚拟)DVD介质上找到它。

xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles \
./efi.img -e /efi.img -no-emul-boot 

(注意,选项-e需要ISO内的文件地址。)

要从(虚拟)硬盘或USB记忆棒启动,你可以附加 FAT文件系统映像为0xEF类型的MBR分区:

xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles \
-append_partition 2 0xef ./efi.img

(注意,选项-append_partition要求文件地址为on  本地硬盘。)

两个附加组件可以合并为两个副本 ISO中的FAT图像。 xorriso版本&gt; = 1.4.6可以避免这种重复 通过一个特殊的伪路径“--interval:...”,带选项-e:

xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles \
-append_partition 2 0xef ./efi.img \
-e --interval:appended_partition_2:all:: -no-emul-boot