例子目录结构
project/
├── main.c
├── util.c
└── Makefile
GCC 子命令的执行时序图
sequenceDiagramparticipant U as Userparticipant G as gccparticipant C as cppparticipant CC as cc1participant A as asparticipant AR as arparticipant L as ldparticipant F as FilesU->>G: gcc main.c util.c -o main (简化命令)Note right of G: 开始多文件编译流程%% 处理 main.cG->>C: cpp 处理 main.cC->>F: 生成 main.iG->>CC: cc1 处理 main.iCC->>F: 生成 main.sG->>A: as 处理 main.sA->>F: 生成 main.o%% 处理 util.cG->>C: cpp 处理 util.cC->>F: 生成 util.iG->>CC: cc1 处理 util.iCC->>F: 生成 util.sG->>A: as 处理 util.sA->>F: 生成 util.o%% 可选:使用 ar 创建静态库Note right of G: 如果需要静态库,则插入 ar 步骤U->>AR: ar rcs libutil.a util.oAR->>F: 生成 libutil.aNote right of AR: 中间步骤:打包 util.o 为静态库%% 链接阶段G->>L: ld 处理 main.o 和 libutil.a (-lutil)L->>F: 生成 main (可执行文件)Note right of L: 链接时使用静态库G-->>U: 返回编译成功的可执行文件 main
Makefile 例子1,全部用子命令完成完整GCC预处理,编译,汇编,链接流程:
# 定义工具路径(根据系统调整)
CC1 = /usr/lib/gcc/x86_64-linux-gnu/11/cc1 # 请根据你的 gcc 版本调整路径
CPP = /usr/bin/cpp
AS = /usr/bin/as
LD = /usr/bin/ld
AR = /usr/bin/ar
RM = rm -f# 目标文件和依赖
all: main# 链接阶段:使用独立的 ld 命令
main: main.o libutil.a$(LD) -o main main.o -L. -lutil -lc# 汇编阶段:main.o
main.o: main.s$(AS) main.s -o main.o# 编译阶段:main.s
main.s: main.i$(CC1) main.i -o main.s# 预处理阶段:main.i
main.i: main.c$(CPP) main.c -o main.i# 静态库生成:libutil.a
libutil.a: util.o$(AR) rcs libutil.a util.o# 汇编阶段:util.o
util.o: util.s$(AS) util.s -o util.o# 编译阶段:util.s
util.s: util.i$(CC1) util.i -o util.s# 预处理阶段:util.i
util.i: util.c$(CPP) util.c -o util.i# 清理
clean:$(RM) *.o *.a *.i *.s main
Makefile 例子2: 用 gcc 来分阶段处理:
# 定义工具路径
GCC = /usr/bin/gcc
AR = /usr/bin/ar
RM = rm -f# 目标文件和依赖
all: main# 链接阶段:使用 gcc 调用 ld
main: main.o libutil.a$(GCC) main.o -L. -lutil -o main# 汇编阶段:main.o
main.o: main.s$(GCC) -c main.s -o main.o# 编译阶段:main.s
main.s: main.i$(GCC) -S main.i -o main.s# 预处理阶段:main.i
main.i: main.c$(GCC) -E main.c -o main.i# 静态库生成:libutil.a
libutil.a: util.o$(AR) rcs libutil.a util.o# 汇编阶段:util.o
util.o: util.s$(GCC) -c util.s -o util.o# 编译阶段:util.s
util.s: util.i$(GCC) -S util.i -o util.s# 预处理阶段:util.i
util.i: util.c$(GCC) -E util.c -o util.i# 清理
clean:$(RM) *.o *.a *.i *.s main