突破Android P(Preview 1)对调用隐藏API限制的方法

首先抛开Android P的具体实现过程,安卓系统要实现限制用户代码调用系统隐藏API,至少要做以下两个区分:

  1. 必须区分一个Method(或Field)对用户代码是隐藏的还是公开的。只有隐藏的才需要进行限制。
  2. 必须区分调用者的身份:是用户代码调用的还是系统代码(例如Activity类)调用的。只有用户代码调用时才需要进行限制。

具体到Android P的代码实现,它会在所有通过反射方式和JNI方式获取Method和Field的地方调用以下函数判断是否用户代码调用了系统的隐藏API(位于art/runtime/hidden_api.h),如果这个函数返回true,那么说明用户代码调用了系统的隐藏API,Android P(Preview1)会通过log发出警告,用户代码仍然能够获取到正确的Method或Field,在后续版本中获取到的Method或Field极有可能为空。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

那么它是如何进行上述两个区分的呢?

  1. 每个Method(或Field)都有一个对应的access_flags_(uint32_t类型),原本这个值通过一些特定位(bit)表明其属性(public,private,static等),但是还有一些保留的未定义的位,Android P就利用未定义的几个位,表明这个Method(或Field)是对用户代码隐藏的还是公开的。
  2. 通过回溯调用栈找到调用者所在的Class,然后判断这个Class的ClassLoader是否为BootStrapClassLoader,如果是BoootStrapClassLoader那么就认为调用者是系统代码,否则就认为调用者是用户代码。fn_caller_in_boot就是一个函数指针,它用来判断调用者是否是BootStrapClassLoader,反射调用和JNI调用时fn_caller_in_boot指向不同的函数,具体细节可查看源码。

下面我们以调用android.app.ActivityThread类的currentActivityThread这个隐藏方法为例,讲解绕过限制的方法。

三.绕过方法

绕过方法1

通过上面的论述结合源码分析,我们发现只有在通过反射方式和JNI方式获取Method和Field时,系统才有可能拦截对隐藏API的获取,也就是说直接调用是可以的!因此方法一的核心思想就是想方设法直接调用系统隐藏API。具体实现时需要用Provided方式提供Module或自定义android.jar。下面以一个例子说明实现过程。
我们新建一个普通的android工程,在其MainActivity中直接调用ActivityThread.currentActivityThread();发现IDE提示找不到类ActivityThread,这是因为在sdk的android.jar(位于SDK/platforms/android-XX目录下)中并没有这个类的声明,但是在实际运行时这个类是存在于系统中的。我们的解决方法是以Provided方式提供一个Module,在此Module中提供需要的类(Provided方式是为了编译通过,这样的Module的代码并不会编译到最终的apk中)。具体操作如下:
新建一个Module,其类型为Java Library,命名为libfakeandroid,然后在app的build.gradle中以Provided方式依赖libfakeandroid

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

之后在libfakeandroid中新建一个类android.app.ActivityThread,并添加需要调用的隐藏API,如下

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

完成以上操作之后,MainActivity中就能直接调用ActivityThread.currentActivityThread();方法了。在Android P(Preview1)系统上运行不会出现警告log,成功!
注意:如果需要调用的隐藏API所在的类已经位于android.jar中,Provided方式不再适用,此时需要自定义android.jar,将需要的Method或Field添加到android.jar中。
优点:实现起来非常简单方便,并且稳定性很好。
缺点:只能调用访问权限为public和default的Method和Field,不能直接调用protected和private的。

绕过方法2

现在回头看"限制原理"中论述的两个区分,其实只要我们能够混淆任何一个区分点都能够成功绕过此限制。混淆第一个区分点,会让系统错误地认为原本隐藏的API是公开的;混淆第二个区分点,会让系统错误地将用户代码调用识别为系统代码调用。方法二的核心思想就是混淆第二个区分点
关注第二个区分点,可以发现,其实只要在BootStrapClassLoader加载的类中有任何一个帮助我们进行反射的类就能绕过这个问题,那么我们能否将我们apk中定义的类的ClassLoader改为BootStrapClassLoader呢?答案是肯定的!查看art/runtime/mirror/class.h可知SetClassLoader函数可以为一个类指定ClassLoader,用IDA查看/system/lib/libart.so确认此函数位于导出符号表中。SetClassLoader的第一个参数类型为ObjPtrmirror::Class,如何将jclass转化为此类型呢?通过在Android源码中查找,在art/runtime/well_known_classes.h中有一个非常合适的函数ToClass能够完成此任务,其声明如下

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

