使用.org指令和.data部分中的数据:与ld相关联

时间:2012-03-07 05:58:39

标签: linux ld gas att

我努力了解如何使用GNU binutils构建一个简单的引导加载程序 gas我遇到过这个问题,你如何告诉链接器将数据放在哪里,在一个文件中使用.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'

1 个答案:

答案 0 :(得分:4)

对于这类工作,你应该自己编写linker script。只需使用-T选项将其传递给链接器。

我的脚本几乎完全相同的问题是:

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

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

开头的0x1000:实际上所有跳转都是相对的,所以没有使用,但稍后需要跳转到保护模式......

endbootsecwendbootsecimage是代码中其他位置使用的符号。