首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >创建可启动的UEFI ISO

创建可启动的UEFI ISO
EN

Stack Overflow用户
提问于 2018-06-08 10:57:33
回答 1查看 5K关注 0票数 1

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

ISO创建:

代码语言:javascript
复制
xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles

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

Hyper-V输出:

编辑:让我们对此有更多的背景,好吗?

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

代码语言:javascript
复制
#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:

代码语言:javascript
复制
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:

代码语言:javascript
复制
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)

我知道Makefile目前真的很糟糕,但我现在只是想让它正常工作。稍后我会让它们变得更好。

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

代码语言:javascript
复制
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似乎并不存在。

EN

回答 1

Stack Overflow用户

发布于 2018-06-08 14:50:06

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

因此,首先需要创建FAT文件系统映像,使程序具有指定的名称。让我们将图像文件命名为"efi.img“,并将其放入当前工作目录中。然后将FAT镜像文件作为数据文件放入ISO中,并将其标记为EFI El Torito引导镜像,以便EFI在(虚拟) DVD介质上找到它。

代码语言:javascript
复制
xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles \
./efi.img -e /efi.img -no-emul-boot 

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

为了从(虚拟)硬盘或U盘引导,您需要将FAT文件系统映像附加为类型为0xEF的MBR分区:

代码语言:javascript
复制
xorriso -as mkisofs -o uefi.iso -iso-level 3 -V UEFI isoFiles \
-append_partition 2 0xef ./efi.img

(请注意,选项-append_partition需要本地硬盘上的文件地址。)

这两个附加组件可以组合在一起,代价是在ISO中拥有两个FAT映像副本。xorriso版本的>= 1.4.6可以通过一个特殊的伪路径"--interval:...“来避免这种重复。使用选项-e:

代码语言:javascript
复制
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
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/50752688

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档