我正在将Windows 10和Windows子系统用于Linux。 我已经开始在Assembly和C中创建自己的操作系统。 我正在学习教程。 我陷入了两个问题。
错误1: 当我链接并创建bin文件时,我收到警告: “ ld:警告:找不到条目符号_start;默认为0000000000001000” 这有关系吗?
错误2: 编译我的代码后,没有错误。但是,当我启动操作系统时,它显示一个错误:磁盘读取错误!
请帮助我。
Boot.asm
[org 0x7c00]
KERNEL_OFFSET equ 0x1000
mov [BOOT_DRIVE], dl
mov bp, 0x9000
mov sp, bp
mov si, MSG_REAL_MODE
call print
call load_kernel
call switch_to_pm
jmp $
%include "printstr.asm"
%include "diskload.asm"
[bits 16]
load_kernel :
mov si, MSG_LOAD_KERNEL
call print
mov bx, KERNEL_OFFSET
mov dh, 15
mov dl, [ BOOT_DRIVE ]
call disk_load
ret
[bits 32]
BEGIN_PM :
mov ebx, MSG_PROT_MODE
call print_string_pm
call KERNEL_OFFSET
jmp $
BOOT_DRIVE db 0
MSG_REAL_MODE db " Started in 16 - bit Real Mode " , 0
MSG_PROT_MODE db " Successfully landed in 32 - bit Protected Mode " , 0
MSG_LOAD_KERNEL db " Loading kernel into memory. " , 0
times 510 -( $ - $$ ) db 0
dw 0xaa55
diskload.asm
disk_load :
push dx
mov ah , 0x02
mov al , dh ;
mov ch , 0x00
mov dh , 0x00
mov cl , 0x02
int 0x13
jc disk_error
pop dx
cmp dh , al
jne disk_error
ret
disk_error :
mov si , DISK_ERROR_MSG
call prints
jmp $
; Variables
DISK_ERROR_MSG db " Disk read error !" , 0
prints:
lodsb
or al, al
jz printdones
mov ah, 0eh
int 10h
jmp prints
printdones:
ret
编译命令:
nasm boot.asm -f bin -o boot.bin
nasm kernel_entry.asm -f elf64 -o kernel_entry.o
gcc -ffreestanding -c kernel.c -o kernel.o
ld -o kernel.bin -Ttext 0x1000 kernel_entry.o kernel.o --oformat binary
cat boot.bin kernel.bin > os.iso
答案 0 :(得分:1)
ld:警告:找不到条目符号_start;默认为0000000000001000
此错误的结果是,可执行文件中存储的入口点地址不正确。
但是,“原始”二进制文件仅包含一些内存内容;它们不包含任何其他信息(例如入口点),就像ELF或COFF文件一样。
换句话说:对于“原始”二进制文件(--oformat binary
),此警告消息根本没有任何意义。
但是当我启动操作系统时,它显示一个错误:磁盘读取错误!
我不确定,但是可能有两个错误:
您确定ES
寄存器包含正确的值吗?
(如果要将内核加载到绝对地址0x1000中,则必须将ES
设置为0。)
许多驱动器类型不支持一次读取太多扇区。
(但是,真正的1440K驱动器应支持从#2扇区开始读取15个扇区。)
答案 1 :(得分:0)
我也遇到了这个问题,磁盘读取错误可能是由于您的最终os映像对于15个扇区的读取而言太小了。
如果您使用qemu,请尝试使用以下命令来调整操作系统图像的大小:
map(.title) | max_by(length)
这会将您的图片大小调整为20 KB (您可以输入自己的价值而不是2万)。
我认为qemu将os映像视为整个磁盘,因此您应该相应地调整其大小。