global reference table overflow

JNI全局引用溢出解析

https://blog.youkuaiyun.com/a396604593/article/details/105771093

ErrorLog:

    04-26 18:20:16.112 12201 12201 F DEBUG   : *** *** *** *** *** *** *** *** *** *** *** *** *** *** *** ***
    04-26 18:20:16.112 12201 12201 F DEBUG   : Build fingerprint:
    'htc/ocmdugl_00401/htc_ocmdugl:9/PQ2A.190205.003/1076299.1:user/release-keys'
    04-26 18:20:16.113 12201 12201 F DEBUG   : Revision: '0'
    04-26 18:20:16.113 12201 12201 F DEBUG   : ABI: 'arm64'
    04-26 18:20:16.113 12201 12201 F DEBUG   : pid: 1773, tid: 7880, name: Binder:1773_20  >>> system_server <<<
    04-26 18:20:16.113 12201 12201 F DEBUG   : signal 6 (SIGABRT), code -6 (SI_TKILL), fault addr --------
    04-26 18:20:16.113 12201 12201 F DEBUG   : Abort message: 'java_vm_ext.cc:676] JNI ERROR (app bug): global reference table
    overflow (max=51200)global reference table dump:'

1、百度搜索global reference table overflow (max=51200)

JNI层coding经常会遇到ReferenceTable overflow(引用表溢出造成的崩溃)问题,特别是当jni函数被反复调用上千上万次的时候。

2、log中搜索java_vm_ext.cc:676发现

48560 of com.android.server.am.PendingIntentRecord (48560 unique instances)

max=51200,而PendingIntentRecord就有 48560个。

这时如果能抓取system_server的hprof文件进行查看,会很方便的看出哪个对象导致的内存溢出。

3、如果没有hprof文件,只看log文件。

从上述log中得到pid: 1773, tid: 7880, name: Binder:1773_20,在log中

搜索Binder:1773_20:

04-26 18:20:15.119  1773  7880 F system_server: runtime.cc:558] "Binder:1773_20" prio=5 tid=135 Runnable

查看log发现

    04-26 18:20:15.121  1773  7880 F system_server: runtime.cc:558]   at android.os.Parcel.nativeWriteStrongBinder(Native method)
    04-26 18:20:15.121  1773  7880 F system_server: runtime.cc:558]   at ndroid.os.Parcel.writeStrongBinder(Parcel.java:737)
    04-26 18:20:15.121  1773  7880 F system_server: runtime.cc:558]   at android.app.IActivityManager$Stub.onTransact$getIntentSender$(IActivityManager.java:10438)
    04-26 18:20:15.121  1773  7880 F system_server: runtime.cc:558]   at android.app.IActivityManager$Stub.onTransact(IActivityManager.java:775)
    04-26 18:20:15.121  1773  7880 F system_server: runtime.cc:558]   at com.android.server.am.ActivityManagerService.onTransact(ActivityManagerService.java:3435)
    04-26 18:20:15.121  1773  7880 F system_server: runtime.cc:558]   at android.os.Binder.execTransact(Binder.java:740)

发现bind相关操作导致,这个是导致崩溃的最后一根稻草

4、搜索7880,
发现activityManager多次bindService、unbindService。

导致system_server的内存溢出。

源码分析:

JNI ERROR (app bug): global reference table overflow (max=51200)

全局引用表溢出,说明全局对象个数太多

报错的行数java_vm_ext.cc:676

相关代码:

    jobject JavaVMExt::AddGlobalRef(Thread* self, ObjPtr<mirror::Object> obj) {
      // Check for null after decoding the object to handle cleared weak globals.
      if (obj == nullptr) {
        return nullptr;
      }
      IndirectRef ref;
      std::string error_msg;
      {
        WriterMutexLock mu(self, *Locks::jni_globals_lock_);
        ref = globals_.Add(kIRTFirstSegment, obj, &error_msg);//error_msg的来源
      }
      if (UNLIKELY(ref == nullptr)) {
        LOG(FATAL) << error_msg;//报错的行数
        UNREACHABLE();
      }
      CheckGlobalRefAllocationTracking();
      return reinterpret_cast<jobject>(ref);
    }

可以看出是globals_.Add中出了问题

