makefile分层目录结构

来源:互联网 发布:新淘宝店如何破零知乎 编辑:程序博客网 时间:2024/05/24 03:25
 多目录Makefile(库及分层目录) 2012-06-13 10:28:01

分类: LINUX

工程目录名称为lib-test,其下有3个代码目录及一个头文件目录,分别是foo(foo.c及common.c文件)、bar(bar.c及common.c文件)、bt(backtrace.c文件)和configs(foo.h、bar.h及backtrace.h文件);其中bt目录为backtrace代码,它是作者的工程库的一部分。
目录路径、文件分布如下:
lib-test
|-->main.c
|-->example.c
|-->Makefile
|              |-->foo.c
|-->foo-->|-->common.c
|              |-->Makefile
|
|               |-->bar.c
|-->bar-->|-->common.c
|               |-->Makefile
|
|              |-->backtrace.c
|-->bt--->|
|              |-->Makefile
|
|                   |-->foo.h
|-->configs->|-->bar.h
|                   |-->backtrace.h
Makefile组织
1、各个子目录单独使用Makefile,主要生成相关的库(这里特指静态库);
2、顶层Makefile负责将本目录下源代码文件编译成目标文件(如果有的话),并依次进入各种子目录编译生成相关库文件,最后进行链接,生成可执行文件。该Makefile关键语句如下:
for dir in $(SUBDIRS); \
do $(MAKE) -C $$dir all || exit 1; \
done
意思是进入指定子目录,并执行子目录的Makefile文件(子目录只负责生成库文件)。
库相互调用测试(详见代码):
1、foo.c文件中hello_foo函数调用bar.c文件中的bar函数以及同一目录下common.c文件的common_bar函数;
2、bar.c文件中hello_bar函数调用foo.c文件中的foo函数以及同一目录下common.c文件的common_foo函数。
代码示例:
1、foo目录下:
1)、foo.h
#ifndef FOO_H_
#define FOO_H_
void foo(int i);
int hello_foo(void);
#endif
2)、foo.c
#include 
void foo(int i)
{
        printf("hell from %s() in file: %s, num:%d\n", __func__, __FILE__, i);
}
int hello_foo(void)
{
        printf("in func: %s()\n", __func__);
        bar(100);
        common_foo();
        printf("=========================================\n");
        return 0;
}
3)、common.c
#include 
int common_foo(void)
{
        printf("this is a common function in foo/common.c\n");
        return 0;
}
2、bar目录
1)、bar.h
#ifndef BAR_H_
#define BAR_H_
void bar(int i);
int hello_bar(void);
#endif
2)、bar.c
#include 
void bar(int i)
{
        printf("hell from %s() in file: %s, num:%d\n", __func__, __FILE__, i);
}
int hello_bar(void)
{
        printf("in func: %s()\n", __func__);
        foo(200);
        common_bar();
        printf("=========================================\n");
        return 0;
}
3)、common.c
#include 
int common_bar(void)
{
        printf("this is a common function in bar/common.c\n");
        return 0;
}
3、example.c
#include 
void foo1(int a)
{
        int i;
        i = a;
}
int fun(int a, int b, int c)
{
        char buf[14];
        int sum;
        sum = a + b + c;
        foo1(a);
        return sum;
}
4、main.c
#include 
#include 
#include 
#include 
int main(void)
{
        printf("hello from %s()\n\n", __func__);
        hello_foo();
        hello_bar();
       
        print_trace(11);
}
(题外话:
这里的3个自定义头文件使用了尖括号(<>),如果按教科书的说法是不正确的,因为尖括号是从标准库的路径搜索头文件的。然而作者经常看到一些代码中标准库的头文件是使用双引号来包含的(如#include "stdio.h"),这里反其道而行之。因为在Makefile中使用-I选项指定了额外的头文件路径,使用
[latelee@latelee lib-test-latelee.org]$ gcc -g -c main.c example.c --verbose -Iconfigs
命令来编译时显示了gcc搜索头文件路径的过程,如下:
#include "..." search starts here: 
#include <...> search starts here: 
 configs
 /usr/local/include 
 /usr/lib/gcc/i386-redhat-linux/4.3.2/include 
 /usr/include 
End of search list.
当然,实际开发中不提倡这种方法,然而在U-Boot中却是十分常见的,只是有些人不明白其中的道理罢了。
题外话 结束)
库目录的Makefile(这里显示的是foo目录下的Makefile,其它者修改库名称及代码文件名称即可)
# A simple Makefile for lib(libxxx.a)  
# By Late Lee(http://www.latelee.org)  
AR = ar
ARFLAGS = cr
LIB = libfoo.a
RM = -rm -rf
OBJS := foo.o common.o

all: $(LIB)
$(LIB)$(OBJS)
        $(AR) $(ARFLAGS) $@ $(OBJS)
clean:
        $(RM) $(OBJS) $(LIB) *.bak *~
.PHONY: all clean
顶层目录的Makefile文件
#################################################################  
# A simple Makefile  
# By Late Lee(http://www.latelee.org)  

#   
# bugs:  
#      1. 需要显式指定库位置、名称;  
#      2. make 及 make clean处理得不好(对于库,要么删除再编译,要么无操作);  
#################################################################  

CC=gcc
CFLAGS = -Wall
DEBUG = y

ifeq ($(DEBUG), y)
CFLAGS += -g
else
CFLAGS += -O2
endif

SUBDIRS := foo bar bt

LIBS := bt/libbt.a foo/libfoo.a bar/libbar.a

LDFLAGS = $(LIBS)

RM = -rm -rf

__OBJS = main.o
__OBJS += example.o

__SRCS = $(subst .o,.c,$(__OBJS))

target = a.out

MAKE = make

#all: clean $(target)  
all: $(target)

$(__OBJS)$(__SRCS)
        $(CC) $(CFLAGS) -c $^ -I ./configs/

$(target)$(__OBJS)
        for dir in $(SUBDIRS); \
        do $(MAKE) -C $$dir all || exit 1; \
        done
        $(CC) $(CFLAGS) $^ -o $@ $(LDFLAGS)

clean:
        @for dir in $(SUBDIRS)do make -C $$dir clean|| exit 1; done
        $(RM) $(__OBJS) $(target) *.bak *~

.PHONY: all clean 
执行make的过程:
[latelee@latelee lib-test]$ make 
gcc -Wall -g -c main.c example.c -I ./configs/ 
example.c: In function ‘fun’:
example.c:10: warning: unused variable ‘buf’
for dir in foo bar bt  ; \ 
        do make -C $dir all || exit 1  ; \ 
        done
make[1]: Entering directory `/home/latelee/linux-c/lib-test/foo' 
cc    -c -o foo.o foo.c
cc    -c -o common.o common.c
ar cr libfoo.a foo.o common.o
make[1]: Leaving directory `/home/latelee/linux-c/lib-test/foo' 
make[1]: Entering directory `/home/latelee/linux-c/lib-test/bar' 
cc    -c -o bar.o bar.c
cc    -c -o common.o common.c
ar cr libbar.a bar.o common.o
make[1]: Leaving directory `/home/latelee/linux-c/lib-test/bar' 
make[1]: Entering directory `/home/latelee/linux-c/lib-test/bt' 
cc    -c -o backtrace.o backtrace.c
ar cr libbt.a backtrace.o
make[1]: Leaving directory `/home/latelee/linux-c/lib-test/bt' 
gcc -Wall -g main.o example.o -o a.out bt/libbt.a foo/libfoo.a bar/libbar.a
运行结果:
[latelee@latelee lib-test]$ ./a.out 
hello from main()
in func: hello_foo()
hell from bar() in file: bar.c, num:100
this is a common function in foo/common.c 
=========================================
in func: hello_bar()
hell from foo() in file: foo.c, num:200
this is a common function in bar/common.c 
=========================================
Obtained 4 stack frames.
./a.out [0x8048733] 
./a.out [0x80486cf] 
/lib/libc.so.6(__libc_start_main+0xe5) [0x7ea6d5] 
./a.out [0x8048601]
转自 http://blog.chinaunix.net/uid-22145625-id-3241368.html

zz:http://www.latelee.org/programming-under-linux/113-multi-makefile-for-app.html
0 0
原创粉丝点击