使用Makefile创建库文件的正确方法是什么? [C]

时间:2013-05-11 22:57:14

标签: c unix makefile

我正在为一个简单的shell系统创建我的第一个Makefile。我需要制作库文件,但由于某种原因,库部分没有成功。在错误消息中,它表示库文件不存在(显然)。

我错过了一些可以解决这个问题的明显问题吗?另外,还有其他方法可以让Makefile更有效吗?

# Beginning of Makefile

OBJS = obj/shutil.o obj/parser.o obj/sshell.o obj/history.o obj/hash_table.o obj/variables.o
HEADER_FILES = include/shell.h include/parser.h include/history.h include/hash_table.h include/variables.h
EXECUTABLE = sshell
LIBS = lib/libshell.so lib/libparser.so lib/libhistory.so lib/libhash_table.so lib/libvariables.so
LIBCFLAGS = $(CFLAGS) -D_REENTRANT -fPIC
CFLAGS = -Wall
CC = gcc
# End of configuration options

#What needs to be built to make all files and dependencies
all: $(EXECUTABLE)

#Create the main executable
$(EXECUTABLE): $(OBJS) $(LIBS)
    $(CC) -o $(EXECUTABLE) obj/sshell.o -Llib -lparser -lshell -lhistory -lhash_table -lvariables

#Create the library files
$(LIBS): $(OBJS)
    $(CC) $(LIBCFLAGS) -shared -o $(LIBS) $(OBJS)

#Recursively build object files
obj/%.o: src/%.c
    $(CC) $(CFLAGS) -I./include/ -c $< -o $@

#Define dependencies for objects based on header files
#We are overly conservative here, parser.o should depend on parser.h only
$(OBJS) : $(HEADER_FILES)

clean:
    -rm -f $(EXECUTABLE) obj/*.o lib/*.so lib/*.a
    -rm -f .sshell_history.txt

run: $(EXECUTABLE)
    (export LD_LIBRARY_PATH=lib; ./$(EXECUTABLE))

# End of Makefile

谢谢! -Lily Banks

修改

在我尝试更改它之前,这是我对库文件的看法。

$(LIBS): $(OBJS)
    $(CC) -shared -o lib/libparser.a obj/parser.o
    $(CC) -shared -o lib/libshell.a obj/shutil.o
    $(CC) -shared -o lib/libhistory.a obj/history.o
    $(CC) -shared -o lib/libhash_table.a obj/hash_table.o
    $(CC) -shared -o lib/libvariables.a obj/variables.o

问题在于它将每个文件编译五次,这根本没有效率。所以我试图做的就是一气呵成。

EDIT2:

#Create the library files
lib/libparser.so: obj/parser.o
    $(CC) $(LIBFLAGS) -shared lib/libparser.a -o $@

lib/libshell.so: obj/shutil.o
    $(CC) $(LIBFLAGS) -shared lib/libshell.a -o $@

lib/libhistory.so: obj/history.o
    $(CC) $(LIBFLAGS) -shared lib/libhistory.a -o $@

lib/libhash_table.so: obj/hash_table.o
    $(CC) $(LIBFLAGS) -shared lib/libhash_table.a -o $@

lib/variables.so: obj/variables.o
    $(CC) $(LIBFLAGS) -shared lib/libvariables.a -o $@

不幸的是,这是我得到的错误:

make: *** No rule to make target `lib/libvariables.so', needed by `sshell'.  Stop.

思想?

EDIT3:

#Create the library files
lib/libparser.so: obj/parser.o
    $(CC) $(LIBFLAGS) -shared $^ -o lib/libparser.a

lib/libshell.so: obj/shutil.o
    $(CC) $(LIBFLAGS) -shared $^ -o lib/libshell.a

lib/libhistory.so: obj/history.o
    $(CC) $(LIBFLAGS) -shared $^ -o lib/libhistory.a

lib/libhash_table.so: obj/hash_table.o
    $(CC) $(LIBFLAGS) -shared $^ -o lib/libhash_table.a

lib/libvariables.so: obj/variables.o
    $(CC) $(LIBFLAGS) -shared $^ -o lib/libvariables.a

这有效,但还有什么我需要改变的吗?感谢

2 个答案:

答案 0 :(得分:2)

这一行是完全错误的:

$(LIBS): $(OBJS)
        $(CC) $(LIBCFLAGS) -shared -o $(LIBS) $(OBJS)

如果您展开了所有变量,此行将如下所示(为清晰起见,添加换行符):

lib/libshell.so lib/libparser.so lib/libhistory.so lib/libhash_table.so lib/libvariables.so: \
            obj/shutil.o obj/parser.o   obj/sshell.o obj/history.o obj/hash_table.o obj/variables.o
        gcc -Wall -D_REENTRANT -fPIC -shared -o lib/libshell.so lib/libparser.so lib/libhistory.so \
            lib/libhash_table.so lib/libvariables.so obj/shutil.o obj/parser.o obj/sshell.o obj/history.o \
            obj/hash_table.o obj/variables.o

应该很清楚,这是非常不对的。这是不对的我甚至不能真正说出你想要完成的事情。您是否真的想为每个.o文件创建一个共享库,其中每个共享库包含一个.o?如果是这样,为什么要尝试将.o文件和共享库链接到单个可执行文件中?

如果你解释一下你真正想做的事情,在更高层次上,我们可以提供帮助。

答案 1 :(得分:2)

每个库都需要一组目标文件。并且-o标志只接受一个作为输出文件的参数,您试图输出所有库文件,而这些文件只能通过一次gcc调用来完成。

您需要执行以下操作:

lib/libshell.so: obj/sshell.o
    $(CC) $(LIBFLAGS) -shared obj/sshell.o -o lib/libshell.so

lib/libparser.so: obj/parser.o
    $(CC) $(LIBFLAGS) -shared obj/parser.o -o lib/libparser.so

每个库。