Gradle在链接期间失败,涉及NDK和CMake构建。

5

我正在构建一个非常复杂的Android库,使用Gradle Android Build Tools v2.2.0,它链接到十几个预构建的.a文件,输出应该是一个.so文件。在尝试从ndk-build转换到cmake时,我无法正确地链接.so文件,因为生成的ninja构建似乎无法找到静态库的头文件。

CMakeLists.txt

cmake_minimum_required(VERSION 3.4.1)

set(CMAKE_VERBOSE_MAKEFILE on)

include_directories(${CMAKE_SOURCE_DIR}
    ${CMAKE_SOURCE_DIR}/prebuilt/include )

add_library(precompiledA STATIC IMPORTED)
set_target_properties(precompiledA PROPERTIES 
    IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/prebuilt/dcmtk/lib/libprecompiledA.a)

add_library(precompiledB STATIC IMPORTED)
set_target_properties(precompiledB PROPERTIES 
    IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/prebuilt/dcmtk/lib/libprecompiledB.a)

add_library(precompiledC STATIC IMPORTED)
set_target_properties(precompiledC PROPERTIES 
    IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/prebuilt/dcmtk/lib/libprecompiledC.a)



add_library( jni-library

             SHARED

             hello.cpp )

find_library( log-lib
              log )

target_link_libraries( jni-library 

                       precompiledA
                       precompiledB
                       precompiledC

                       ${log-lib} )

build.gradle

apply plugin: 'com.android.library'

android {
    compileSdkVersion 24
    buildToolsVersion "24.0.2"

    defaultConfig {
        minSdkVersion 19
        targetSdkVersion 24

        ndk {
            abiFilters 'armeabi-v7a'
        }

        externalNativeBuild {
            cmake {
                arguments '-DANDROID_STL=gnustl_static',
                          '-DANDROID_CPP_FEATURES=exceptions'
            }
        }
    }

    externalNativeBuild {
        cmake {
            path 'src/main/cpp/CMakeLists.txt'
        }
    }
}

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
}

错误 - 简介

ld: error: cannot find -lprecompiledA
undefined references galore...

当调用gradle时,会生成.o文件,但在链接时,与静态库关联的头文件似乎丢失了,导致链接失败。是否有选项告诉cmake在链接时如何找到静态库的头文件?


1
我不是gradle的专家,但错误信息“cannot find -lprecompiledA”看起来像是CMake将传递给“target_link_libraries”的库解释为常规(系统)库,并忽略了之前的“add_library(IMPORTED)”调用。但是发布的CMake代码似乎是正确的...你所说的“headers”是什么?链接器根本不使用头文件,它们完全在编译阶段处理。 - Tsyvarev
1个回答

2

纠正未定义的引用

我犯的错误是没有使用target_include_directories,没有给库提供绝对路径以及忘记使用${ANDROID_ABI}变量获取正确版本的二进制文件。

cmake_minimum_required( VERSION 3.4.1 )

set( CMAKE_VERBOSE_MAKEFILE on )

add_library( precompiledA STATIC IMPORTED )
set_target_properties( precompiledA 
                       PROPERTIES 
                       IMPORTED_LOCATION
                       ${CMAKE_SOURCE_DIR}/prebuilt_libs/mylib/lib/${ANDROID_ABI}/libprecompiledA.a )

add_library( precompiledB STATIC IMPORTED )
set_target_properties( precompiledB 
                       PROPERTIES 
                       IMPORTED_LOCATION
                       ${CMAKE_SOURCE_DIR}/prebuilt_libs/mylib/lib/${ANDROID_ABI}/libprecompiledB.a )

add_library( precompiledC STATIC IMPORTED )
set_target_properties( precompiledC 
                       PROPERTIES 
                       IMPORTED_LOCATION
                       ${CMAKE_SOURCE_DIR}/prebuilt_libs/mylib/lib/${ANDROID_ABI}/libprecompiledC.a )

find_library( log-lib
              log )


target_include_directories( dicom-jni PRIVATE
                            prebuilt_libs/mylib/${ANDROID_ABI}/include )

add_library( jni-library
             SHARED
             hello.cpp )

target_link_libraries( jni-library 

                       precompiledA
                       precompiledB
                       precompiledC

                       ${log-lib} )

1
谢谢,这是一个简洁的示例,演示如何将其他预构建库添加到我的NDK项目中。提醒其他人注意:在target_link_libraries()中使用precompiledA而不是${precompiledA}!我花了将近2个小时才找到这个问题 :( - Zhiyong

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接