Swift:如何调用从dylib加载的C函数

时间:2016-01-08 05:13:30

标签: c swift macos dylib

有没有办法调用从Swift的dylib加载的C函数?

这是我的 dylib 文件:

cppdemofile.cpp

#include "cppdemofile.h"

int add(int a, int b) {
    return a + b;
}

cppdemofile.h

#ifndef __CppDemoLibrary__cppdemofile__
#define __CppDemoLibrary__cppdemofile__

#pragma GCC visibility push(default)

extern "C" int add(int a, int b);

#pragma GCC visibility pop

#endif

编译到dylib并检查:

nm -gU libCppDemoLibrary.dylib
0000000000000f80 T _add

...将libCppDemoLibrary.dylib复制到~/lib ...

Swift程序

@IBAction func buttonClick(sender: NSButton) {
    let handle = dlopen("libCppDemoLibrary.dylib", RTLD_NOW)
    if (handle != nil) {
        var sym = dlsym(handle, "add")
        if (sym != nil) {
            let pointer = UnsafeMutablePointer<(CInt, CInt) -> CInt>(sym)

            // When debugging, I'm reaching up to this point...
            // but now, how do I call the 'add' function here???
            // var result = ???

            // label.stringValue = "Total: " + String(result)
        }
    }
}

如何调用add功能?使用dylib可以吗?我应该将这些来源添加到我的快速项目吗?

2 个答案:

答案 0 :(得分:11)

因为你,可以从Swift调用add函数 将其定义为与extern "C"建立C链接。

使库成为Swift模块(如上面的jtbandes所示) 评论)可能是更好的解决方案, 但是这里是如何使用来自Swift的dlsym()返回的函数指针:

首先添加

typedef int(*addFunc)(int, int);

到桥接头文件,或者定义

typealias addFunc = @convention(c) (CInt, CInt) -> CInt
在斯威夫特。然后以下工作:

let handle = dlopen(path, RTLD_NOW)
if (handle != nil) {
    var sym = dlsym(handle, "add")
    if (sym != nil) {
        let f = unsafeBitCast(sym, addFunc.self)
        let result = f(12, 45)
        print(result)
    }
    dlclose(handle)
}

如果addFuncif let handle = dlopen(path, RTLD_NOW) { if let sym = dlsym(handle, "add") { let f = unsafeBitCast(sym, to: addFunc.self) let result = f(12, 45) print(result) } dlclose(handle) } 不符,那么这会崩溃 加载函数的实际签名。

更新 Swift 3:

{{1}}

答案 1 :(得分:0)

为了在Swift中使用C ++代码,您需要将它包装在C函数或Objective-C类中。

另见Call a C++ function from Swift