Я думаю, что вы должны связать свою программу со стандартной библиотекой C (libc.so), добавив опцию -lc к аргументам ld.
ld -T linker.ld -lc -dynamic-linker /lib/ld-linux.so.2 -o test test.o
Также у вас могут возникнуть некоторые проблемы с запуском вашей программы (ошибки сегментации), потому что у вашего test.o нет точки входа в программу (символ _start). Поэтому вам потребуется дополнительный объектный файл с точкой входа, который вызывает вашу функцию main () внутри test.o, а затем прерывает выполнение кода, вызывая системный вызов exit ().
Вот код start.s
# Linux system calls constants
.equ SYSCALL_EXIT, 1
.equ INTERRUPT_LINUX_SYSCALL, 0x80
# Code section
.section .text
.globl _start
_start: # Program entry point
call main # Calling main function
# Now calling exit() system call
movl %eax, %ebx # Saving return value for exit() argument
movl $SYSCALL_EXIT, %eax # System call number
int $INTERRUPT_LINUX_SYSCALL # Raising programm interrupt
Тогда вы должны собрать свою программу
gcc test.c -S
as test.s -o test.o
as start.s -o start.o
ld start.o test.o -o test -lc --dynamic-linker=/lib/ld-linux.so.2
Вы также можете проверить эту статью https://blogs.oracle.com/ksplice/entry/hello_from_a_libc_free, чтобы узнать больше о том, как работает компилятор C и стандартная библиотека.