将 .org 指令与 .data 部分中的数据一起使用:与 ld 有关



在我努力理解如何使用GNU binutils构建一个简单的引导加载程序的过程中我遇到了一个问题,你如何告诉链接器将数据放在一个文件中,该文件使用 .org 推进位置计数器,同时将文件大小保持在 512 字节。我似乎找不到一种方法来做到这一点。

尝试执行此操作的汇编程序是:

# Author: Matthew Hoggan
# Date Created: Tuesday, Mar 6, 2012
.code16                      # Tell assembler to work in 16 bit mode
.section .data
msg:                         # Message to be printed to the screen
  .asciz "hello, world"
.section .text
.globl _start                # Help linker find start of program
_start:
  xor  %ax,  %ax             # Zero out ax register (ah used to specify bios function to Video Services) 
  movw %ax,  %ds             # Since ax is 0x00 0x00 use it to set data segment to 0
  mov  $msg, %si             # Use source index as pointer to string
loop:
  movb %ds:(%si), %al        # Pass data to BIOS function in low nibble of ax
  inc  %si                   # Advance the pointer
  or   %al, %al              # If byte stored in al is equal to zero then...
  jz   _hang                 # Zero signifies end of string
  call print_char            # Print current char in al
  jmp loop                   # Repeat
#
# Function that uses bios function calls
# to print char to screen
#
print_char:
  movb $0x0e, %ah             # Function to print a character to the screen
  movb $0x07, %bl             # color/style to use for the character
  int  $0x10                  # Video Service Request to Bios
  ret
#
# Function used as infinite loop
#
_hang:
  jmp  _hang
.org 510
.byte 0x55, 0xAA
.end

更新使用以下命令,我收到以下错误:

mehoggan@mehoggan-laptop:~/Code/svn_scripts/assembly/bootloader/gas$ ld --oformat binary -o string string.o 
string.o: In function `_start':
(.text+0x5): relocation truncated to fit: R_X86_64_16 against `.data'

对于此类工作,您应该编写自己的链接器脚本。只需使用-T选项将其传递给链接器即可。

我针对一个几乎相同的问题的脚本是:

SECTIONS
{
    . = 0x1000;
    bootsec :
    {
        *(.text)
        *(.data)
        *(.rodata)
        *(.bss)
        endbootsec = .;
        . = 0x1FE;
        SHORT(0xAA55)
    }
    endbootsecw = endbootsec / 2;
    image = 0x1200;
}

有了这个技巧,你不需要把0x55 0xAA放在汇编器中!

开头的0x1000:其实所有的跳转都是相对的,所以不用了,但以后跳转到保护模式需要它......

endbootsecwendbootsecimage是代码中其他地方使用的符号。

最新更新