带有不安全构建标志的 Swift 包依赖项:目标完整性错误
Posted
技术标签:
【中文标题】带有不安全构建标志的 Swift 包依赖项:目标完整性错误【英文标题】:Swift package dependency with unsafe build flags: target integrity error 【发布时间】:2020-11-13 18:20:44 【问题描述】:TL;DR
尝试在 Xcode 中的 ios 项目中使用本地 Swift 包依赖项时,我收到以下错误:
The package product 'DlibWrapper' cannot be used as a dependency of this target because it uses unsafe build flags.
(我使用不安全标志来指定静态库位置)
我也尝试将包导入为基于远程分支的依赖项,但它还是失败了。
根据 Swift 论坛上的this post,该问题已在不久前得到解决,相应的pull request 已被合并。
问题出现在 Swift 5.2.4 (Xcode 11.6) 和 5.3 (Xcode 12 beta 3) 中。
任何线索可能是什么问题?
详情
我正在尝试构建一个包装 dlib 库的 Swift 包,并在 iOS 应用程序中使用它。由于平台的原因,我不能使用.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
)
我尝试在module.modulemap
中使用link
属性,但编译器似乎忽略了它。此外,在目标清单中的 .linkedLibrary()
中提供库的绝对路径也无济于事,链接器会抱怨它找不到库。
任何解决方法的想法? (作为最后的手段,我可能会将所有内容打包在一个框架中)
不胜感激。
谢谢
【问题讨论】:
我也遇到了这个问题。我的解决方案是停止尝试构建第三方库的 Swift 包。我将尝试为他们构建 Xcode 项目。 【参考方案1】:更新
原来下面的方法实际上不起作用。当包被编译为静态库时,Swift 完全忽略了.linkedLibrary
(检查swift build --verbose
)。该设置仅用于动态库。然而,:
语法在后一种情况下没有任何作用——无论如何都找不到库。
(由于历史原因保留以下答案,也许它可以帮助某人找到解决方案)
旧答案
解决方法是在链接器选项中使用 clang/gcc 冒号语法,即-l:
(减去-el-冒号)。详情请见this answer。
这是包清单中的相关 sn-p:
// 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
链接器选项,类似于cxxSettings
中的headerSearchPath
。
【讨论】:
Xcode 13 允许您在被分支名称引用时使用带有不安全标志的模块。 我昨天遇到了类似的问题:github.com/Subito-it/SBTUITestTunnel/issues/133以上是关于带有不安全构建标志的 Swift 包依赖项:目标完整性错误的主要内容,如果未能解决你的问题,请参考以下文章