查看libart.so可知,ToClass函数也在其导出符号表中,因此ToClass函数是一个恰当的函数。方法二的具体实现代码见下图

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

其中,my_dlsym与dlsym类似,其功能是根据函数的导出符号寻找函数在进程中的地址。my_dlsym是我们自定义的一个函数。
makeHiddenApiAccessable调用成功之后,使用com.test.hidefix.ReflectionHelper类反射寻找隐藏API,不会再出现log警告,成功!
实际工程中使用时可以将ReflectionHelper类作为一个工具类,代码中所有反射寻找Method和Field的地方均使用ReflectionHelper处理。注意:ReflectionHelper类只能调用系统类,不能调用自己app代码中的任何类!否则会因为ClassLoader的全盘委托机制出现问题!
优点:能够调用所有隐藏API;仅需要寻找两个导出函数,适配性较好;没有使用Hook,稳定性好
缺点:JNI方式获取Method和Field时也需要转到ReflectionHelper工具类完成

绕过方法3

结尾

我还总结出了互联网公司Android程序员面试涉及到的绝大部分面试题及答案,并整理做成了文档,以及系统的进阶学习视频资料分享给大家。
(包括Java在Android开发中应用、APP框架知识体系、高级UI、全方位性能调优,NDK开发,音视频技术,人工智能技术,跨平台技术等技术资料),希望能帮助到你面试前的复习,且找到一个好的工作,也节省大家在网上搜索资料的时间来学习。

image
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
真题+项目实战源码》点击传送门,即可获取!**

  • 26
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android中,调用相机通常涉及到使用系统的Camera API或更现代的CameraX库。以下是基本步骤: 1. **权限检查**:确保在AndroidManifest.xml文件中添加了使用相机的权限 `<uses-permission android:name="android.permission.CAMERA" />`。 2. **环境检测**:在运行时检查设备是否支持相机,是否有可用的相机。 3. **选择相机接口**: - Camera API(适用于API级别19及以上):`Camera` 类和 `CameraPreview` 用于打开相机并显示预览。 - CameraX(推荐,从API级别21开始):`CameraX` 提供了更简洁的API,如 `CameraSelector`, `CameraCaptureSession` 和 `ImageCapture`。 4. **创建相机实例**: - Camera API: 实例化 `Camera` 对象,并监听其预览。 - CameraX: 使用 `CameraSelector` 创建 `CameraProvider` 并选择相机。 5. **设置预览**: - Camera API: 设置 `SurfaceView` 为预览输出。 - CameraX: 创建 `SurfaceControl` 对象并将预览输出到该视图。 6. **拍照或录像**: - Camera API: 调用 `takePicture()` 或 `startPreview()`。 - CameraX: 使用 `CaptureRequest` 和 `CaptureSession` 进行拍摄或录像。 7. **处理结果**:对于拍照,需要在回调中处理图片数据;录像则需要处理视频流。 8. **关闭相机**: - Camera API: 释放 `Camera` 对象。 - CameraX: 使用 `cameraProvider.unbindAll()` 关闭连接。 ```java // 示例代码片段(使用CameraX) CameraX.cameraSelector(CameraSelector.DEFAULT_BACK_CAMERA) .bindToLifecycle(lifecycleOwner) .addOnSuccessListener { camera –> val preview = Preview.Builder().build() val captureSession = camera.createCaptureSession(arrayOf(preview.surface)) preview.setOnPreviewOutputUpdateListener { _, _ -> captureSession.configure(CaptureRequest.Builder().build()) } captureSession.startCapture(CaptureRequest.Builder().build(), null) } .addOnFailureListener { e –> // 处理错误 } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值