ABI 管理
大家先来看看Google官方对Android ABI管理的介绍:
不同 Android 手机使用不同的 CPU,因此支持不同的指令集。CPU 与指令集的每种组合都有其自己的应用二进制界面(或 ABI)。 ABI 可以非常精确地定义应用的机器代码在运行时如何与系统交互。 您必须为应用要使用的每个 CPU 架构指定 ABI。
典型的 ABI 包含以下信息:
- 机器代码应使用的 CPU 指令集。
- 运行时内存存储和加载的字节顺序。
- 可执行二进制文件(例如程序和共享库)的格式,以及它们支持的内容类型。
- 用于解析内容与系统之间数据的各种约定。这些约定包括对齐限制,以及系统如何使用堆栈和在调用函数时注册。
- 运行时可用于机器代码的函数符号列表 - 通常来自非常具体的库集。
本页枚举了 NDK 支持的 ABI,并且提供每个 ABI 如何运行的信息。
支持的 ABI
每个 ABI 支持一个或多个指令集。表 1 提供每个 ABI 支持的指令集概览。
表 1. ABI 和支持的指令集。
ABI | 支持的指令集 | 说明 |
---|---|---|
armeabi | ARMV5TE 和更高版本Thumb-1 | 无硬浮点。 |
armeabi-v7a | armeabiThumb-2VFPv3-D16其他(可选) | 与 ARMv5、v6 设备不兼容。 |
arm64-v8a | AArch-64 | |
x86 | x86 (IA-32)MMXSSE/2/3SSSE3 | 不支持 MOVBE 或 SSE4。 |
x86_64 | x86-64MMXSSE/2/3SSSE3SSE4.1、4.2POPCNT | |
mips | MIPS32r1 及更高版本 | 使用硬浮点,并且假设 CPU:FPU 时钟比率为 2:1 以获取最大兼容性。 不提供 micromips 或 MIPS16。 |
mips64 | MIPS64r6 |
Android 平台 ABI 支持
Android 系统在运行时知道它支持哪些 ABI,因为版本特定的系统属性会指示:
- 设备的主要 ABI,与系统映像本身使用的机器代码对应。
- 可选的辅助 ABI,与系统映像也支持的另一个 ABI 对应。
此机制确保系统在安装时从软件包提取最佳机器代码。
为实现最佳性能,应直接针对主要 ABI 进行编译。例如,基于 ARMv5TE 的典型设备只会定义主要 ABI:armeabi
。 相反,基于 ARMv7 的典型设备将主要 ABI 定义为 armeabi-v7a
,而将辅助 ABI 定义为 armeabi
,因为它可以运行为每个 ABI 生成的应用原生二进制文件。
许多基于 x86 的设备也可运行 armeabi-v7a
和 armeabi
NDK 二进制文件。对于这些设备,主要 ABI 将是 x86
,辅助 ABI 是 armeabi-v7a
。
基于 MIPS 的典型设备只定义主要 ABI:mips
。
安装时自动解压缩原生代码
安装应用时,软件包管理器服务将扫描 APK,查找以下形式的任何共享库:
lib/<primary-abi>/lib<name>.so
如果未找到,并且您已定义辅助 ABI,该服务将扫描以下形式的共享库:
lib/<secondary-abi>/lib<name>.so
找到所需的库时,软件包管理器会将它们复制到应用的 data
目录 (data/data/<package_name>/lib/
) 下的 /lib/lib<name>.so
。
如果根本没有共享对象文件,应用也会构建并安装,但在运行时会崩溃。
问题所在
armeabi-v7a一种平台的库,这种方式很好解决,直接使用Google官方提供的方案打包即可,方法如下:
1.在项目application model的build.gradle添加如下节点
android {
splits {
abi {
enable true
reset()
include 'armeabi-v7a'
universalApk true
}
}
}
参数解析
参数 | 释义 |
---|---|
splits | 拆分apk |
abi | 根据ABI拆分 |
enable | 是否启用拆分 |
include | 包含哪些ABI类型 |
universalApk | 是否生成通用的APK |
我们要解决的问题就是:如何打包多个平台的so包到同一个apk中,我提供两种解决方案解决这个问题
第一种:
1.在项目application model的build.gradle添加如下节点
android {
defaultConfig {
ndk {
abiFilters 'armeabi' ,'armeabi-v7a'
}
}
}
然后在project的根目录(和local.properties同级)找到gradle.properties文本文件,
文件里面加入下面这行
android.useDeprecatedNdk=true
第二种:
1.在项目application model的build.gradle添加如下节点
android {
packagingOptions {
exclude 'lib/armeabi/**' //exclude 'lib/armeabi-v7a/**' exclude 'lib/arm64-v8a/**' exclude 'lib/mips/**' exclude 'lib/mips64/**' //exclude 'lib/x86/**' exclude 'lib/x86_64/**'
}
}
参数解析
参数 | 释义 |
---|---|
packagingOptions | 打包配置 |
exclude | 排除掉哪些文件 |