如何 link 带有 .asm 文件的 .a 库?

How to link an .a library with an .asm file?

如问题所述,我正在尝试 link 一个带有 .asm 文件的 .a 库。我正在使用 Ubuntu、nasm 和 gcc 通过以下 MRE 和命令执行此操作:

nasm -f elf64 -o use.o use.asm
nasm -f elf64 -o strlen.o strlen.asm

ar rc libstring.a strlen.o
ranlib libstring.a

ld -static -nostdlib -build-id=none -L. -lstring -o use.bin use.o

使用的文件如下所示: strlen.asm:

BITS 64
GLOBAL strlen

;-------------------------------------------------------------------------------
; @func strlen
;
; @params String str
;
; @returns uint_32 length
;-------------------------------------------------------------------------------

strlen:
push rcx
push rsi

xor eax, eax
xor ecx, ecx
mov rsi, [rsp + 0x10]

jmp .foreach

.inc_i:
inc ecx

.foreach:
lodsb
or al, al
jnz .inc_i

.return:
mov eax, ecx
pop rsi
pop rcx
ret

use.asm:

BITS 64

GLOBAL _start
EXTERN strlen

_start:
push str
call strlen
cli
hlt

str: DB "String!", 0x00

我不希望他们做任何事情,他们应该只是编译。目前 linker 给出了这个错误:

ld: use.o: in function `_start':
use.asm:(.text+0x6): undefined reference to `strlen'

由于错误没有提到找不到文件,我猜.a文件的命名约定是正确的。但是为什么找不到方法呢?

为了解决这个问题,必须以某种方式对 -L 和 -l 标志进行排序,将它们放在需要它们的文件之后。在不同位置使用多个库时,必须确保重新定位不会将 -L 标志放错位置,这也可能导致一些问题。

所以在这个 MRE 中,ld 命令应该是这样的:
ld -static -nostdlib -build-id=none use.o -L . -l string -o use.bin

另外在使用bash脚本编译的时候不要忘记sudo chmod +x make.sh,那可能会让你头疼5分钟^^