8

TL;博士

尝试在 Xcode 的 iOS 项目中使用本地 Swift 包依赖项时,出现以下错误:

The package product 'DlibWrapper' cannot be used as a dependency of this target because it uses unsafe build flags.

(我使用不安全标志来指定静态库位置)

我还尝试将包作为基于远程分支的依赖项导入,但无论如何它都失败了。

根据Swift 论坛上的这篇文章,该问题已在不久前得到解决,并且相应的拉取请求已被合并。

Swift 5.2.4 (Xcode 11.6) 和 5.3 (Xcode 12 beta 3) 都会出现这个问题。

任何线索可能是什么问题?


细节

我正在尝试构建一个包装dlib库并在 iOS 应用程序中使用它的 Swift 包。由于平台的原因,我无法使用.systemLibrary目标链接dlib. 所以我将它预编译在一个静态库中,并与包装器代码一起打包,如下所示:

DlibWrapper/
  Libraries/
    dlib/
      include/
          ...
      lib/
        arm64/
          libdlib.a

  Sources/
    CWrapper/
      include/
        module.modulemap
        cwrapper.h
      cwrapper.cpp

    SwiftWrapper/
      SwiftWrapper.swift

  Package.swift

的简化内容DlibWrapper/Package.swift

// swift-tools-version:5.3

import PackageDescription

let package = Package(
    name: "DlibWrapper",
    platforms: [
        .iOS(.v13)
    ],
    products: [
        .library(
            name: "DlibWrapper",
            targets: ["CWrapper", "SwiftWrapper"])
    ],
    dependencies: [],
    targets: [
        .target(
            name: "SwiftWrapper",
            dependencies: ["CWrapper"]
        ),
        .target(
            name: "CWrapper",

            cxxSettings: [.headerSearchPath("../../Libraries/dlib/include")],

            linkerSettings: [
                .linkedLibrary("dlib"),
                .linkedFramework("Accelerate", .when(platforms: [.iOS])),

                // The error is caused by this line
                .unsafeFlags(["-LLibraries/dlib/lib/arm64"], .when(platforms: [.iOS])),
            ]
        ),
    ],
    cxxLanguageStandard: .cxx1z
)

我试图在link内部使用该属性,module.modulemap但编译器似乎忽略了它。此外,在目标清单中提供库的绝对路径.linkedLibrary()也无济于事,链接器抱怨它找不到库。

任何解决方法的想法?(作为最后的手段,我可​​能会将所有内容打包在一个框架中)

将不胜感激任何帮助。

谢谢

4

1 回答 1

2

更新

原来下面的方法实际上不起作用。.linkedLibrary当包编译为静态库时,Swift 完全忽略(检查swift build --verbose)。该设置仅用于动态库。然而,:在后一种情况下,语法没有任何作用——无论如何都找不到库。

(由于历史原因保留以下答案,也许它可以帮助某人找到解决方案)


旧答案

解决方法是在链接器选项中使用 clang/gcc 冒号语法,即-l:(minus-el-colon)。有关更多详细信息,请参阅此答案

以下是包清单中的相关片段:

// swift-tools-version:5.3

import PackageDescription

let package = Package(
    name: "DlibWrapper",
    //...
    targets: [
        //...
        .target(
            name: "CWrapper",

            cxxSettings: [.headerSearchPath("../../Libraries/dlib/include")],

            linkerSettings: [
                // Note the `:` in front of the relative path
                .linkedLibrary(":Libraries/dlib/lib/arm64/libdlib.a", .when(platforms: [.iOS])),
                //...
            ]
        ),
    ],
    //...
)

解释

链接器.linkedLibrary设置必须为链接器提供-l标志。所以,无论你放在那里作为参数,都会被翻译成这样的东西

swift build -Xlinker -l<your-linkedLibrary-string>

:语法允许使用非规范名称(即实际文件名)指定库。最后,因为链接器的工作目录设置为包根目录,我们可以为它提供库的相对路径。

我仍然认为这是一种解决方法,因为正确的方法是使用类似的东西librarySearchPath,即-L链接器选项,类似于headerSearchPathin cxxSettings

于 2020-07-24T09:13:45.610 回答