Fuse API和CMake Build

时间:2015-09-12 15:24:49

标签: python c cmake pkg-config

我最近重新开了一个我几年前工作过的项目。我写了一个小的python脚本来构建项目。我想把它移植到CMake上。

我遇到的问题是脚本在linux上使用pkg-config来查找保险丝头和库。我无法将其移植到CMake。

这是当前的python脚本

import subprocess, sys, os, shutil

def call( command ):

  c = subprocess.Popen( command.split(), stdout=subprocess.PIPE )
  c.wait()
  return c.stdout.read()

class GCC:

  args = None

  def __init__( self, initial_args ):

    self.args = initial_args

  def addPKG( self, package ):

    self.args.extend( package )

  def addFile( self, name ):

    self.args.append( name )

  def compile( self, out_name ):

    self.args.extend(["-o", out_name])
    print " ".join( self.args )
    gcc = subprocess.Popen( self.args )
    return gcc.wait() == 0


if __name__ == '__main__':

  cflags = call("pkg-config fuse --libs --cflags").split()

  print cflags

  gcc = GCC(["gcc","-g","-Wall","-pg"])
  gcc.addFile("argsparse.c")
  gcc.addFile("hidden.c")
  #gcc.addFile("fs.c")
  gcc.addFile("initialization.c")
  gcc.addFile("resolve.c")
  gcc.addFile("utilities.c")
  gcc.addFile("winhomefs0.4.c")
  gcc.addPKG(cflags)
  gcc.addFile("-lulockmgr")

  if gcc.compile("winhomefs") and 'install' in sys.argv:
    if os.getuid() == 0:
      shutil.copy("winhomefs", "/usr/local/bin/winhomefs")

这是我当前的CMakeLists.txt文件。

cmake_minimum_required (VERSION 2.8.11)
project (HomeFS)

set (CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH}
    "${CMAKE_SOURCE_DIR}/CMakeModules/")

find_package(FUSE REQUIRED)

add_library(argsparse argsparse.c)
add_library(hidden hidden.c)
add_library(initialization initialization.c)
add_library(resolve resolve.c)
add_library(utilities utilities.c)

add_executable(homefs winhomefs0.4.c)

我遇到的问题是查找保险丝部分。我尝试了几种不同的排列,包括以下内容......

https://github.com/tarruda/encfs/blob/master/CMakeModules/FindFUSE.cmake https://github.com/Pronghorn/pronghorn/blob/master/FindFUSE.cmake

我似乎没有工作:

...argsparse.c:21:22: fatal error: fuse_opt.h: No such file or directory
 #include <fuse_opt.h>
                      ^
compilation terminated.

然而,python脚本有效,这表明cmake的配置方式有问题。

作为参考,上面的pkg-config行在我的系统上输出以下内容。 -D_FILE_OFFSET_BITS=64 -I/usr/include/fuse -lfuse -pthread

感谢您的帮助!

Per Fraser的反馈我更新了两件事。 我的CMakeLists.txt现在看起来像:

cmake_minimum_required (VERSION 2.8.11)
project (HomeFS)

set (CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH}
    "${CMAKE_SOURCE_DIR}/CMakeModules/")

find_package(FUSE REQUIRED)

add_executable(homefs
    argsparse.c
    hidden.c
    initialization.c
    resolve.c
    utilities.c
    winhomefs0.4.c)

set(CMAKE_C_FLAGS "-D_FILE_OFFSET_BITS=64 -lulockmgr")

target_include_directories(homefs PRIVATE ${FUSE_INCLUDE_DIR})
target_link_libraries(homefs ${FUSE_LIBRARIES})

所有对<fuse.h><fuse_opt.h>的引用都已更新为<fuse/fuse.h>,依此类推。我还必须添加标志-D_FILE_OFFSET_BITS=64,它现在可以完全编译。

但是我仍然遇到链接器错误。

winhomefs0.4.c:(.text+0x10b2): undefined reference to `ulockmgr_op'
collect2: error: ld returned 1 exit status

我尝试将lib -lulockmgr添加到c标志但这不起作用。 谷歌还不是我的朋友,很少有人提到ulockmgr我需要实现一个FindULOCKMGR CMake模块,还是我需要在其他地方添加这条线?

好的经过一些试验和错误+逻辑思考我解决了这个问题后我需要将-lulockmgr字符串从CFLAGS移到target_link_libraries行。

1 个答案:

答案 0 :(得分:1)

您可能只是错过了CMakeLists.txt中的几个电话。

find_package(FUSE REQUIRED)将尝试查找FUSE标头和FUSE lib的路径。两个FindFUSE.cmake文件顶部的注释块提供了每个设置变量的详细信息。以encfs为例。它会将FUSE_FOUND设置为true或false,如果找不到FUSE,则可以使用有用的错误消息退出脚本。

变量FUSE_INCLUDE_DIR将设置为包含FUSE标头的文件夹的绝对路径。 FUSE_LIBRARIES将被设置为FUSE库的绝对路径列表。

您的CMakeLists.txt目前缺少的是使用这些变量。

您可以在拨打target_include_directoriestarget_link_libraries时使用它们,例如

target_include_directories(homefs PRIVATE ${FUSE_INCLUDE_DIR})
target_link_libraries(homefs ${FUSE_LIBRARIES})

另一个问题是,您通过五次add_library次调用创建了五个单独的库,但之后却没有使用它们。至少我预计会通过target_link_libraries电话将这些内容与exe相关联。

我不太了解Python以了解原始脚本正在做什么,但我认为更可能的解决方案是这些应该只是exe的一部分:

add_library(argsparse argsparse.c) add_library(hidden hidden.c) add_library(initialization initialization.c) add_library(resolve resolve.c) add_library(utilities utilities.c) add_executable(homefs argsparse.c hidden.c initialization.c resolve.c utilities.c winhomefs0.4.c)