我有一个使用本机代码库的应用程序——碰巧是 ffmpeg 的自定义构建,尽管这并不严格相关。
到目前为止,我已将可执行文件添加到我的 res/raw 目录,然后在运行时将其解压缩到数据文件夹,然后使用 exec() 调用它。但是,从 compileSdkVersion=29 开始,由于 W^X 漏洞的新安全性,它已经停止工作。现在调用 exec() 会导致异常:
java.io.IOException: Cannot run program "..." (in directory "..."): error=13, Permission denied
我已阅读这些页面:
权限被拒绝使用 Android Q ffmpeg”:错误 = 13,权限被拒绝
https://issuetracker.google.com/issues/152645643
因此,据我了解,正确的方法是:
- 把我的库放到项目的/libs目录下
- 添加
android:extractNativeLibs = "true"
到 AndroidManifest.xml - 确保文件名与“lib*.so”模式匹配
然后安装应该将库解压缩到一个可执行位置,我应该能够在运行时读取它。
它只是不工作。我一定错过了一个重要的步骤。
我的代码
应用程序/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 files('libs/libffmpeg.so')
> dependencies { ... } 中添加 - 添加
implementation fileTree(dir: "libs", include: ["lib*.so"])
- 创建 libs/x86 和 libs/arm64 子目录并将 libffmpeg.so 放入其中
我已经展开 .apk 文件并将其解压缩,其中没有 lib 目录。就好像构建过程不知道本机库的存在并且没有将其复制进去。