Android原生模拟器运行ARM APP

  • INSTALL_FAILED_NO_MATCHING_ABIS
  • Android原生APP安装微信(Android 6.0)

INSTALL_FAILED_NO_MATCHING_ABIS

安装APK的时候出现这个错误,其实是当你试图安装一个具有本地库的应用程序时,它没有你的CPU架构的本地库。例如,如果您为armv7编译了一个应用程序,并试图将其安装在使用英特尔架构的仿真器上,则无法正常工作。而Android Studio提供的模拟器是X86架构的,当然也有Arm架构的模拟器,只是慢得看不到开机成功的界面!
当你需要安装的APP是自己开发的APP时,你可以在build.gradle中加入以下代码(网上找的,没试过):

splits {    abi {        enable true        reset()        include 'x86', 'armeabi-v7a'        universalApk true    }}

然而当你是需要安装微信等市场上的APP时,就不行了,因为大多数的APP是ARM架构,而不是X86,所以如果需要在Android原生模拟器上运行微信等APP,则需要一个ARM 到 X86 的指令集翻译器,能把apk里的arm用的so库在运行时动态地转换成x86的指令,所有x86的Android设备里都有这个,而Intel刚好提供了这样一个东西,只不过是闭源的。

Android原生APP安装微信(Android 6.0)

先贴一堆资料:

  1. 移动测试基础 Android 模拟器 Genymotion 安装配置与 ARM 支持
  2. Android Emulator - INSTALL_FAILED_NO_MATCHING_ABIS: Failed to extract native
    libraries, res=-113
  3. 如何打开Android X86对houdini的支持
  4. Android模拟器知识以及改造
  5. Github/Rprop/libhoudini
  6. Android-X86集成houdini(Arm兼容工具)

