编写一个真正可以用的makefile,遇到的问题及特殊符号含义汇总

来源:互联网 发布:javascript函数传值 编辑:程序博客网 时间:2024/05/20 18:48

最近发现网上很多关于Makefile的文章都过于基础,事实上太笨拙的写法在工程中是不可能用到的,因为你的makefile一般会面对几十个.c .h,会有诸多的动态库需要加载。这里做一个汇总和总结,也是对我自己遇到的问题有个简单记录,方便将来自己复习。

一 特殊符号,极大简化Makefile代码。

$? 代表依赖文件列表中被改变过的所有文件。

$^ 代表所有通过目录搜索得到的依赖文件的完整路径名(目录 + 一般文件名)列表。

$@ 代表规则的目标。

$< 代表规则中通过目录搜索得到的依赖文件列表的第一个依赖文件。

$%仅当目标是函数库文件中,表示规则中的目标成员名。

1.AR        

归档维护程序的名称,默认值为 ar

2.ARFLAGS   

归档维护程序的选项。
3.AS        

汇编程序的名称,默认值为 as
4.ASFLAGS   

汇编程序的选项。
5.CC        

C编译器的名称,默认值为 cc
6.CCFLAGS   

C编译器的选项。
7.CPP       

C预编译器的名称,默认值为 $(CC) -E
8.CPPFLAGS  

C预编译的选项。

9.-c -o -O

-c是指到编译那一步为止,也就是生成.o

-o是指生成指定名称的可执行文件。

-O是优化,后面可以加数字指定优化等级。

下面贴一段我的makefile:


SRCS=$(wildcard *.c)OBJS=$(SRCS:.c=.o)CC=gccLDFLAGS=-L/usr/local/lib -lintlc -lsvml -limf enc_sub.soCFLAGS =-I/home/knight/Desktop/DolbySDK/Code/system_code/cli_frontend/include/encoder: $(OBJS)       $(CC) -o $@ $^ $(LDFLAGS)%.o: %.c        $(CC) -c $^ $(CFLAGS)   clean:        rm *.o
可以看到里面有.c .h的编译,动态库的调用。

SRCS=$(wildcard *.c)
OBJS=$(SRCS:.c=.o)</span>
这里两句是指先获取所有的.c,再把所有的.c编译成.o,当然这里不是真的编译,相当于指定一种编译规则。

CC=gcc
指定编译器
LDFLAGS=-L/usr/local/lib -lintlc -lsvml -limf enc_sub.so
这里引用动态库,ldflags指的是在哪个目录搜索动态库,后面接一个大写的L,小写的l是指在指定目录下搜索那些库,看我的调用顺序,因为这几个库是有依赖关系的,具体可以用ldd命令查看某个库的依赖。被其他库所依赖的放前面,前三个库全名是libintlc.so等等,搜索的时候自动去除开头的lib 和 结尾的 .so ,包括 .so.5,这都是自动匹配的。最后一个库为什么不加-l,而且加上了.so呢,因为我这个库的名字就叫enc_sub.so,不需要掐头去尾的搜索,不加-l,直接写全名就好。
CFLAGS =-I/home/knight/Desktop/DolbySDK/Code/system_code/cli_frontend/include/
这里是你所需要的头文件的路径。
encoder: $(OBJS)
encoder是最后要生成的可执行文件。$(OBJS)OBJS是开头定义的变量,指代所有生成的.o,其实makefile就是shell脚本,这里属于脚本语法。
$(CC) -o $@ $^ $(LDFLAGS)

这句我们对应每个空格来说:

用gcc编译器 生成 encoder 依赖于objs这个变量里的所有.o 并且调用XX库文件

<span style="font-family: Arial, Helvetica, sans-serif;">%.o: %.c</span>
规则很简单,要生成.o,依赖.c
$(CC) -c $^ $(CFLAGS) 
用gcc编译器 生成.o 依赖于所有的.c 引用那些头文件

为什么我有一行调用ldflags,有一行调用cflags呢?这就是编译的四个阶段不同导致的。.c->.o->encoder。

这里提两个可能遇见的问题,因为其他的我记不清了=。=:

1.如果你再安装一个软件,./configure时加上如上动态库引用时报错了,说让你用什么交叉编译。那很可能使你把库名字写错了,或者引用格式没写对。

2.如果make的时候出现错误:undefined reference to 'xxxxx'

那你不是库文件没引用成功,就是头文件没引用成功,好好看看你写的对不对。这里贴一段引用网上一段解释,觉得解释的相当到位。

为什么会出现undefined reference to 'xxxxx'错误?
首先这是链接错误,不是编译错误,也就是说如果只有这个错误,说明你的程序源码本身没有问题,是你用编译器编译时参数用得不对,你没有指定链接程序要用到得库,比如你的程序里用到了一些数学函数,那么你就要在编译参数里指定程序要链接数学库,方法是在编译命令行里加入-lm
4、-l参数和-L参数
-l参数就是用来指定程序要链接的库,-l参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是m,他的库文件名是libm.so,很容易看出,把库文件名的头lib和尾.so去掉就是库名了
好了现在我们知道怎么得到库名,当我们自已要用到一个第三方提供的库名字libtest.so,那么我们只要把libtest.so拷贝到/usr/lib里,编译时加上-ltest参数,我们就能用上libtest.so库了(当然要用libtest.so库里的函数,我们还需要与libtest.so配套的头文件)
放在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接了,但如果库文件没放在这三个目录里,而是放在其他目录里,这时我们只用-l参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld: cannot find -lxxx”,也就是链接程序ld在那3个目录里找不到libxxx.so,这时另外一个参数-L就派上用场了,比如常用的X11的库,它在/usr/X11R6/lib目录下,我们编译时就要用-L/usr/X11R6/lib -lX11参数,-L参数跟着的是库文件所在的目录名。再比如我们把libtest.so放在/aaa/bbb/ccc目录下,那链接参数就是-L/aaa/bbb/ccc -ltest
另外,大部分libxxxx.so只是一个链接,以RH9为例,比如libm.so它链接到/lib/libm.s
o.x,/lib/libm.so.6又链接到/lib/libm-2.3.2.so。

欢迎批评指正探讨。

0 0
原创粉丝点击