0

我有一个使用本机代码库的应用程序——碰巧是 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

因此,据我了解,正确的方法是:

  1. 把我的库放到项目的/libs目录下
  2. 添加android:extractNativeLibs = "true"到 AndroidManifest.xml
  3. 确保文件名与“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 目录。就好像构建过程不知道本机库的存在并且没有将其复制进去。

4

1 回答 1

0

更新:经过大量谷歌搜索,我想我找到了答案。我从我的 build.gradle 文件ndk abiFilters中丢失了。sourceSets

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

  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/

最终项目布局

  1. 应用程序的 build.gradle 应包含以下条目
  • android > defaultConfig > ndk > abiFilters
  • 安卓 > 源集
  • 依赖项 > 实现文件树

这是完整的 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 和 Activity 代码与我原来的帖子没有变化。

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
于 2020-09-29T18:45:55.883 回答