indirect_reference_table.cc#Add中

    if (top_index == max_entries_) {//下标是否达到最大值
        if (resizable_ == ResizableCapacity::kNo) {//是否可以调整表的容量
          std::ostringstream oss;
          oss << "JNI ERROR (app bug): " << kind_ << " table overflow "
              << "(max=" << max_entries_ << ")"
              << MutatorLockedDumpable<IndirectReferenceTable>(*this);
          *error_msg = oss.str();
          return nullptr;
    }

其中max_entries_当前值定义为51200, java_vm_ext.cc中定义了最大值如下:

static constexpr size_t kGlobalsMax = 51200;  // Arbitrary sanity check. (Must fit in 16 bits.)

每个Binder实体在jni层都会对应一个JavaBBinder实例.

bindServer时,为什么在art中会存在一个全局引用对象?

调用过程

<1>writeStrongBinder(IBinder val)

<2>android_os_Parcel_writeStrongBinder()

<3>ibinderForJavaObject(env, object)

<4> JavaBBinderHolder::get(env, obj)

<5> new JavaBBinder(env, obj);

<6> mObject(env->NewGlobalRef(object)

<7> JNI::NewGlobalRef()

<8> art::JavaVMExt::AddGlobalRef()

<9> art::IndirectReferenceTable::Add()

参考链接:

问题分析过程:

https://www.jianshu.com/p/bcc5462a78ce

https://blog.youkuaiyun.com/chi_wy/article/details/83104748

抓取hprof文件:

https://blog.youkuaiyun.com/happy_develop_/article/details/79099227

bindService启动Service与Binder服务实体的流程:android-N的

https://www.jianshu.com/p/9128f1b65586

jni文档:

https://docs.oracle.com/javase/7/docs/technotes/guides/jni/spec/functions.html
————————————————
版权声明:本文为优快云博主「纵容_伊人倩影」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.youkuaiyun.com/a396604593/article/details/105771093

@namespace: flow-my-submissions; @import (reference) '@/global/theme/variables.less'; .@{namespace} { &amp;-pageContainer { display: flex; flex-direction: column; height: 100%; overflow-x: hidden; .ant-table-wrapper { overflow: hidden; } --white-color: #fff; --primary-color: #006eff; } &amp;-proSearch { background-color: #fff; padding: 8px; border-radius: 4px; .ant-row { margin: 0 !important; .ant-col { padding-left: 0 !important; padding-right: 4px; .ant-select-sm-style(); .ant-form-item-label { padding: 0 !important; } } } } &amp;-tableContainerCurrent { min-height: 400px; // margin-top: 10px; border-radius: 4px; height: 100%; flex: 1; overflow: hidden; .ant-table { background-color: transparent; } .ant-pro-table .ant-table-wrapper .ant-pagination, .ant-table-pagination { margin-block-end: 8px !important; } .ant-table-body { background: var(--ant-color-bg-container); padding-bottom: 1px; &amp;::-webkit-scrollbar-thumb { background: rgba(48, 54, 62, 0); &amp;:hover { background: rgba(48, 54, 62, 0.3); } } &amp;:hover::-webkit-scrollbar-thumb { background: rgba(48, 54, 62, 0.5); } } .ant-table-placeholder { height: 80%; .ant-table-cell { border-bottom: 0; } } .ant-pro-card { background-clip: content-box !important; margin: 8px 0 0 0 !important; .ant-pro-card-body { position: relative; min-height: 40px !important; background-color: #fff; border-radius: 4px; .ant-pagination { margin: 10px; } .ant-pro-table-alert { position: absolute; left: 10px; bottom: 10px; margin: 0; z-index: 1; background: none; .ant-pro-table-alert-container { padding-block: 0px; padding-inline: 0px; } } } } } &amp;-buttonContainer { padding: 0; max-width: 100%; span { color: var(--primary-color); } } } less文件中@namespace这个的作用
最新发布
08-19
<think>在Less中,`@namespace`是一个变量,用于定义命名空间。在这个例子中,`@namespace: flow-my-submissions;`定义了一个名为`flow-my-submissions`的命名空间。然后,在后续的样式中,通过`&amp;`符号和插值语法`@{namespace}`来使用这个命名空间,生成带命名空间的类名。 具体来说,在代码中: ```less .@{namespace} { ... } ``` 会被编译为: ```css .flow-my-submissions { ... } ``` 这样做的目的是为了避免样式冲突,通过将一组样式封装在一个特定的命名空间下,使得这些样式只作用于拥有该命名空间的元素。 在给出的代码中,所有在`.@{namespace}`块内的样式规则,都会以`.flow-my-submissions`作为前缀。例如: ```less .@{namespace} { &amp;-pageContainer { ... } } ``` 编译后为: ```css .flow-my-submissions-pageContainer { ... } ``` 这种用法在大型项目中很常见,用于模块化CSS,确保样式的隔离性。 §§相关问题§§ 1. Less中的命名空间有什么作用?它如何帮助避免样式冲突? 2. 在Less中,`&amp;`符号有哪些常见的用法? 3. 如何通过Less的嵌套规则提高CSS代码的可维护性? 4. 在Less中,变量(如`@namespace`)的使用场景还有哪些? 5. 如何将Less代码编译成CSS?有哪些常用的工具或方法?</think>在 Less 预处理器中,`@namespace` 是一个**自定义变量**,用于定义样式命名空间的前缀。在您提供的代码中: ```less @namespace: flow-my-submissions; ``` ### 核心作用: 1. **命名空间定义** 定义了一个名为 `flow-my-submissions` 的命名空间前缀,用于封装组件样式,避免全局污染。 2. **动态类名生成** 通过插值语法 `@{namespace}` 动态生成带命名空间的类名: ```less .@{namespace} { // 编译为 .flow-my-submissions &amp;-pageContainer { // 编译为 .flow-my-submissions-pageContainer ... } &amp;-proSearch { // 编译为 .flow-my-submissions-proSearch ... } } ``` ### 实际编译结果: ```css .flow-my-submissions { /* 命名空间根样式 */ } .flow-my-submissions-pageContainer { /* 子组件样式 */ } .flow-my-submissions-proSearch { /* 子组件样式 */ } ``` ### 关键优势: 1. **避免样式冲突** 将组件所有样式封装在独立命名空间下(如 `flow-my-submissions-xxx`),防止与其他组件类名冲突。 2. **提升可维护性** - 修改命名空间只需调整 `@namespace` 变量值 - 嵌套结构清晰展示组件层级关系 3. **支持模块化开发** 符合 BEM(Block-Element-Modifier)方法论: - `Block` = `flow-my-submissions` - `Element` = `pageContainer`/`proSearch` 等 ### 技术细节: - **插值语法**:`@{var}` 是 Less 的变量插值语法,将变量值嵌入选择器 - **嵌套规则**:`&amp;` 指代父选择器,用于拼接后缀(如 `&amp;-pageContainer`) - **变量作用域**:定义在文件顶部的变量是全局的,可在整个文件中使用
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值