本篇文章主要参考:https://blog.imlk.top/,反复查看这位大佬的文章之后,最后成功运行微信,感谢大佬指路。

  1. 改build.prop里的ro.product.cpu.abilist和ro.product.cpu.abilist32为x86,armeabi-v7a,armeabi,骗过包安装器,让它能把微信装上(参考第4篇文章)

  2. 改default.prop里的ro.dalvik.vm.native.bridge=0为ro.dalvik.vm.native.bridge=libhoudini.so,开启系统内的NativeBridge(参考第4篇文章)

    这里必须说一下,这个default.prop不在system.img里面,在ramdisk.img里面,ramdisk.img是只读的,只在启动的时候读一次到内存里。所以对default.prop的修改重启后会丢失,唯一的办法是手动编辑一个ramdisk.img,然后用emulator的-ramdisk选项指定修改后的ramdisk.img文件
    emulator的参数可以到官网去查看

    编辑的方法也不难,搜索cpio 就知道要怎么编辑了。
    用gunzip解压它,再用cpio解包到一个目录里
    在这目录里找到default.prop进行修改

    在封ramdisk.img的时候,不要用cpio打包,可能会有问题导致开机不了(尝试了一次,真的开不了机),用mkbootfs ./你之前解包到的目录 (可以从这下载mkbootfs https://github.com/shenyuanv/mkboot-tools )最后的ramdisk-new.img就是修改过的ramdisk.img了,在启动avd时用emulator的-ramdisk选项指定它即可。

  3. 第3篇文章里面设置里那个叫Enable native
    bridge的选项我一直没有找到,它说效果只是persist.sys.nativebridge从0改成了1,保险起见,我在build.prop里加了persist.sys.nativebridge=1(我尝试了一下,发现并没什么用)

  4. 执行enable_nativebridge
    这几篇文章里都提到了enable_nativebridge这个东西,但是我找了一番,我的AVD里面没有这个脚本,只要从网易的木木模拟器中提取出来。不过还好,在万能的github上面找到了一段enable_nativebridge,大概读一读能发现,恰好是Android7用的,而且考虑了各种情况,甚至还可以在线下载文件(显然访问不了,不过有办法解决的),链接地址:https://gist.github.com/41a5d8ba498ceecca28e9d1069a32ede

代码如下:

#!/system/bin/shPATH=/system/bin:/system/xbinhoudini_bin=0dest_dir=/system/lib$1/arm$1binfmt_misc_dir=/proc/sys/fs/binfmt_miscif [ -z "$1" ]; thenif [ "`uname -m`" = "x86_64" ]; thenv=7_yurl=http://goo.gl/SBU3iselsev=7_xurl=http://goo.gl/0IJs40fielsev=7_zurl=http://goo.gl/FDrxVNfiif [ -s /system/lib$1/libhoudini.so ]; thenlog -pi -thoudini "found /system/lib$1/libhoudini.so"elif [ -e /system/etc/houdini$v.sfs ]; thenmount /system/etc/houdini$v.sfs $dest_direlseif mountpoint -q $dest_dir; thenkill -9 `fuser -m $dest_dir`umount -f $dest_dirfimkdir -p /data/armcd /data/armwhile ! mount houdini$v.sfs $dest_dir; dowhile [ "$(getprop net.dns1)" = "" ]; dosleep 10donewget $url -cO houdini$v.sfs && continuerm -f houdini$v.sfssleep 30donefi[ -s /system/lib$1/libhoudini.so ] || mount --bind $dest_dir/libhoudini.so /system/lib$1/libhoudini.so# this is to add the supported binary formats via binfmt_miscif [ ! -e $binfmt_misc_dir/register ]; thenmount -t binfmt_misc none $binfmt_misc_dirficd $binfmt_misc_dirif [ -e register ]; then[ -e /system/bin/houdini$1 ] && dest_dir=/system/bin# register Houdini for arm binariesif [ -z "$1" ]; thenecho ':arm_exe:M::\\x7f\\x45\\x4c\\x46\\x01\\x01\\x01\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x02\\x00\\x28::'"$dest_dir/houdini:P" > registerecho ':arm_dyn:M::\\x7f\\x45\\x4c\\x46\\x01\\x01\\x01\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x03\\x00\\x28::'"$dest_dir/houdini:P" > registerelseecho ':arm64_exe:M::\\x7f\\x45\\x4c\\x46\\x02\\x01\\x01\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x02\\x00\\xb7::'"$dest_dir/houdini64:P" > registerecho ':arm64_dyn:M::\\x7f\\x45\\x4c\\x46\\x02\\x01\\x01\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x00\\x03\\x00\\xb7::'"$dest_dir/houdini64:P" > registerfiif [ -e arm${1}_exe ]; thenhoudini_bin=1fielselog -pe -thoudini "No binfmt_misc support"fiif [ $houdini_bin -eq 0 ]; thenlog -pe -thoudini "houdini$1 enabling failed!"elselog -pi -thoudini "houdini$1 enabled"fi[ "$(getprop ro.zygote)" = "zygote64_32" -a -z "$1" ] && exec $0 64exit 0

如果有兴趣进行尝试的小伙伴,可以把 log -pi -thoudini 改成echo 方便查看运行结果。

熟悉linux的小伙伴一眼就知道这是Shell。
它的作用只是下载houdini.sfs文件并挂载到so库到/system/lib/arm下,过程中挂在等等出问题的可以是文件不存在等等问题,不行就自己建立一个文件夹。
我在尝试运行这个脚本的时候发现网络有问题,而且貌似没有wget等工具
不过在github上还发现了一个仓库: https://github.com/Rprop/libhoudini,这里面各个版本的sfs文件都有,阅读脚本发现,可以手动下载这个文件然后放到/system/etc/,然后更改脚本自行挂载。
Android原生模拟器运行ARM APP_第1张图片
X 是X86,Y Z是X86_64,Z不清楚AArch64是啥!

还有个办法就是直接将所需要的so库直接push到模拟器中。
Andorid原生模拟器的默认端口为5555,可以用adb connect 127.0.0.1:5555 进行连接

当你push到系统文件夹的时候,如果提示 Read Only的话,你可以remount一下。
如果remount也不行,那就先 adb disable-verity,在重启应该就行了。

adb disable-verityadb rebootadb remount

总结步骤:

  1. 修改build.prop

    ro.product.cpu.abilist=x86,armeabi-v7a,armeabiro.product.cpu.abilist32=x86,armeabi-v7a,armeabi添加可执行 ro.enable.native.bridge.exec=1 ```
  2. push所需要的.so文件到/system/lib下:具体是arm 文件夹 以及libhoudini.so

  3. 修改ramdisk.img中的default.prop文件

     # ro.dalvik.vm.native.bridge=0 ro.dalvik.vm.native.bridge=libhoudini.so

然后就完事儿了!!!!!!最后附上结果图:

Android原生模拟器运行ARM APP_第2张图片
当这个持续很久的时候,基本上有戏!

终于安装成功
Android原生模拟器运行ARM APP_第3张图片

Android原生模拟器运行ARM APP_第4张图片
真香
Android原生模拟器运行ARM APP_第5张图片

更多相关文章

  1. Android 8.0的WIFI 架构差异分析
  2. Android studio怎样添加assets文件
  3. 【Android】(转)android模拟器命令详解
  4. 反编译和重新打包Android的dex文件
  5. Android中最重要的采用pull解析xml文件
  6. 修改Android Studio默认配置文件路径
  7. Android XML文件中的@、?、@+的该怎么理解?

随机推荐

  1. [转]微信ANDROID客户端-会话速度提升70%
  2. 创建自己的Repo Server
  3. /mnt/sdcard与Environment.getExternalSt
  4. Android(安卓)- Fragment(碎片) 详解
  5. Android网络访问的基本方法
  6. Android的四个组件类型的使用实例
  7. Android(安卓)NDK之JNI陷阱
  8. Android(安卓)logback代码配置详解
  9. flutter 学习总结
  10. 可信执行环境(TEE)技术介绍