Hotspot源码解析-第22章-vtable和itable的重新初始化

第22章-vtable和itable的重新初始化

这个函数的调用流程是

init.cpp->init_globals()

universe.cpp->universe_post_init()

bool universe_post_init() {// 防止二次初始化assert(!is_init_completed(), "Error: initialization not yet completed!");Universe::_fully_initialized = true;EXCEPTION_MARK;{ ResourceMark rm;// 解释器初始化Interpreter::initialize();      // needed for interpreter entry pointsif (!UseSharedSpaces) {HandleMark hm(THREAD);// 获取Object klass的句柄KlassHandle ok_h(THREAD, SystemDictionary::Object_klass());// 对vtable进行重新初始化,细节看`章节22.1`Universe::reinitialize_vtable_of(ok_h, CHECK_false);// 对itable重新进行初始化:vtable我们在`章节20.1 20.2`中已经讲过,itable就是针对Java接口方法的实现方法,只有普通类才有,Interface本身没有itableUniverse::reinitialize_itables(CHECK_false);}}HandleMark hm(THREAD);Klass* k;instanceKlassHandle k_h;// 创建一个空 java.lang.Class 数组Universe::_the_empty_class_klass_array = oopFactory::new_objArray(SystemDictionary::Class_klass(), 0, CHECK_false);// 解析 OutOfMemoryError 类k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_OutOfMemoryError(), true, CHECK_false);// 生成句柄k_h = instanceKlassHandle(THREAD, k);// 分配空间存放 OutOfMemoryError 内存溢出时各对应属性字段Universe::_out_of_memory_error_java_heap = k_h->allocate_instance(CHECK_false);Universe::_out_of_memory_error_metaspace = k_h->allocate_instance(CHECK_false);Universe::_out_of_memory_error_class_metaspace = k_h->allocate_instance(CHECK_false);Universe::_out_of_memory_error_array_size = k_h->allocate_instance(CHECK_false);Universe::_out_of_memory_error_gc_overhead_limit =k_h->allocate_instance(CHECK_false);Universe::_out_of_memory_error_realloc_objects = k_h->allocate_instance(CHECK_false);// 解析 NullPointerException 类及设置对应属性字段k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_NullPointerException(), true, CHECK_false);Universe::_null_ptr_exception_instance = InstanceKlass::cast(k)->allocate_instance(CHECK_false);// 解析 ArithmeticException 类及设置对应属性字段k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_ArithmeticException(), true, CHECK_false);Universe::_arithmetic_exception_instance = InstanceKlass::cast(k)->allocate_instance(CHECK_false);//  解析虚拟机错误 VirtualMachineError 类及设置对应属性字段k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_VirtualMachineError(), true, CHECK_false);bool linked = InstanceKlass::cast(k)->link_class_or_fail(CHECK_false);if (!linked) {tty->print_cr("Unable to link/verify VirtualMachineError class");return false; // initialization failed}Universe::_virtual_machine_error_instance =InstanceKlass::cast(k)->allocate_instance(CHECK_false);Universe::_vm_exception = InstanceKlass::cast(k)->allocate_instance(CHECK_false);if (!DumpSharedSpaces) {// 创建各错误/异常抛出时需要设置的 msg 信息Handle msg = java_lang_String::create_from_str("Java heap space", CHECK_false);java_lang_Throwable::set_message(Universe::_out_of_memory_error_java_heap, msg());msg = java_lang_String::create_from_str("Metaspace", CHECK_false);java_lang_Throwable::set_message(Universe::_out_of_memory_error_metaspace, msg());msg = java_lang_String::create_from_str("Compressed class space", CHECK_false);java_lang_Throwable::set_message(Universe::_out_of_memory_error_class_metaspace, msg());msg = java_lang_String::create_from_str("Requested array size exceeds VM limit", CHECK_false);java_lang_Throwable::set_message(Universe::_out_of_memory_error_array_size, msg());msg = java_lang_String::create_from_str("GC overhead limit exceeded", CHECK_false);java_lang_Throwable::set_message(Universe::_out_of_memory_error_gc_overhead_limit, msg());msg = java_lang_String::create_from_str("Java heap space: failed reallocation of scalar replaced objects", CHECK_false);java_lang_Throwable::set_message(Universe::_out_of_memory_error_realloc_objects, msg());msg = java_lang_String::create_from_str("/ by zero", CHECK_false);java_lang_Throwable::set_message(Universe::_arithmetic_exception_instance, msg());// Setup the array of errors that have preallocated backtracek = Universe::_out_of_memory_error_java_heap->klass();assert(k->name() == vmSymbols::java_lang_OutOfMemoryError(), "should be out of memory error");k_h = instanceKlassHandle(THREAD, k);int len = (StackTraceInThrowable) ? (int)PreallocatedOutOfMemoryErrorCount : 0;Universe::_preallocated_out_of_memory_error_array = oopFactory::new_objArray(k_h(), len, CHECK_false);for (int i=0; i<len; i++) {oop err = k_h->allocate_instance(CHECK_false);Handle err_h = Handle(THREAD, err);java_lang_Throwable::allocate_backtrace(err_h, CHECK_false);Universe::preallocated_out_of_memory_errors()->obj_at_put(i, err_h());}Universe::_preallocated_out_of_memory_error_avail_count = (jint)len;}// 对 Finalizer 类的处理InstanceKlass::cast(SystemDictionary::Finalizer_klass())->link_class(CHECK_false);Method* m = InstanceKlass::cast(SystemDictionary::Finalizer_klass())->find_method(vmSymbols::register_method_name(),vmSymbols::register_method_signature());if (m == NULL || !m->is_static()) {tty->print_cr("Unable to link/verify Finalizer.register method");return false; // initialization failed (cannot throw exception yet)}Universe::_finalizer_register_cache->init(SystemDictionary::Finalizer_klass(), m);// 对 Unsafe 类的处理InstanceKlass::cast(SystemDictionary::misc_Unsafe_klass())->link_class(CHECK_false);m = InstanceKlass::cast(SystemDictionary::misc_Unsafe_klass())->find_method(vmSymbols::throwIllegalAccessError_name(),vmSymbols::void_method_signature());if (m != NULL && !m->is_static()) {// Note null is okay; this method is used in itables, and if it is null,// then AbstractMethodError is thrown instead.tty->print_cr("Unable to link/verify Unsafe.throwIllegalAccessError method");return false; // initialization failed (cannot throw exception yet)}Universe::_throw_illegal_access_error_cache->init(SystemDictionary::misc_Unsafe_klass(), m);// 对 ClassLoader 类的处理InstanceKlass::cast(SystemDictionary::ClassLoader_klass())->link_class(CHECK_false);m = InstanceKlass::cast(SystemDictionary::ClassLoader_klass())->find_method(vmSymbols::addClass_name(), vmSymbols::class_void_signature());if (m == NULL || m->is_static()) {tty->print_cr("Unable to link/verify ClassLoader.addClass method");return false; // initialization failed (cannot throw exception yet)}Universe::_loader_addClass_cache->init(SystemDictionary::ClassLoader_klass(), m);// 对 ProtectionDomain 类的处理InstanceKlass::cast(SystemDictionary::ProtectionDomain_klass())->link_class(CHECK_false);m = InstanceKlass::cast(SystemDictionary::ProtectionDomain_klass())->find_method(vmSymbols::impliesCreateAccessControlContext_name(),vmSymbols::void_boolean_signature());// Allow NULL which should only happen with bootstrapping.if (m != NULL) {if (m->is_static()) {// NoSuchMethodException doesn't actually work because it tries to run the// <init> function before java_lang_Class is linked. Print error and exit.tty->print_cr("ProtectionDomain.impliesCreateAccessControlContext() has the wrong linkage");return false; // initialization failed}Universe::_pd_implies_cache->init(SystemDictionary::ProtectionDomain_klass(), m);}// The folowing is initializing converter functions for serialization in// JVM.cpp. If we clean up the StrictMath code above we may want to find// a better solution for this as well.initialize_converter_functions();// This needs to be done before the first scavenge/gc, since// it's an input to soft ref clearing policy.{MutexLocker x(Heap_lock);Universe::update_heap_info_at_gc();}// ("weak") refs processing infrastructure initializationUniverse::heap()->post_initialize();// Initialize performance counters for metaspacesMetaspaceCounters::initialize_performance_counters();CompressedClassSpaceCounters::initialize_performance_counters();MemoryService::add_metaspace_memory_pools();MemoryService::set_universe_heap(Universe::_collectedHeap);
#if INCLUDE_CDSif (UseSharedSpaces) {SharedClassUtil::initialize(CHECK_false);}
#endifreturn true;
}

