Binder 代理对象泄露问题分析

Android Framework 系列教程:yuandaimaahao.github.io/AndroidFram...

视频教程、源码、答疑服务与进入 Framework 技术交流群请联系微信 zzh0838

在 Android P 以前,一些恶意 App(更多是我们自己写的屎山代码),会短时间频繁向系统进程发起 binder 回调注册。

当客户端向服务端发起回调注册时,服务端会创建 BinderProxy 对象(Java层)与 BpBinder 对象(Native层),短时间过多的对象创建,会使得内存耗尽,导致卡顿死机等问题。

那么解决的办法就是限制对象的创建,前面我们分析过了 BinderProxy 是 native 层 BpBinder 的马甲,BinderProxy 只是一套 Java 层接口,核心功能均通过 BpBinder 提供,所以我们限制 BpBinder 对象的创建即可。

这个 Bug 在 Android P 就解决了,我们使用 Android Q 的代码来看看是怎么解决的:

cpp 复制代码
BpBinder* BpBinder::create(int32_t handle) {
    int32_t trackedUid = -1;
    if (sCountByUidEnabled) { //sCountByUidEnabled 标志变量为 true,才对 BpBinder 的创建过程做限制
        //拿到客户端的 Uid
        trackedUid = IPCThreadState::self()->getCallingUid();
        AutoMutex _l(sTrackingLock);
        //sTrackingMap 类型是 std::unordered_map<int32_t,uint32_t>
        //保存了每个 UID 创建的 BpBinder 数量
        uint32_t trackedValue = sTrackingMap[trackedUid];
        if (CC_UNLIKELY(trackedValue & LIMIT_REACHED_MASK)) { //没有超过 BpBinder 数量限制
            if (sBinderProxyThrottleCreate) { // false ,不进入
                return nullptr;
            }
        } else { //超过 BpBinder 数量限制
            if ((trackedValue & COUNTING_VALUE_MASK) >= sBinderProxyCountHighWatermark) {
                ALOGE("Too many binder proxy objects sent to uid %d from uid %d (%d proxies held)",
                      getuid(), trackedUid, trackedValue);
                sTrackingMap[trackedUid] |= LIMIT_REACHED_MASK;
                //调用注册的函数指针回调处理过多的 BpBinder 创建
                if (sLimitCallback) sLimitCallback(trackedUid);
                if (sBinderProxyThrottleCreate) {
                    ALOGI("Throttling binder proxy creates from uid %d in uid %d until binder proxy"
                          " count drops below %d",
                          trackedUid, getuid(), sBinderProxyCountLowWatermark);
                    return nullptr;
                }
            }
        }
        //记录加1
        sTrackingMap[trackedUid]++;
    }
    //创建新的
    return new BpBinder(handle, trackedUid);
}

可以看出整个流程还是比较清晰简单的:

  • 用一个 map 来保存每个 uid 创建的 BpBinder 数量
  • 每次创建 BpBinder 的时候,从 map 中找出客户端 uid 创建的 BpBinder 数量
  • 如果操作限制了,打印 Log,并调用 sLimitCallback 函数指针回调来处理

BpBinder 把处理的工作通过回调的方式扔给了 Binder 服务所在的进程,我们以 AMS 为例,来看看注册的回调是怎样的:

java 复制代码
            BinderInternal.setBinderProxyCountCallback(
                    new BinderInternal.BinderProxyLimitListener() {
                        @Override
                        public void onLimitReached(int uid) {
                            Slog.wtf(TAG, "Uid " + uid + " sent too many Binders to uid "
                                    + Process.myUid());
                            BinderProxy.dumpProxyDebugInfo();
                            if (uid == Process.SYSTEM_UID) {
                                Slog.i(TAG, "Skipping kill (uid is SYSTEM)");
                            } else {
                                killUid(UserHandle.getAppId(uid), UserHandle.getUserId(uid),
                                        "Too many Binders sent to SYSTEM");
                            }
                        }
                    }, mHandler);

AMS 的做法简单粗暴有效:

  • 打印错误信息
  • 如果客户端不是系统进程,直接杀掉
相关推荐
CV资深专家3 小时前
Android 各分区模块编译配置(mk/bp)总结
android
louisgeek5 小时前
Java 线程池取消的方式
android
Billy_Zuo5 小时前
人工智能机器学习——模型评价及优化
android·人工智能·机器学习
tangweiguo030519876 小时前
Flutter与原生混合开发:实现完美的暗夜模式同步方案
android·flutter
雨白7 小时前
深入理解 Android 触摸事件:以实现 ViewPager 为例
android
shenshizhong7 小时前
看懂鸿蒙系统源码 比较重要的知识点
android·harmonyos
一只修仙的猿9 小时前
再谈性能优化,一次项目优化经历分享
android·性能优化
雮尘11 小时前
Android性能优化之枚举替代
android
2501_9159090612 小时前
苹果上架App软件全流程指南:iOS 应用发布步骤、App Store 上架流程、uni-app 打包上传与审核技巧详解
android·ios·小程序·https·uni-app·iphone·webview
2501_9159214312 小时前
iOS 文件管理与能耗调试结合实战 如何查看缓存文件、优化电池消耗、分析App使用记录(uni-app开发与性能优化必备指南)
android·ios·缓存·小程序·uni-app·iphone·webview