android-ndk - 在 Android Studio 中将没有 header 的共享对象库与 NDK 链接

标签 android-ndk linker shared-libraries rust

我有一个共享库文件,faceblaster-engine.so,为 arm-linux-androideabi 编译,放在 Android Studio 的 jniLibs 文件夹中。我在 jni 文件夹中还有一个简单的 cpp 文件。

我的库是用 Rust 编写的,所以我没有头文件,我想通过 cpp 文件调用其中的函数,但似乎无法使库正确链接。为了测试,我做了一个简单的函数:

使用rust

#[no_mangle]
pub extern fn rust_test() -> c_int {
    82 as c_int
}

C++

extern "C" {

// Test for calling rust function
int rust_test();

jint
Java_com_fureality_faceblaster_MainActivity_testRustLaunch(JNIEnv* env, jobject thiz)
{
    return rust_test();
}

} // End extern

Android.mk

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE := faceblaster-engine
LOCAL_SRC_FILES := ../jniLibs/$(TARGET_ARCH_ABI)/libfaceblaster-engine.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE    := faceblaster
LOCAL_SRC_FILES := gl-tests.cpp
LOCAL_SHARED_LIBRARIES := faceblaster-engine
include $(BUILD_SHARED_LIBRARY)

错误

/home/nathan/Development/projects/faceblaster-android/app/src/main/jni/gl-tests.cpp

Error:(23) undefined reference to `rust_test'
Error:error: ld returned 1 exit status
make: *** [/home/nathan/Development/projects/faceblaster-android/app/build/intermediates/ndk/debug/obj/local/arm64-v8a/libfaceblaster.so] Error 1
Error:Execution failed for task ':app:compileDebugNdk'.
> com.android.ide.common.internal.LoggedErrorException: Failed to run command:
    /home/nathan/Development/bin/android-ndk-r10d/ndk-build NDK_PROJECT_PATH=null APP_BUILD_SCRIPT=/home/nathan/Development/projects/faceblaster-android/app/build/intermediates/ndk/debug/Android.mk APP_PLATFORM=android-21 NDK_OUT=/home/nathan/Development/projects/faceblaster-android/app/build/intermediates/ndk/debug/obj NDK_LIBS_OUT=/home/nathan/Development/projects/faceblaster-android/app/build/intermediates/ndk/debug/lib APP_ABI=all
  Error Code:
    2
  Output:
    /home/nathan/Development/projects/faceblaster-android/app/build/intermediates/ndk/debug/obj/local/arm64-v8a/objs/faceblaster//home/nathan/Development/projects/faceblaster-android/app/src/main/jni/gl-tests.o: In function `Java_com_fureality_faceblaster_MainActivity_testRustLaunch':
    /home/nathan/Development/projects/faceblaster-android/app/src/main/jni/gl-tests.cpp:23: undefined reference to `rust_test'
    collect2: error: ld returned 1 exit status
    make: *** [/home/nathan/Development/projects/faceblaster-android/app/build/intermediates/ndk/debug/obj/local/arm64-v8a/libfaceblaster.so] Error 1

我不知道我的 makefile 是否没有被提取,或者它是否正确?有人知道如何正确链接到此 .so 文件并注册我想使用的函数吗?

在此先感谢您的帮助!

编辑-添加Java源码和rust代码编译方法


Java

public class MainActivity extends Activity {

    // External libraries to load
    static {
        System.loadLibrary("faceblaster-engine");
        System.loadLibrary("faceblaster");
    }

    // External functions to register
    public native int testRustLaunch();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // Other stuff omitted for brevity
        Log.d(TAG, "Testing call...");
        int test = testRustLaunch();
        Log.d(TAG, "Received: " + test);
    }
}

Rust 编译

cargo build --target=arm-linux-androideabi

# /project/.cargo/config file
[target.arm-linux-androideabi]
linker = "/opt/ndk_standalone/bin/arm-linux-androideabi-gcc"

# Cargo.toml
[lib]
name = "faceblaster-engine"
crate_type = ["dylib"]

编辑2


我已经通过 build.gradle 脚本进行了编辑,我知道我的 Android.mk 现在正在被读取和使用,但我仍然遇到相同的编译错误:(

编辑3


事实证明,以下两个答案都有助于解决问题。这主要是在 Android Studio 的一部分没有获取我的 makefile,rust 代码没有被正确声明为 #[no_mangle] pub extern 和我的 makefile 都被提升了。

最佳答案

试一试:

#[no_mangle]
pub extern fn rust_test() -> i32 {
    82 // Note simplified implementation
}

具体要尝试的是#[no_mangle]pubpub 会将函数标记为可从编译库外部调用。 #[no_mangle] 指示编译器不要更改函数名称,因此导出的符号将是文字 rust_test

我还冒昧地使实际的方法体更加地道。

另一个注意事项是您应该更紧密地匹配您的 Rust 和 C 类型。如果你想在 C 中使用 int,你应该使用 Rust 类型 c_int。 C 的 int 可以根据您的平台改变大小!你也可以在 Rust 中使用 int32,但你应该在 C 中使用类似 int32_t 的东西。

关于android-ndk - 在 Android Studio 中将没有 header 的共享对象库与 NDK 链接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28161628/

相关文章:

c - 在不知道其原型(prototype)的情况下在运行时动态调用 C 函数

android - 使用 Android NDK 构建 native OpenCV 给出 "undefined reference to ' cv::String::deallocate( )'"

android - 为什么在 Android NDK 代码中 #include 时看不到包含路径?

linux - 错误: undefined reference

linux - 所有 Linux 版本都使用/etc/ld.so.preload 吗?

c - 强制链接与库不同的 SONAME

android - "Compatible side by side NDK version was not found. Default is 20.0.5594570."是什么意思?

java - 如何在android中构建CLM

c++ - 如何将 DLL 链接到我的项目?错误 LNK2019 : unresolved external symbol

插件之间的 C++ 未解析引用