程序集:用于自定义操作系统键盘支持的引导加载程序

标签 assembly x86 att osdev multiboot

我有一个工作简单的自定义操作系统(目前没有做太多事情:D)。现在我正在使用一个没有键盘支持的汇编文件(boot.s)。

汇编文件(boot.s):

# set magic number to 0x1BADB002 to identified by bootloader 
.set MAGIC,    0x1BADB002
# set flags to 0
.set FLAGS,    0
# set the checksum
.set CHECKSUM, -(MAGIC + FLAGS)
# set multiboot enabled
.section .multiboot
# define type to long for each data defined as above
.long MAGIC
.long FLAGS
.long CHECKSUM
# set the stack bottom 
stackBottom:
# define the maximum size of stack to 512 bytes
.skip 512
# set the stack top which grows from higher to lower
stackTop:
.section .text
.global _start
.type _start, @function

_start:
  # assign current stack pointer location to stackTop
    mov $stackTop, %esp
  # call the kernel main source
    call KERNEL_MAIN
    cli
# put system in infinite loop
hltLoop:
    hlt
    jmp hltLoop
.size _start, . - _start

我认为这是缺失的部分,但它采用英特尔语法,我无法使用它。

load_idt:
mov edx, [esp + 4]
lidt [edx]
sti
ret

read_port:
mov edx, [esp + 4]
in al, dx   
ret

write_port:
mov edx, [esp + 4]    
mov al, [esp + 4 + 4]  
out dx, al  
ret

keyboard_handler:                 
call keyboard_handler
iretd

我正在使用以下命令编译 boot.s:

as --32 boot.s -o boot.o

谁能帮我将键盘部分(Intel 语法)翻译成 AT&T? :)

最佳答案

有关如何将 NASM Intel 语法转换为 GAS 的 AT&T 语法的信息可以在 Stackoverflow Answer 中找到。 ,并且此IBM article中提供了很多有用的信息。 。您的代码尤其如下所示:

load_idt:
    mov 4(%esp), %edx
    lidt (%edx)
    sti
    ret

read_port:
    mov 4(%esp), %edx
    in %dx, %al
    ret

write_port:
    mov 4(%esp), %edx
    mov 8(%esp), %al
    out %al, %dx
    ret

keyboard_handler:                 
    call keyboard_handler
    iret

总的来说,最大的区别是:

  • 对于 AT&T 语法,源位于左侧,目标位于右侧,而 Intel 则相反。
  • 使用 AT&T 语法,寄存器名称前面带有 %
  • 对于 AT&T 语法,立即值前面带有 $
  • 内存操作数可能是最大的区别。 NASM 使用[segment:disp+base+index*scale]而不是GAS 的segment:disp(base, index,scale) 语法。

其他观察结果

我建议将堆栈从多重引导部分移至 .bss 部分。 BSS 部分通常不会占用输出可执行文件中的空间(假设使用正常或默认的链接描述文件)。可以在 .text 部分之后以这种方式定义堆栈:

.section .bss
    .lcomm stackBottom 512
stackTop:

关于程序集:用于自定义操作系统键盘支持的引导加载程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54960681/

相关文章:

linux - 如何在 x86 程序集中使用 "nanosleep"?

程序集 BIOS 调用不起作用

macos - 在 x64 ASM 中循环并打印 argv[]

c - 理解简单的汇编程序

assembly - Intel 64架构中CALLF(Far Call)可以有64位地址内存操作数吗?

algorithm - AND、OR 或 XOR 汇编语言 MASM X86 IRVINE

assembly - 处理器如何知道程序结束?

使用 PAPI_read_counters 计算 L1 缓存未命中会产生意外结果

assembly - 在汇编中推送并打印浮点值

c - 这两个函数序言指令序列有什么区别?