问题描述
我有一个使用本机代码库的应用程序-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
据我所知,正确的方法是:
- 将我的库放在项目的/ libs目录中
- 将
android:extractNativeLibs = "true"
添加到AndroidManifest.xml - 确保文件名匹配模式“ 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())
}
}
结果(在模拟器上)
/data/app/~~Q0jHO-XwEvieuwr-92_Fyg==/com.example.nativelibrarytest-byqCoqnbYsgtmCOVg361NA==/lib/x86/libffmpeg.so does not exist
为什么我的运行时代码找不到库?
我尝试过:
- 在build.gradle>依赖项{...} 中添加
- 添加
implementation fileTree(dir: "libs",include: ["lib*.so"])
- 创建libs / x86和libs / arm64子目录并将libffmpeg.so放在这些子目录中
implementation files('libs/libffmpeg.so')
我已经扩展了.apk文件并将其解压缩,并且其中没有lib目录。好像构建过程没有意识到本机库存在并且没有将其复制到其中。
解决方法
更新:经过大量的Google搜索,我认为我已经找到了答案。我的build.gradle文件中缺少ndk abiFilters
和sourceSets
。
所以我的最终项目设置是:
- 在模块的根目录下创建一个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/
-
该应用程序的
- 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