当前位置: 首页 > news >正文

Android技术-修改SO导出符号

背景

经常在使用第三方SDK的时候会莫名其妙报错,其中最常见的一种就是SO符号冲突,比如libA.so静态链接了libC.a,而libB.so动态链接了libC.so。这样便会导致符号冲突。又或者在使用不同版本的动态库,也会造成符号冲突。

报错案例

  • 案例1
    DEBUG : Abort message: 'terminating with unexpected exception of type std::bad_cast: std::bad_cast'
  • 案例2
A/DEBUG: Softversion: PD1911C_A_1.9.7
A/DEBUG: Time: 2020-06-12 15:30:46
A/DEBUG: *** *** *** *** *** *** *** *** *** *** *** *** *** *** *** ***
A/DEBUG: Build fingerprint: 'vivo/PD1911/PD1911:9/PKQ1.181030.001/compiler02261735:user/release-keys'
A/DEBUG: Revision: '0'
A/DEBUG: ABI: 'arm'
A/DEBUG: pid: 18115, tid: 19792, name: Thread-158  >>> com.renderer.app <<<
A/DEBUG: signal 11 (SIGSEGV), code 0 (SI_USER), fault addr --------
A/DEBUG:     r0  c3f7e350  r1  ffffffff  r2  c3f7e31c  r3  00000014
A/DEBUG:     r4  c3f7e25c  r5  c3f7e308  r6  f2deed8c  r7  c3f7e250
A/DEBUG:     r8  c3f7e350  r9  c204a028  r10 00000000  r11 00000001
A/DEBUG:     ip  d1011f04  sp  c3f7e248  lr  d0ff766f  pc  d0ff7856
A/DEBUG: backtrace:
A/DEBUG:     #00 pc 0006e856  /data/app/com.renderer.app-Ibeyey3dw8tSdgPla9iLpg==/lib/arm/libc++_shared.so
A/DEBUG:     #01 pc 0006e66b  /data/app/com.renderer.app-Ibeyey3dw8tSdgPla9iLpg==/lib/arm/libc++_shared.so
A/DEBUG:     #02 pc 0006bffd  /data/app/com.renderer.app-Ibeyey3dw8tSdgPla9iLpg==/lib/arm/libc++_shared.so
A/DEBUG:     #03 pc 0006beaf  /data/app/com.renderer.app-Ibeyey3dw8tSdgPla9iLpg==/lib/arm/libc++_shared.so (__gxx_personality_v0+78)
A/DEBUG:     #04 pc 000e4bfc  /data/app/com.renderer.app-Ibeyey3dw8tSdgPla9iLpg==/lib/arm/libpg_sdk.so
A/DEBUG:     #05 pc 000e5740  /data/app/com.renderer.app-Ibeyey3dw8tSdgPla9iLpg==/lib/arm/libpg_sdk.so

解决方案

基于以上背景,可能存在一下几种解决方案

  1. 源代码重编译SO,添加动态依赖或者-fPIC等
  2. 使用dlopen等函数进行动态调用
  3. 修改导出符号表,将冲突的函数隐藏掉
  4. 修改导出符号表,将冲突的函数重命名

具体执行

  1. 如果没有源代码,这种方案就行不通
  2. 可行方案。如果动态库没有导出 C 接口,直接使用 dlopen 和 dlsym 调用 C++ 接口不现实。需要再封装一个动态库 lib_wapper.so,包装 C++ 方法调用,导出 C 接口。程序里使用 dlopen 和 dlsym 调用 lib_wapper.so 导出的 C 接口。
  3. 不可行。使用 ELFkickers 里的 rebind 工具更改符号表,将方法可见性从DEFAULT修改为HIDDEN,运行时仍然与系统库冲突。查阅资料发现,rebind 主要是用于修改 .o 目标文件。可以修改 .o 目标文件的绑定属性和符号可见性,修改 .so 无效
  4. 可行。使用Python的LIEF模块最为方便

实际案例

解决跟libc++_shared.so符号冲突的so重命名

import lief
import sysif __name__ == '__main__':cpp_shared_so_path = "libc++_shared.so"cpp_shared_so = lief.parse(cpp_shared_so_path)cpp_shared_so_exported_symbols = cpp_shared_so.exported_symbolscpp_shared_so_exported_symbols_list = [sym.name for sym in cpp_shared_so_exported_symbols]targetElf = sys.argv[1]elf = lief.parse(targetElf)exported_symbols = elf.exported_symbols #dynamic_symbolsfor symbol in exported_symbols:if symbol.name in cpp_shared_so_exported_symbols_list:sym_export = elf.get_symbol(symbol.name)sym_export.name = symbol.name.replace("_Z","_M")elf.write(targetElf.replace(".so","_out.so"))

参考

- 安卓符号冲突解决方案,修改动态库函数符号表
- 修改so导出函数名称

http://www.lryc.cn/news/221899.html

相关文章:

  • flutter 打包apk
  • Halcon如何使用SaperaLT库连接dalsa相机
  • Vue 嵌套路由 多级路由规则
  • pandas教程:Introduction to pandas Data Structures pandas的数据结构
  • MinIO 分布式文件(对象)存储
  • HTML表单标签
  • 【黑马程序员】SpringCloud——Eureka
  • 目标跟踪(DeepSORT)
  • 2 任务2: 使用趋动云GPU进行猫狗识别实践
  • 技术分享 | app自动化测试(Android)--显式等待机制
  • 机器学习基础之《回归与聚类算法(5)—分类的评估方法》
  • 如何在macbook上删除文件?Mac删除文件的多种方法
  • Java代码Demo——Map根据key或value排序
  • 一个Linux自动备份脚本的示例
  • [论文阅读]PV-RCNN++
  • 测试老鸟整理,Postman加密接口测试-Rsa/Aes对参数加密(详细总结)
  • JavaScript使用对象
  • 微带线的ABCD矩阵的推导、转换与级联-Matlab计算实例
  • “网站不安全”该如何解决
  • gitlab数据备份和恢复
  • 嵌入式Linux和stm32区别? 之间有什么关系吗?
  • 【Redis】String字符串类型-内部编码使用场景
  • 电脑发热发烫,具体硬件温度达到多少度才算异常?
  • 计算机网络第4章-IPv6和寻址
  • Lazarus安装和入门资料
  • mediapipe流水线分析 二
  • 1.性能优化
  • 使用Plsql+oracle client 连接 Oracle数据库
  • centos获取服务器公网ip
  • 思谋科技进博首秀:工业多模态大模型IndustryGPT V1.0正式发布