使用PKG_CHECK_MODULES时,autoconf生成的Makefile不传递库头的标志

时间:2012-04-18 11:36:43

标签: autotools autoconf automake pkg-config

我的项目取决于库(更准确地说,GTK +)所以我在configure.ac中添加了以下配置:

PKG_CHECK_MODULES([GTK], [gtk+-2.0])
AC_SUBST([GTK_CFLAGS])
AC_SUBST([GTK_LIBS])

我的Makefile.am是:

bin_PROGRAMS = secretary
secretary_SOURCES = secretary.c

反过来,我的secretary.c如下:

#include <gtk/gtk.h>

int main(int argc, char *argv[]) {
    gtk_init(&argc, &argv);
    GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_widget_show(window);
    gtk_main();
    return 0;
}

但是,当我运行make时(当然,在调用./configure之后),我收到了此错误:

gcc -DHAVE_CONFIG_H -I.     -g -O2 -MT secretary.o -MD -MP -MF .deps/secretary.Tpo -c -o secretary.o secretary.c
secretary.c:1:21: fatal error: gtk/gtk.h: File or directory not found.

我错过了什么?为什么autoconf没有将正确的标志传递给gcc?

2 个答案:

答案 0 :(得分:2)

使用PKG_CHECK_MODULES时,需要在Makefile.am中指定标志。最简单的方法是将其添加到AM_LDFLAGS和AM_CPPFLAGS:

AM_LDADD = @GTK_LIBS@
AM_CPPFLAGS = @GTK_CFLAGS@

如果您想更具体,可以添加:

secretary_LDADD = @GTK_LIBS@
secretary_CPPFLAGS = @GTK_CFLAGS@

根本不使用PKG_CHECK_MODULES可能更容易,让用户通过通常的机制指定库的位置(分配LDFLAGS或在标准位置安装库)。

答案 1 :(得分:2)

从@William Pursell的建议开始,我找到了一个解决方案。这个答案有点冗长,因为我觉得有必要证明为什么我不接受这个有用的帖子作为答案。

注意如果您正在寻找一些魔术线,请跳到最后的“解决方案”部分。

尝试提出的解决方案

我尝试过William Pursell解决方案,但发现了一个问题:GCC 4.6.1在处理某些参数的排序时特别苛刻。所以当我设置变量如下:

secretary_CPPFLAGS = @GTK_CFLAGS@ # DOES NOT WORK!
secretary_LDFLAGS = @GTK_LIBS@    # DOES NOT WORK!

我得到了以下gcc调用行:

gcc -std=gnu99  -g -O2 -pthread -lgtk-x11-2.0 -lgdk-x11-2.0 -latk-1.0 -lgio-2.0 \
    -lpangoft2-1.0 -lpangocairo-1.0 -lgdk_pixbuf-2.0 -lcairo -lpango-1.0 \
    -lfreetype lfontconfig -lgobject-2.0 -lgmodule-2.0 -lgthread-2.0 -lrt \
    -lglib-2.0    -o secretary secretary-secretary.o  

.o目标代码之前将库传递给编译器。海湾合作委员会不接受它并给了我这个错误:

secretary-secretary.o: In function `main':
/home/adam/software/secretary-gtk/secretary.c:4: undefined reference to `gtk_init'
/home/adam/software/secretary-gtk/secretary.c:5: undefined reference to `gtk_window_new'
/home/adam/software/secretary-gtk/secretary.c:6: undefined reference to `gtk_widget_show'
/home/adam/software/secretary-gtk/secretary.c:7: undefined reference to `gtk_main'

继续研究

寻找解决方案,我发现@ uidzer0有the same problem并解决了它 - 但没有发布全面的解释......所以我去查看his project。我查看了configure.ac,其中我找到了PKG_CHECK_MODULES

的用法
PKG_CHECK_MODULES([FUSE], [fuse >= 2.8.3])
PKG_CHECK_MODULES([GLIB], [glib-2.0 >= 2.22.5])
PKG_CHECK_MODULES([GTHREAD], [gthread-2.0])
PKG_CHECK_MODULES([CURL], [libcurl >= 7.16.0])

所以我查找了生成的变量(FUSE_LIBS等)的使用位置。我在src/Makefile.am文件中找到了它们:

stormfs_CFLAGS = -D_REENTRANT \
                 -DFUSE_USE_VERSION=26 \
                 -D_FILE_OFFSET_BITS=64 \
                 -DSYSCONFDIR=\"${sysconfdir}\" \
                 ${FUSE_CFLAGS} \
                 ${CURL_CFLAGS} \
                 ${GLIB_CFLAGS} \
                 ${GTHREAD_CFLAGS}
stormfs_LDADD = ${LIBS} \
                ${FUSE_LIBS} \
                ${CURL_LIBS} \
                ${GLIB_LIBS} \
                ${GTHREAD_LIBS}

解决方案

所以我得出结论,我不应该设置*_CPPFLAGS / *_LDFLAGS,而应设置*_CFLAGS*_LDADD标志。我得到的(工作)配置是:

bin_PROGRAMS = secretary
secretary_SOURCES = secretary.c
secretary_CFLAGS = @GTK_CFLAGS@
secretary_LDADD = @GTK_LIBS@