【问题标题】:Auto-dependency generation in Makefile: undefined reference to `main' (mixing conda channels)Makefile 中的自动依赖生成:未定义对“main”的引用(混合 conda 通道)
【发布时间】:2024-01-09 08:01:01
【问题描述】:

我想将依赖项添加到我的Makefile 中,这样每次修改标题时都会重新编译对应的翻译单元。目前只考虑对源文件的更改。我非常关注this example

您可以在下面找到一个 MWE,它输出 undefined reference to `main'。只要变量DEPFILES 从编译步骤中删除,代码就会成功编译。

Makefile:

CC = g++
EXEC = v1_beam.exe
RM = rm -f

BASEDIR := $(shell pwd)
SRCDIR  := src
INCDIR  := include
DEPDIR  := .deps

DEPFLAGS = -M -MT $@ -MMD -MP -MF $(DEPDIR)/$*.d
DEBUG_LEVEL := -g -fdiagnostics-color=always
EXTRA_CCFLAGS := -Wall -std=c++17 -O -pedantic -pedantic-errors
CXXFLAGS        = $(DEBUG_LEVEL) $(EXTRA_CCFLAGS)
CCFLAGS         = $(CXXFLAGS)

SRCS := $(wildcard *.cc) \
    $(wildcard $(SRCDIR)/*.cc)

OBJS := $(patsubst %.cc, %.o, $(SRCS))

DEPFILES := $(patsubst %.cc, $(DEPDIR)/%.d, $(notdir $(SRCS)))

.PHONY: all clean
.DEFAULT_GOAL = all

all: $(DEPDIR) $(EXEC)

$(EXEC): $(OBJS)
    @echo 3
    @echo Dependencies: $(wildcard $(DEPFILES))
    $(CC) $(CCFLAGS) $^ -o $@
    @echo Executable $(EXEC) created.

%.o: %.cc
%.o: %.cc Makefile
    @echo 1 
    $(CC) $(DEPFLAGS) $(CCFLAGS) -c $< -I$(BASEDIR) -o $@

$(SRCDIR)/%.o: $(SRCDIR)/%.cc $(DEPDIR)/%.d | $(DEPDIR)
    @echo 2
    $(CC) $(DEPFLAGS) $(CCFLAGS) -c $< -I$(BASEDIR) -o $@

$(DEPDIR):
    mkdir -p $@

$(DEPFILES):

clean:
    $(RM) $(OBJS) $(EXEC)

include $(wildcard $(DEPFILES))

main.cc:

#include <iostream>
#include "include/dummy.h"

int main() {
  MyClass obj;
  obj.print();
  return 0;
}

include/dummy.h:

#ifndef DUMMY
#define DUMMY

#include <iostream>

class MyClass {
  MyClass() { std::cout << "constructor" << std::endl; }
  void print();
};

#endif 

src/dummy.cc:

#include "include/dummy.h"

void MyClass::print()  {
  std::cout << "print" << std::endl;
}

编辑:

问题似乎不在于Makefile(尽管应该使用-include 以避免在干净编译中出现潜在问题,其中依赖项仍然不存在),而是在我使用的conda 环境中.来自defaultconda-forge 频道的多个软件包被混合在一起。要查找其中一些包并验证它们来自不同的conda 频道,我运行:

conda list | grep gcc
conda list | grep compiler

解决问题:

conda upgrade -c conda-forge --all

从而确保所有正在使用的包都来自conda-forge 频道。

【问题讨论】:

    标签: c++ makefile conda undefined-reference conda-forge


    【解决方案1】:

    我这样做:

    ifneq (,$(wildcard ${DEPDIR}/*}
    include ${DEPDIR}/*
    endif
    

    并编译:

    DEPFLAGS = -MT $@ -MMD -MP -MF ${DEPDIR}/$*.Td
    

    我认为你的方法会有问题。我不知道干净编译会发生什么,但你的 DEPFILES 还不存在。

    【讨论】:

    • 要做一个可选的包含,你可以简单地使用-include
    • @HolyBlackCat 谢谢你的提示。事实证明Makefile 基本上是正确的,导致问题的原因是我使用的conda 安装。具体来说,一些*cxx* 包来自default 频道,而其他*gcc* 包使用conda-forge。很抱歉没有在原始问题中指定这一点,因为我不怀疑这可能是问题(我仍然想知道为什么这与Makefile 中的依赖项相关联)。我会将答案标记为正确并更新原始问题。谢谢。
    最近更新 更多