macOS上的Clang无法从ncurses链接lmenu

时间:2019-06-16 19:35:58

标签: c macos menu clang ncurses

我对使用ncurses库是陌生的,因此我一直在尝试重新创建此页面上的一些示例,http://www.tldp.org/HOWTO/NCURSES-Programming-HOWTO/index.html

我已经获得了有关创建菜单的部分,特别是示例21。我编写的程序可以在Linux(特别是Ubuntu 18.04)上运行,但是当我使用Menu库时无法编译。我使用JUST ncurses编写的所有其他示例程序都可以正常编译而没有问题,这只是在我尝试使用Menu库时。

我用于在Linux和macOS上构建的命令是

gcc libmenutest.c -o test -lmenu -lncurses

我尝试移动-lmenu -lncurses并将macOS上的顺序更改为未成功。我已经通过brew安装了ncurses,并尝试使用brew中的gcc-8,但那里也没有成功。

我正在运行全新安装的macOS和最新的命令行工具。我可以在/usr/lib中看到libmenu,与libncurses一样。所以我很困惑为什么编译器找不到它。

这是我一直在尝试诊断问题的一些测试代码。

#include <curses.h>
#include <menu.h>
#include <stdlib.h>

#define ARRAY_SIZE(a) (sizeof a / sizeof a[0])

int main (void)
{
  int i;
  int nchoices;

  char *choices[] = {
    "Choice 1", "Choice 2", "Choice 3", "Exit", (char *) NULL,
  };

  // Test that the types are present, this should test for the include headers
  ITEM **items;
  MENU *menu;
  WINDOW *win;

  // This will test for includes and to see if libncurses can be linked
  initscr ();
  noecho ();
  cbreak ();
  keypad (stdscr, TRUE);

  // this bit will test for libmenu include and if it can be linked
  nchoices = ARRAY_SIZE (choices);
  items = calloc (nchoices, sizeof (ITEM *));
  if (items == NULL) exit (1);
  for (i = 0; i < nchoices; i++)
    items[i] = new_item (choices[i], choices[i]);

  // write smarmy message to screen :^^^^^^)
  printw ("This worked :^)");
  refresh ();
  getch ();

  // clean up
  for (i = 0; i < nchoices; i++)
    free_item (items[i]);
  endwin ();

  return 0;
}

这是我现在得到的输出...

Undefined symbols for architecture x86_64:
  "_free_item", referenced from:
      _main in libmenutest-0f0c39.o
  "_new_item", referenced from:
      _main in libmenutest-0f0c39.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

2 个答案:

答案 0 :(得分:1)

您可以执行以下操作:

brew install ncurses

由于macOS已经包含ncurses版本,brew将其替代版本安装在/usr/local/opt/ncurses中。

为了使编译器和链接器可以访问它,构建命令现在应如下所示:

gcc -I/usr/local/opt/ncurses/include -L/usr/local/opt/ncurses/lib libmenutest.c -o test -lmenu -lncurses

当您最终调用程序时,将输出以下内容:

This worked :^) 

CMake

对于使用CMake的人,您的CMakeLists.txt可能看起来像这样:

cmake_minimum_required(VERSION 3.14)
project(libmenutest C)

set(CMAKE_C_STANDARD 99)

include_directories(/usr/local/opt/ncurses/include)

link_directories(/usr/local/opt/ncurses/lib)

add_executable(libmenutest libmenutest.c)

target_link_libraries(libmenutest menu ncurses)

答案 1 :(得分:0)

同样的问题。我终于用以下命令解决了它:

  1. 编译为位码

clang -c -o libmenutest.o libmenutest.c

  1. 没有 -syslibroot 的链接,它会自动生成

ld -demangle -lto_library /Library/Developer/CommandLineTools/usr/lib/libLTO.dylib -no_deduplicate -dynamic -arch x86_64 -platform_version macos 10.15.0 10.15.6 -lSystem /Library/Developer/CommandLineTools/usr/lib/clang/12.0.0/lib/darwin/libclang_rt.osx.a -o a.out libmenutest.o -lmenu -lncurses

为什么上面的代码有效? ncurses 库的默认搜索路径为:

-lmenu => /usr/lib/libmenu.dylib

-lncurses => /usr/lib/ncurses.dylib

使用 -syslibroot /Library/Developer/CommandLineTools/SDKs/MacOSX10.15.sdk,它们变成:

-lmenu => /Library/Developer/CommandLineTools/SDKs/MacOSX10.15.sdk/usr/lib/libmenu.dylib

-lncurses => /Library/Developer/CommandLineTools/SDKs/MacOSX10.15.sdk/usr/lib/ncurses.dylib

这里是 ld 人文档:

<块引用>

ld 维护用于搜索库或框架的目录列表 使用。默认库搜索路径是 /usr/lib 然后是 /usr/local/lib。 -L 选项将添加新的库搜索路径。默认框架 搜索路径是 /Library/Frameworks 然后是 /System/Library/Frameworks。 (注意:之前,/Network/Library/Frameworks 位于 默认路径。如果您需要该功能,则需要明确添加 -F/网络/库/框架)。 -F 选项将添加一个新框架 搜索路径。 -Z 选项将删除标准搜索路径。这 -syslibroot 选项将为所有搜索路径添加前缀。