22.1 vtable重新初始化

22.1.1 universe.cpp

22.1.1.1 reinitialize_vtable_of
void Universe::reinitialize_vtable_of(KlassHandle k_h, TRAPS) {// 从句柄中取出Object KlassKlass* ko = k_h();// 从Object Klass 中取出vtableklassVtable* vt = ko->vtable();// 对 Object klass 本身vtable重新初始化,继续看`章节22.1.2.1`if (vt) vt->initialize_vtable(false, CHECK);// 如果是普通对象实例,那就继续处理子类的vtableif (ko->oop_is_instance()) {InstanceKlass* ik = (InstanceKlass*)ko;// 拿到子类for (KlassHandle s_h(THREAD, ik->subklass());s_h() != NULL;// next_sibling() 沿着兄弟子类继续找s_h = KlassHandle(THREAD, s_h()->next_sibling())) {// 层层子类 vtable 重新初始化,递归reinitialize_vtable_of(s_h, CHECK);}}
}

22.1.2 klassVtable.cpp

22.1.2.1 initialize_vtable
void klassVtable::initialize_vtable(bool checkconstraints, TRAPS) {// Java的super类,那还是Object,也就是说,这里还是取的Object Klass句柄KlassHandle super (THREAD, klass()->java_super());int nofNewEntries = 0;// 判断是否共享bool is_shared = _klass->is_shared();if (PrintVtables && !klass()->oop_is_array()) {ResourceMark rm(THREAD);tty->print_cr("Initializing: %s", _klass->name()->as_C_string());}#ifdef ASSERT// 对象的限制地址oop* end_of_obj = (oop*)_klass() + _klass()->size();// vtable的限制地址oop* end_of_vtable = (oop*)&table()[_length];// vtable的限制地址 必须小于等于  对象的限制地址,`章节20.1`中有画图描述assert(end_of_vtable <= end_of_obj, "vtable extends beyond end");
#endif// 当上面的_klass 是Object klass的时候,这里就是 trueif (Universe::is_bootstrapping()) {assert(!is_shared, "sanity");// 遍历清空vtable的值for (int i = 0; i < _length; i++) table()[i].clear();return;}// 当上面的_klass 不是Object Klass,是其他普通类的 klass时,走下面的逻辑// 这一步是取出超类的vtable的长度int super_vtable_len = initialize_from_super(super);// klass不能是数组类型的,因为数组没有函数if (klass()->oop_is_array()) {assert(super_vtable_len == _length, "arrays shouldn't introduce new methods");} else {assert(_klass->oop_is_instance(), "must be InstanceKlass");// 当前 klass的方法数组Array<Method*>* methods = ik()->methods();// 当前 klass的方法数组的长度int len = methods->length();// 超类的vtable长度int initialized = super_vtable_len;// Check each of this class's methods against super;// if override, replace in copy of super vtable, otherwise append to end// 对照超类的vtable中的方法,比较当前 klass 的方法,看看有没有覆盖(重写),如果有覆盖的,那么就要在super vtable的副本中替换,否则追加到末尾for (int i = 0; i < len; i++) {// update_inherited_vtable can stop for gc - ensure using handlesHandleMark hm(THREAD);assert(methods->at(i)->is_method(), "must be a Method*");methodHandle mh(THREAD, methods->at(i));// 这里就是判断有没有覆盖(重写),返回true,就是要新建,并追加到末尾bool needs_new_entry = update_inherited_vtable(ik(), mh, super_vtable_len, -1, checkconstraints, CHECK);if (needs_new_entry) {// 放到数组末尾put_method_at(mh(), initialized);// 设置当前vtable indexmh()->set_vtable_index(initialized); // set primary vtable indexinitialized++;}}// 用默认的方法更新vtable,逻辑跟上面的一样,也是遍历并比较覆盖,默认方法的讲解,在后续类加载时会具体讲Array<Method*>* default_methods = ik()->default_methods();if (default_methods != NULL) {len = default_methods->length();if (len > 0) {Array<int>* def_vtable_indices = NULL;if ((def_vtable_indices = ik()->default_vtable_indices()) == NULL) {assert(!is_shared, "shared class def_vtable_indices does not exist");def_vtable_indices = ik()->create_new_default_vtable_indices(len, CHECK);} else {assert(def_vtable_indices->length() == len, "reinit vtable len?");}for (int i = 0; i < len; i++) {HandleMark hm(THREAD);assert(default_methods->at(i)->is_method(), "must be a Method*");methodHandle mh(THREAD, default_methods->at(i));bool needs_new_entry = update_inherited_vtable(ik(), mh, super_vtable_len, i, checkconstraints, CHECK);// needs new entryif (needs_new_entry) {put_method_at(mh(), initialized);if (is_preinitialized_vtable()) {// At runtime initialize_vtable is rerun for a shared class// (loaded by the non-boot loader) as part of link_class_impl().// The dumptime vtable index should be the same as the runtime index.assert(def_vtable_indices->at(i) == initialized,"dump time vtable index is different from runtime index");} else {def_vtable_indices->at_put(i, initialized); //set vtable index}initialized++;}}}}// 所谓 miranda methods 就是 Java Interface/abstract 接口方法,这一步就是把接口类方法添加进去if (!ik()->is_interface()) {initialized = fill_in_mirandas(initialized);}assert(initialized <= _length, "vtable initialization failed");// vtable数组长度默认最大是256,这里如果实际存在的函数不足_length,那剩下的坑位就设置为NULLfor(;initialized < _length; initialized++) {put_method_at(NULL, initialized);}NOT_PRODUCT(verify(tty, true));}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/642483.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++】list容器功能模拟实现

介绍 上一次介绍了list队容器的迭代器模拟&#xff0c;这次模拟实现list的简单功能&#xff0c;尤其要注意构造函数、析构函数、以及赋值运算符重载的实现。 list容器需要接纳所有类型的数据&#xff0c;因此&#xff0c;结构设置与迭代器设置同理&#xff0c;需要引入结点&…

Python中的open与JSON的使用

目录 1 使用 open 函数进行文件操作 2 使用 json 模块进行 JSON 数据处理&#xff1a; 2.1 写入JSON 文件 2.2 读取JSON 文件 在 Python 中&#xff0c;open 函数和 json 模块常用于文件的读写和 JSON 数据的处理。 1 使用 open 函数进行文件操作 open 函数用于打开文件…

复现NAS with RL时pytorch的相关问题

optimizer.zero_grad()是什么&#xff1f; optimizer.zero_grad()是PyTorch中的一个操作&#xff0c;它用于清零所有被优化变量&#xff08;通常是模型的参数&#xff09;的梯度。 在PyTorch中&#xff0c;当你计算某个张量的梯度时&#xff08;比如通过调用.backward()函数&…

“智汇语言·驭领未来”——系列特辑:LLM大模型信息获取与企业应用变革

“智汇语言驭领未来”——系列特辑&#xff1a;LLM大模型信息获取与企业应用变革 原创 认真的飞速小软 飞速创软 2024-01-16 09:30 发表于新加坡 本期引言 LLM&#xff08;Large Language Model&#xff09;大型语言模型以其自然语言理解和生成能力&#xff0c;正以前所未有的…

libtorch学习第六

构建卷积网络 #include<torch/torch.h> #include<torch/script.h> #include<iostream>using std::cout; using std::endl;class LinearBnReluImpl : public torch::nn::Module { private:torch::nn::Linear ln{ nullptr };torch::nn::BatchNorm1d bn{ nullp…

在jetson上对fastdeploy进行源码编译(完整版)

文章目录 准备工作下载飞桨推理库编译代码可能遇到的问题Could NOT find PythonNo CMAKE_CUDA_COMPILER could be found.准备工作 jetpackcudaopencv下载飞桨推理库 根据jetpack的版本选择对应的推理库版本 https://www.paddlepaddle.org.cn/inference/v2.6/guides/install/do…

开源的测试平台快2千星了,能带来多少收益呢

最近看了下自己去年初开源的测试平台&#xff0c;star一起算的话也到1.7k了&#xff1a; 做开源的初心一方面是想把自己的理解和思想展示出来&#xff0c;另一方面是想进一步打造个人IP&#xff0c;提升影响力&#xff08;其实这个想法很早之前就有了&#xff0c;计划过无数次但…

Day 28 | 回溯 93.复原IP地址 、78.子集 、 90.子集II

93.复原IP地址 题目 文章讲解 视频讲解 思路&#xff1a;每轮开始的位置需要变化就需要设置start class Solution {List<String> result new ArrayList<>();public List<String> restoreIpAddresses(String s) {if (s.length() < 4 ||s.length() >…

2765. 最长交替子数组 ( leetcode 01 - 23 每日 )

链接 : 最长交替子数组 思路 : 对于每个起点&#xff0c;都是x,x1,x,x1....这样的循环&#xff0c;那么枚举每个起点模拟即可 &#xff1b; 代码 : class Solution {public int alternatingSubarray(int[] nums) {int ans 0-1 ;int n nums.length , i 0 ; while(i &…

高精度运算合集,加减乘除,快速幂,详细代码,OJ链接

文章目录 零、前言一、加法高精度加法步骤P1601 AB 二、减法高精度减法步骤P2142 高精度减法 三、乘法高精度乘法步骤P1303 A*B 四、除法高精度除法步骤P1480 A/B 五、高精度快速幂麦森数 零、前言 高精度运算是某些题目涉及大数值运算且范围超出语言内置类型允许范围时采取的处…

服务器数据恢复—服务器进水导致阵列中磁盘同时掉线的数据恢复案例

服务器数据恢复环境&#xff1a; 数台服务器数台存储阵列柜&#xff0c;共上百块硬盘&#xff0c;划分了数十组lun。 服务器故障&检测&#xff1a; 外部因素导致服务器进水&#xff0c;进水服务器中一组阵列内的所有硬盘同时掉线。 北亚数据恢复工程师到达现场后发现机房内…

从0开始学习C++ 第十三课:结构体和联合体

第十三课&#xff1a;结构体和联合体 学习目标&#xff1a; 理解结构体的基本概念和如何定义结构体。学习如何使用结构体来组织相关数据。了解联合体的概念及其与结构体的不同点。 学习内容&#xff1a; 结构体 概念&#xff1a; 结构体是C中用于表示一组相关数据的复合数据…

Shopee夏季选品策略:吸引消费者,提高销售的10个有效方法

在Shopee平台上进行夏季选品时&#xff0c;卖家需要采取一些策略来吸引消费者并提高销售。本文将介绍十个有效的方法&#xff0c;帮助卖家在夏季市场做好准备&#xff0c;提供符合季节需求的产品&#xff0c;从而提高销售业绩。 先给大家推荐一款shopee知虾数据运营工具知虾免…

Vue-32、Vue单文件组件

1、School.vue <template><h1>{{schoolName}}</h1><h2>{{address}}</h2><button click"showname">点我提示学校名</button> </template> <!----> <script>export default {name: "School",d…

System.Data.SqlClient.SqlException:“在与 SQL Server 建立连接时出现与网络相关的或特定于实例的错误

目录 背景: 过程: SQL Express的认识: 背景: 正在运行程序的时候&#xff0c;我遇到一个错误提示&#xff0c;错误信息如下&#xff0c;当我将错误信息仔细阅读了一番&#xff0c;信息提示的很明显&#xff0c;错误出现的来源就是连接数据库代码这块string connStr "s…

C# 实现 希尔密码

希尔密码&#xff08;也称为Hill Cipher&#xff09;是一种经典的对称密码算法&#xff0c;用于加密和解密文本。它由美国数学家莱斯利麦保尔希尔&#xff08;Leslie McBride Hill&#xff09;于1929年提出。 希尔密码基于线性代数和矩阵运算的原理。它将明文划分为若干个长度为…

【Linux工具篇】编辑器vim

目录 vim的基本操作 进入vim(正常模式&#xff09; 正常模式->插入模式 插入模式->正常模式 正常模式->底行模式 底行模式->正常模式 底行模式->退出vim vim正常模式命令集 vim插入模式命令集 vim末行模式命令集 vim操作总结 vim配置 Linux编译器…

如何从软硬件层面优化MySQL?

如何从软硬件层面优化MySQL? 在软件方面&#xff0c;数据库性能取决于很多因素&#xff0c;比如表设计&#xff0c;查询语句以及数据库配置等。在硬件方面&#xff0c;则主要取决于CPU和IO操作。作为一名软件开发者&#xff0c;需要我们从现有的软硬件配置中获得最佳的数据库…

服务器的异步通信——RabbitMQ

目录 一、同步通信 VS 异步通信 二、MQ——消息队列 RabbitMQ RabbitMQ安装 RabbitMQ的整体架构 常见消息模型 基本消息队列&#xff08;BasicQueue&#xff09; 工作消息队列&#xff08;WorkQueue&#xff09; 发布、订阅&#xff08;Publish、Subscribe&#xff0…

CSS文本外观属性内容(知识点1)

知识引入 使用HTML可以对文本外观进行简单的控制&#xff0c;但是效果并不理想&#xff0c;为此CSS提供了一系列的文本外观样式属性&#xff0c;具体如下。 color:文本颜色 color属性用于定义文本的颜色&#xff0c;其取值方式有以下三种。 &#xff08;1&#xff09;预定义…