适用于x86架构的安卓5.0以上需要root,或者离线改文件,但并非全部适用;本文仅为个人经验,未涵盖的地方需要自行探索。

手动移植Android x86的arm兼容库(houdini/ndk_translation):简略思路与移植演示_哔哩哔哩_bilibili

简略版以及移植演示,请参考上边视频。需要了解一些东西的初步概念以理解本文再谨慎动手实践,若不了解,请先在模拟环境测试环境下尝试,备份资料。

Houdini兼容库(Intel)NDK_Translation兼容库(Google)是目前x86处理器运行安卓时兼容arm应用的两种方案。本篇文章参考安卓x86各处的源码,Android-x86项目与Bliss-x86项目的一些文件,还有bliss os开发人员写的兼容库配置脚本(https://github.com/casualsnek/waydroid_script/blob/main/waydroid_extras.py)以尝试实现手动移植这些兼容库。(若要在编译x86安卓时包含兼容库,请移步其他教程)

网上已经存在Android-x86项目的arm兼容启用教程,特别对于在sfs格式的只读system系统目录情况,本篇文章将不适用,请移步网上教程。

其中有些网上的教程还会配置L3级别的widevine,这里不讲。

简述waydroid的脚本的配置过程(install_houdini()与install_ndk()部分):

  1. 挂载系统镜像,下载压缩包验证解压,扩容镜像(waydroid部分不详述)
  2. 设置兼容库文件权限为可执行
  3. 复制文件到系统镜像 
  4. 添加内容到build.prop 
  5. 添加内容到init.rc 
  6. 卸载镜像

针对以上三个序号①②③分别展开


①兼容库文件构成

(不同安卓版本的文件可能稍不同,不同的来源也可能目录结构均有不同,因此可有多种,但目录结构是由相关配置或者是动态库的预设位置关联的,因此应尽量与移植来源一致)

兼容库文件权限设置

其中lib和etc目录下,相关的文件夹权限均为drwxr-xr-x (755),相关的文件权限为-rw-r--r-- (644),bin目录下的相关可执行文件权限为-rwxr-xr-x (755)

预先说明:部分目录结构含有以下文件不会列出,会在后边详细说明

android so x86兼容arm 安卓x86 arm兼容库_目录结构

以下为兼容库文件如何构成的一些例子,移植时可以尽量参考。同时,部分兼容库可能会存放于vendor中,甚至是各种软链接,需要弄清楚它们关联

图例颜色说明:

android so x86兼容arm 安卓x86 arm兼容库_android so x86兼容arm_02

一、Houdini(经典目录结构):

android so x86兼容arm 安卓x86 arm兼容库_Android_03

二、Houdini(waydroid目录结构,linker在bin的arm文件夹中

android so x86兼容arm 安卓x86 arm兼容库_根目录_04

三、Houdini (remix os 5.1版本目录结构,houdini可执行文件以及libhoudini.so动态库在/lib/arm/下;类似的,许多安卓模拟器的houdini均在此文件夹,但libhoudini.so在原位置

android so x86兼容arm 安卓x86 arm兼容库_根目录_05

Ndk_Translation,9.0及以上,9.0目前有32位版本,9.0与11.0以上有所不同

四、Ndk_Translation(waydroid目录结构,与11.0类似)

android so x86兼容arm 安卓x86 arm兼容库_根目录_06

五、Ndk_Translation(9.0版本32位目录结构)

android so x86兼容arm 安卓x86 arm兼容库_android_07


②binfmt_misc注册arm二进制执行文件

关于这个模块,可以阅读 以了解更多。

上边已提前说明部分目录结构包含binfmt_misc,在不同的设备/系统上,这部分的支持可以在超级多的地方,但主要是一个工作:在系统初始化时,将binfmt_misc目录下的那些内容拷贝到/proc/sys/fs/binfmt_misc/register 下进行注册。以下举多种例子

一、Waydroid配置脚本的内容过程解释:

先找到init.rc(在旧版本中init.rc位于根目录/ramdisk,在新版本中init.rc位于/system/etc/init/hw/),然后将以下内容拷贝进去:

android so x86兼容arm 安卓x86 arm兼容库_目录结构_08

②binfmt_misc注册arm二进制执行文件

关于这个模块,可以阅读 以了解更多。

上边已提前说明部分目录结构包含binfmt_misc,在不同的设备/系统上,这部分的支持可以在超级多的地方,但主要是一个工作:在系统初始化时,将binfmt_misc目录下的那些内容拷贝到/proc/sys/fs/binfmt_misc/register 下进行注册。以下举多种例子

一、Waydroid配置脚本的内容过程解释:

先找到init.rc(在旧版本中init.rc位于根目录/ramdisk,在新版本中init.rc位于/system/etc/init/hw/),然后将以下内容拷贝进去:

android so x86兼容arm 安卓x86 arm兼容库_根目录_09

解释以上配置文本:在初始化早期,挂载binfmt_misc;检测具有“启用兼容库的执行=√”这个属性时,注册内容

二、谷歌AS模拟器,安卓9 x86的32位:

system\etc\init\ndk_translation.rc的内容:(与waydroid有些类似)

android so x86兼容arm 安卓x86 arm兼容库_android so x86兼容arm_10

三、chrome os (部分机型,此内容查看于guybrush机型)

安卓容器根目录中的init.bertha.rc 部分内容:

android so x86兼容arm 安卓x86 arm兼容库_android so x86兼容arm_11

四、Android-x86项目和Bliss OS等系统,部分安卓模拟器:

android so x86兼容arm 安卓x86 arm兼容库_根目录_12

 

android so x86兼容arm 安卓x86 arm兼容库_目录结构_13

可以看到Android-x86项目在系统初始化时将脚本文件enable_nativebridge设置成了一个服务,引入了一个属性persist.sys.nativebridge来控制服务的运行与终止。同时enable_nativebridge中也是关于binfmt_misc的配置。一些模拟器则会改动很多,注册的内容可能还会在其他地方

参考其中一种arm_dyn和arm_exe及其文本内容,

:arm_dyn:M::\x7f\x45\x4c\x46\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\x00\x28::/system/bin/houdini:P
:arm_exe:M::\x7f\x45\x4c\x46\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28::/system/bin/houdini:P

 由名字可猜测分别对应动态库和可执行程序,而里边的注册内容尾部也标定了arm代码由哪个文件执行。对于houdini,则指定houdini文件的存放位置;对于ndk翻译库,则指定ndk_translation_program_runner_binfmt_misc文件的存放位置。其余部分详解网上有,移植时基本上照搬即可。在移植翻译库时除了复制这些文件,也应考虑如何适配这个binfmt_misc的注册。


③build.prop, default.prop等系统属性内容配置

1. 参考waydroid配置脚本内容:

android so x86兼容arm 安卓x86 arm兼容库_Android_14

2.参考Android-x86配置内容:

android so x86兼容arm 安卓x86 arm兼容库_目录结构_15

3.参考谷歌AS模拟器9.0 x86 32位:

android so x86兼容arm 安卓x86 arm兼容库_android_16

4.参考chrome os,安卓11子系统,ndk翻译器:

android so x86兼容arm 安卓x86 arm兼容库_根目录_17

android so x86兼容arm 安卓x86 arm兼容库_android so x86兼容arm_18

总结以上例子:这些属性中,部分属性是可选的,而部分属性是必选,但其位置很可能在各种地方,如default.prop或vendor下的build.prop,以下为这些属性总结:

1.abi列表:根据需要,增加或删除abi,以及调整abi顺序

ro.product.cpu.abilist=x86_64,x86,armeabi-v7a,armeabi,arm64-v8a
ro.product.cpu.abilist32=x86,armeabi-v7a,armeabi
ro.product.cpu.abilist64=x86_64,arm64-v8a

上边中,第一行包含全部abi及优先级(arm64优先级最低),第二第三行分别对应32位abi和64位abi。这几行abi属性影响应用安装与运行,大多数情况,应用安装包根据此处判断是否支持的abi以及优先使用的abi。

2. ro.dalvik.vm.isa.arm64=x86_64和ro.dalvik.vm.isa.arm=x86

这两个,如果含有64位或32位arm翻译库,则prop应包含对应的那行属性,具体作用网上有介绍

3. ro.enable.native.bridge.exec=1

这一行属性是安卓系统启用兼容库的标志,也可能与binfmt_misc在rc文件的的注册代码有关;部分包含64位翻译库的系统包含ro.enable.native.bridge.exec64=1,这应该由系统配置,或者init.rc相关的内容决定其作用,依情况添加

4. ro.dalvik.vm.native.bridge

这一个属性很重要,其后参数代表使用的翻译库名称,指向system的lib或lib64目录下的对应文件。同时这一行属性一般可能在根目录的default.prop,部分系统则在build.prop

其后在目前有三种可能:

    ①libhoudini.so,使用houdini兼容库

    ②libndk_translation.so,使用ndk_translation兼容库

    ③libnb.so,使用Android-x86项目特有的中间层,其动态库位置由源代码指定(详细后讲)

5. persist.sys.nativebridge

Android-x86项目特有的属性,控制enable_nativebridge服务的运行与终止,以及决定libnb是否起作用。在系统设置中有其对应的开关。

6. dalvik.vm.isa.{abi}.variant;dalvik.vm.isa.{abi}.features

一般原系统已定义,不同设备(比如atom处理器的平板)可能会有不同的特殊的参数,模拟器和Android-x86一般都是默认;其中AS模拟器额外添加了arm的属性

7.其他可选

一些尚未知晓其作用的属性,如ro.ndk_translation.version,参考兼容库的移植来源。


额外说明 

1. houdini 兼容库存在x,y,z这三种版本,同时具有基本对应的安卓版本,可执行的houdini和libhoudini.so中均可获取版本号。版本号形式举个例子:

Houdini 9.0.5c_y.51331

9.0.5c为大版本号,9对应安卓9;51331为小版本号,可作为区分更细版本。中间的y即代表y版本。含义——

        x:32位系统内核的arm32翻译库(x86→arm),

        y:64位系统内核的arm32翻译库(x86_64→arm),

        z:64位系统内核的arm64翻译库(x86_64→arm64)。

NDK翻译库应该也是这么区分两种32位模式,但是难以辨别,要从移植来源确认是32位还是64位linux内核。

最前边的数字存在特例,安卓7的x版本使用的是houdini 6.1.2d_x,但部分模拟器用的libhoudini.so是7.1.1b_x(安卓9.0也能用houdini8)。此外安卓5存在houdini5.0.*和houdni5.2.*,部分使用houdini5.0.*的英特尔cpu的安卓5设备不能使用houdini5.2.*,原因是那些设备的安卓系统缺少部分源代码,以致不兼容新的houdini5.2.*。改动内容可以参考https://github.com/iConsole/Console-OS_art/commits/consoleos-lollipop

2. 部分包含houdini兼容库的Android-x86系统存在/system/vendor/etc/misc/目录,包含三个文件:

       .OEMBlackList

       .OEMWhiteList

       .ThirdPartySO

其作用在Android-x86源码可查(https://osdn.net/projects/android-x86/scm/git/frameworks-base/commits/280957b03c2c36535975d43ef5c0b0a1b8ad2c87 )需要系统代码支持,

功能简述:含有这些文件的系统,开启了“国产app兼容性检查”。这个检查会将那些有x86库但缺斤少两,或者用arm库以次充好的那些app,均安装arm的版本而不是x86版本。

       .OEMWhiteList:对这个名单的app关闭这个检查功能,有x86库就装x86

       .OEMBlackList:对这个名单的app强制安装arm版本

       .ThirdPartySO:包含第三方库的列表,以供检查

另外,adb调试可以用 adb install --abi {abi名称} {apk路径} 命令以指定abi来安装应用;而Android系统在shell可以用 pm install --abi {abi名称} {apk路径} 命令以指定abi安装应用(pm需要root权限,且低版本安卓的pm没有指定abi的功能)

3. libnb.so库:这是属于Android-x86特有的一个中间层,可以找到其源码。其内容标定了libhoudini.so的存放位置(因此remix os等可以将这个库挪到其他地方);同时源码中含有读取persist.sys.nativebridge属性,以决定是否启用兼容;源码中还含有其他调用兼容库的中间层函数,但目前仅针对houdini优化,因此,若移植ndk_translation则不应使用libnb。

Remix OS的libnb源码:https://github.com/JideTechnology/remixos-device-generic-common/blob/jide_x86_lollipop/nativebridge/src/libnb.cpp

4. 翻译库移植来源:鉴于目前可用的翻译库均是闭源产品,使用应该低调些,在以下地方等等可以提取

Houdini:

①Android-x86项目的下载链接,比如若要下载houdini5_z,则从这个网址以5_z为版本下载

http://dl.android-x86/houdini.php?v=5_z

http://dl.android-x86.org/houdini/5_z/houdini.sfs

②各种市面的安卓模拟器(夜神、雷电、逍遥等等)

③Win11的安卓子系统

④一些Android x86系统,如bliss os,prime os,以前的remix os,凤凰os等等

⑤Intel处理器的chrome os刷机包内的安卓子系统可能含有

⑥寨板、zenfone2手机等intel处理器的安卓设备(Android IA)的刷机包含有

注:houdini自8.0开始就没有发布x版本,8.0仅发布了y版本,10.0则完全没有发布

NDK_Translation:

①谷歌Android Studio模拟器,含有google api的镜像,其中安卓9的x86,安卓11的x86和x86_64,安卓12 x86_64会含有

②部分AMD处理器的chrome os设备的刷机包的安卓子系统会含有

③部分bliss os镜像,waydroid的GitHub资源等等也含有移植过来的翻译库

移植过程大致总结:

1. 获取/提取兼容库,参考兼容库目录文件构成,将文件复制过去,需要注意兼容库版本,目标系统空间剩余

2. 对复制后的兼容库文件/文件夹进行权限设置

3. 配置binfmt_misc,参考上边的过程类似;

       常见的有:

       ①根目录下的init.rc或init.{机型相关}.rc中包含部分注册内容,这些内容将/system/etc/binfmt_misc/下的配置进行注册;

       ②/system/etc/init/,或/vendor/etc/等附近与init有关的rc文件,也与①类似;

       ③Android-x86项目已通过init.rc相关的地方将脚本文件/system/bin/enable_nativebridge设置为服务,注册/配置内容均在此脚本文件中实现。(不同的模拟器改动较大,自行探索)

4. 配置属性(default.prop或build.prop等),参考上边内容