我热衷于装配,并希望从家里开始编码而不是我经常使用的mac。
我真的为这个问题奋斗了4天。
您可以在以下网址找到我的makefile和clone repository:
<code>NAME = libfts.a
ASM_FILES = ft_isascii \
OS := $(shell uname)
ifeq ($(OS), Darwin)
ASM_COMPILER = ~/.brew/bin/nasm -f macho64 -g
else
ASM_COMPILER = nasm -f elf64 -g
endif
ASM_SRC_DIR = srcs/
ASM_OBJ_DIR_NAME = obj
ASM_OBJ_DIR = $(ASM_OBJ_DIR_NAME)/
ASM_OBJ := $(addsuffix .o,$(ASM_FILES))
ASM_OBJ := $(addprefix $(ASM_OBJ_DIR),$(ASM_OBJ))
TEST = maintest.out
TEST_FILES = maintest
C_COMPILER = clang -Wall -Werror -Wextra -O3
TEST_DIR_NAME = test
TEST_DIR = $(TEST_DIR_NAME)/
TEST_OBJ := $(addsuffix .o,$(TEST_FILES))
TEST_OBJ := $(addprefix $(TEST_DIR),$(TEST_OBJ))
OBJ_PATHS := $(ASM_OBJ) $(TEST_OBJ)
all: $(NAME)
$(NAME): $(ASM_OBJ)
ar rc $(NAME) $(ASM_OBJ)
test: re $(TEST_OBJ)
$(C_COMPILER) -L. $(NAME) $(TEST_OBJ) -o $(TEST)
$(ASM_OBJ): $(ASM_OBJ_DIR)%.o: $(ASM_SRC_DIR)%.s
@/bin/mkdir -p $(ASM_OBJ_DIR)
$(ASM_COMPILER) $< -o $@
$(TEST_OBJ): $(TEST_DIR)%.o: $(TEST_DIR)%.c
$(C_COMPILER) -c -I. $< -o $@
clean:
-/bin/rm -f $(OBJ_PATHS)
/usr/bin/find . -name "$(ASM_OBJ_DIR_NAME)" -maxdepth 1 -type d -empty -delete
fclean: clean
-/bin/rm -f $(NAME)
-/bin/rm -f $(TEST)
re: fclean all
.PHONY: all clean fclean re
当我尝试&#34;进行测试&#34;时出现此错误消息在linux上:
test/maintest.o: In function `main':
test/maintest.c:(.text+0x33): undefined reference to `ft_isascii'
undefined reference to etc.
.h内容:
#ifndef _LIBFTS
# define _LIBFTS
#include <stddef.h>
int ft_isascii(int c);
#endif
ft_isascii.s内容:
global _ft_isascii
section .text
_ft_isascii: ; int ft_isascii
and edi, 0xffffff80 ; mask with the 128 firsts bits left to 0 as ASCII range from 0 to 7f in hexa (just below 80)
sete al ; SETE sets AL to 1 if above condition code means "equal", otherwise it sets AL to 0.
movzx eax, al
ret
我真的很感谢任何解决这个问题的技巧......
此致
答案 0 :(得分:2)
有两个问题需要解决:
首先,在ELF目标(除macOS之外的大多数Unix)上,C函数没有用下划线装饰。要修复代码,请从所有符号中删除前导下划线。确保在任何地方都将其删除。
其次,链接器在查看存档(.a
文件)时,只选择它现在需要的文件来满足依赖关系。因此,当您在maintest.o
之前传递归档时,链接器根本不会从归档中获取任何内容,因为此时不需要任何ft_...
符号。只有链接器看到maintest.o
后才需要这些符号。要解决此问题,请将$(NAME)
操作数移至$(TEST_OBJ)
之后。作为一般经验法则,请始终将库放在链接器命令行上的目标文件之后。
在macOS上,你不会发现这个问题,因为它们使用lld,即LLVM链接器,这有点不同寻常,因为它推迟了选择哪些对象从档案中取出,直到它查看了所有符号表。操作数,使您实际上破坏了调用工作。请不要依赖这种行为。