如何将本机代码库嵌入Android应用程序SDK 29+中?

问题描述

我有一个使用本机代码库的应用程序-ffmpeg的自定义构建,尽管这并不严格相关。

到目前为止,我已将可执行文件添加到我的res / raw目录中,然后在运行时将其提取到数据文件夹中,然后使用exec()对其进行调用。但是,由于compileSdkVersion = 29,由于W ^ X漏洞的新安全性,它已停止工作。现在,调用exec()会导致异常:

java.io.IOException: Cannot run program "..." (in directory "..."): error=13,Permission denied

我已阅读以下页面:
permission is denied using Android Q ffmpeg": error=13,Permission denied
https://issuetracker.google.com/issues/152645643

据我所知,正确的方法是:

  1. 将我的库放在项目的/ libs目录中
  2. android:extractNativeLibs = "true"添加到AndroidManifest.xml
  3. 确保文件名匹配模式“ lib * .so”

然后安装程序应该将库解压缩到可执行位置,并且我应该能够在运行时读取它。

这只是不起作用。我一定错过了重要的一步。

我的代码

app / build.gradle:

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'

android {
    compileSdkVersion 30
    buildToolsVersion "30.0.1"

    defaultConfig {
        applicationId "com.example.nativelibrarytest"
        minSdkVersion 29
        targetSdkVersion 30
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'),'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(dir: "libs",include: ["*.jar"])
    implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
    implementation 'androidx.core:core-ktx:1.3.1'
    implementation 'androidx.appcompat:appcompat:1.2.0'
    implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.2'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
}

AndroidManifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.nativelibrarytest">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:extractNativeLibs="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

活动代码(基于基本活动模板):

class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
    }

    override fun onResume() {
        super.onResume()

        // Test code: see if we can find the /libs library
        val libName = "libffmpeg.so"
        val nativeLibsDir = applicationInfo.nativeLibraryDir
        val libPath = "$nativeLibsDir/$libName"
        val libFile = File(libPath)
        val result = if (libFile.exists()) "exists" else "does not exist"

        val label = findViewById<TextView>(R.id.label)
        label.text = "$libPath $result"

        Log.d("MainActivity",label.text.toString())
    }
}

项目树

enter image description here

结果(在模拟器上)

/data/app/~~Q0jHO-XwEvieuwr-92_Fyg==/com.example.nativelibrarytest-byqCoqnbYsgtmCOVg361NA==/lib/x86/libffmpeg.so does not exist

为什么我的运行时代码找不到库?

我尝试过:

  • 在build.gradle>依赖项{...}
  • 中添加implementation files('libs/libffmpeg.so')
  • 添加implementation fileTree(dir: "libs",include: ["lib*.so"])
  • 创建libs / x86和libs / arm64子目录并将libffmpeg.so放在这些子目录中

我已经扩展了.apk文件并将其解压缩,并且其中没有lib目录。好像构建过程没有意识到本机库存在并且没有将其复制到其中。

解决方法

更新:经过大量的Google搜索,我认为我已经找到了答案。我的build.gradle文件中缺少ndk abiFilterssourceSets

所以我的最终项目设置是:

  1. 在模块的根目录下创建一个libs文件夹,其中包含每种体系结构的子文件夹。每个二进制文件都必须严格命名为lib<something>.so,即使它不是严格的共享库。
|app/
|-- build/
|-- libs/
|---- arm64-v8a/
|------ libffmpeg.so (compiled for 64-bit ARM)
|---- armeabi-v7a/
|------ libffmpeg.so (compiled for 32-bit ARM)
|---- x86/
|------ libffmpeg.so (compiled for 32-bit Intel)
|---- x86_64/
|------ libffmpeg.so (compiled for 64-bit Intel)
|-- src/

Final project layout

    该应用程序的
  1. build.gradle应包含以下条目
  • android> defaultConfig> ndk> abiFilters
  • android> sourceSets
  • 依赖关系>实现fileTree

这是完整的build.gradle:

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'

android {
    compileSdkVersion 30
    buildToolsVersion "30.0.1"

    defaultConfig {
        applicationId "com.example.nativelibrarytest"
        minSdkVersion 29
        targetSdkVersion 30
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"

        ndk {
            abiFilters 'armeabi-v7a','arm64-v8a','x86','x86_64'
        }
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'),'proguard-rules.pro'
        }
    }

    sourceSets {
        main {
            // let gradle pack the shared library into apk
            jniLibs.srcDirs = ['libs']
        }
    }
}

dependencies {
    implementation fileTree(dir: "libs",include: ["lib*.so"])
    implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
    implementation 'androidx.core:core-ktx:1.3.1'
    implementation 'androidx.appcompat:appcompat:1.2.0'
    implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.2'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
}

AndroidManifest.xml和活动代码与我的原始帖子相同。

Logcat结果是:

2020-09-29 19:32:50.911 9866-9866/com.example.nativelibrarytest D/MainActivity: /data/app/~~zwvTGRwVjL_Uv5d4joi4jw==/com.example.nativelibrarytest-9G5w819CD5uenr5N1jiAhg==/lib/x86/libffmpeg.so exists

相关问答

依赖报错 idea导入项目后依赖报错,解决方案:https://blog....
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下...
错误1:gradle项目控制台输出为乱码 # 解决方案:https://bl...
错误还原:在查询的过程中,传入的workType为0时,该条件不起...
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct...