使用makefile时未定义的引用

时间:2017-09-26 02:13:28

标签: c makefile header-files

我有一个给定的makefile,由我们的教授编写。 `

SHELL  = /bin/bash
CC     = gcc
CFLAGS = -Wall -W -std=c99 -pedantic
LIBS   =

# Rajouter le nom des executables apres '=', separes par un espace.
# Si une ligne est pleine, rajouter '\' en fin de ligne et passer a la suivante.

# To compile without bor-util.c file 
EXECS = main

# To compile with bor-util.c file 
EXECSUTIL = 

# To compile with bor-util.c & bor-timer.c files
EXECSTIMER = 


.c.o :
    $(CC) -c $(CFLAGS) $*.c

help ::
    @echo "Options du make : help all clean distclean"

all :: $(EXECS) $(EXECSUTIL) $(EXECSTIMER)

$(EXECS) : %: %.o 
    $(CC) -o $@ $@.o $(LIBS)

$(EXECSUTIL) : %: %.o bor-util.o
    $(CC) -o $@ $@.o bor-util.o $(LIBS)

$(EXECSTIMER) : %: %.o bor-util.o bor-timer.o
    $(CC) -o $@ $@.o bor-util.o bor-timer.o $(LIBS)

clean ::
    \rm -f *.o core

distclean :: clean
    \rm -f *% $(EXECS) $(EXECSUTIL) $(EXECSTIMER)
`

我们在这个项目中所要做的就是在其他文件中编写代码并使用这个makefile像往常一样编译。 我写了一个helloWorld函数来测试。我有3个文件 功能.C

#include <stdio.h>
#include "functions.h"


    void printMsg(){
        printf("Hello World !");
    }

FUNCTIONS.H

#ifndef FUNCTIONS_H
#define FUNCTIONS_H

void printMsg();
#endif /* FUNCTIONS_H */

用于测试所有内容的MAIN.C文件

#include "functions.h"


int main(){

    printMsg(); 
    return 0;
}

我已将main添加到makefile中。但是当我编译

时,我收到此错误消息
gcc -o main main.o 
main.o: In function `main':
main.c:(.text+0xa): undefined reference to `printMsg'
collect2: error: ld returned 1 exit status
Makefile:32: recipe for target 'main' failed
make: *** [main] Error 1

有谁知道解决方案是什么?谢谢

2 个答案:

答案 0 :(得分:4)

错误消息明确:链接器未找到printMsg函数。这是完全正常的:执行的链接命令是:

gcc -o main main.o

请参阅?没有实现functions.o函数的printMsg的跟踪。要解决此问题,您必须使用以下命令进行链接:

gcc -o main main.o functions.o

问题是你的Makefile没有提到functions.o作为main的先决条件,它也没有在配方中使用它。要么您不理解教授的指示(谁没有要求您添加functions.cfunctions.h),要么您忘记了他还指出了如何更新Makefile,或者他的Makefile是不符合他自己的指示。在最后两种情况下,您可以通过更改$(EXECS)

的规则来调整Makefile
$(EXECS) : %: %.o functions.o
    $(CC) -o $@ $^ $(LIBS)

$^扩展为所有先决条件的列表,即在您的情况下main.o functions.o。这条新规则将:

  1. 如果mainmain.o发生更改,请重新构建functions.o
  2. 链接main.o functions.o
  3. 警告:如果$(EXECS)中列出的其他可执行文件不依赖于functions.o,或依赖于其他目标文件,或者您有更多其他文件,例如functions.o,你需要更复杂的东西。问一个新问题。

    注意:由于SO是英语,因此最好在示例代码中翻译法语注释。我建议:

      

    在&#39; =&#39;之后添加可执行文件名称,用一个空格分隔。   如果一行已满,请添加一个&#39; \&#39;最后继续下一行。

    最后一点:信件案件很重要。如果您的文件为functions.c,请不要在问题中输入FUNCTIONS.C。与其他文件名相同。

答案 1 :(得分:1)

使用这个makefile它可以工作,更容易理解/更好

GREEN   = \033[1;32m

WHITE   = \033[0;m

NAME    = main

CC      = gcc -g

RM      = rm -f

SRCS    = functions.c    \
          main.c

OBJS    = $(SRCS:.c=.o)

CFLAGS = -I ./include/
CFLAGS += -W -Wall -Wextra

all: $(NAME)

$(NAME): $(OBJS)
         @$(CC) $(OBJS) -o $(NAME) $(LDFLAGS)
         @printf "\n[$(GREEN)OK$(WHITE)] Binary : $(NAME)\n"
         @echo "-------------------\n"

%.o :    %.c
         @$(CC) $(CFLAGS) -c -o $@ $<
         @printf "[$(GREEN)OK$(WHITE)] $<\n"

clean:
        $(RM) $(OBJS)

fclean: clean
    $(RM) $(NAME)

re: fclean all

.PHONY: all clean fclean re

functions.h文件:

#ifndef FUNCTIONS_H_
#define FUNCTIONS_H_

void printMsg();

#endif /* !FUNCTIONS_H_ */

和main.c.一样。您不需要在function.c中包含“functions.h”

Allez l'om,

Nicolas :)