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

QEMU源码全解析37 —— Machine(7)

接前一篇文章:QEMU源码全解析36 —— Machine(6)

本文内容参考:

《趣谈Linux操作系统》 —— 刘超,极客时间

《QEMU/KVM》源码解析与应用 —— 李强,机械工业出版社

特此致谢!

上回书讲完了qemu_create_machine函数中的第一步select_machine函数。本文讲解第2个步骤函数:MACHINE。为了便于理解,再次贴出qemu_create_machine函数代码,在softmmu/vl.c中,如下:

static void qemu_create_machine(QDict *qdict)
{MachineClass *machine_class = select_machine(qdict, &error_fatal);object_set_machine_compat_props(machine_class->compat_props);current_machine = MACHINE(object_new_with_class(OBJECT_CLASS(machine_class)));object_property_add_child(object_get_root(), "machine",OBJECT(current_machine));object_property_add_child(container_get(OBJECT(current_machine),"/unattached"),"sysbus", OBJECT(sysbus_get_default()));if (machine_class->minimum_page_bits) {if (!set_preferred_target_page_bits(machine_class->minimum_page_bits)) {/* This would be a board error: specifying a minimum smaller than* a target's compile-time fixed setting.*/g_assert_not_reached();}}cpu_exec_init_all();page_size_init();if (machine_class->hw_version) {qemu_set_hw_version(machine_class->hw_version);}/** Get the default machine options from the machine if it is not already* specified either by the configuration file or by the command line.*/if (machine_class->default_machine_opts) {QDict *default_opts =keyval_parse(machine_class->default_machine_opts, NULL, NULL,&error_abort);qemu_apply_legacy_machine_options(default_opts);object_set_properties_from_keyval(OBJECT(current_machine), default_opts,false, &error_abort);qobject_unref(default_opts);}
}

第2步代码片段如下:

current_machine = MACHINE(object_new_with_class(OBJECT_CLASS(machine_class)));

笔者开始对这个MACHINE并没有引起重视,以为在QEMU源码根目录下一搜索就能找到,结果搜索了半天,愣是找不到它是在哪里定义的。后来又查找系统目录,怀疑是在系统目录下的哪个头文件中而非QEMU源码中,结果也没有找到。又在网上搜索,也没有任何相关的结果。在参考的书和培训资料中也都是只列出了以上代码,并没有提到具体的定义……

几经周折,终于发现了一些端倪。笔者在google上搜索“X86_MACHINE”,其中一个结果链接给出(指向)了QEMU源码中include/hw/i386/x86.h的老版本源码。在此老版本源码中,是能够找到X86_MACHINE的定义的,如下所示:

由此就联想到既然X86_MACHINE的定义与X86MachineState结构在同一个文件(include/hw/i386/x86.h)中,那么MACHINE的定义就应该和MachineState在同一个文件中。于是在QEMU源码根目录下搜索“MachineState”,最终定位到include/hw/boards.h文件。

又由于X86_MACHINE宏定义所在的include/hw/i386/x86.h文件的内容由于版本更新已经和上边不同了,因此通过原本X86_MACHINE宏被定义的地方,看看新版本中对应的代码是怎样的。结果找到了这一段代码:

#define TYPE_X86_MACHINE   MACHINE_TYPE_NAME("x86")
OBJECT_DECLARE_TYPE(X86MachineState, X86MachineClass, X86_MACHINE)

这就很明显地能够看出,原本是代码“#define X86_MACHINE(obj) OBJECT_CHECK(X86MachineState, (obj),  TYPE_X86_MACHINE)”的地方,在新版本中变成了“OBJECT_DECLARE_TYPE(X86MachineState, X86MachineClass, X86_MACHINE)”。那么与之对应,在include/hw/boards.h文件中,如果也能找到这个“OBJECT_DECLARE_TYPE”关键字,那么想必其所对应的代码就是MACHINE的定义之处。

按照这个思路,在include/hw/boards.h中搜索,果然不出所料,找到了以下代码:

OBJECT_DECLARE_TYPE(MachineState, MachineClass, MACHINE)

那么毋庸置疑,这段代码中一定就包含了MACHINE的定义(尤其都已经看到MACHINE关键字了)。历尽周折,使用推理的方法终于找到了被新版本代码隐藏得很深的MACHINE宏!

关于MACHINE所“藏匿”的这个OBJECT_DECLARE_TYPE宏,在下回中做详细解析。

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

相关文章:

  • 如何将阿里云WiredTiger引擎的MongoDB物理备份文件恢复至自建数据库
  • SAP FIORI Launchpad 403 forbidden error
  • 【MongoDB】高性能非关系型数据库
  • 二、MySql库的操作
  • 【ARM 嵌入式 编译系列 10 -- GCC 编译缩减可执行文件 elf 文件大小】
  • IDEA启动报错java.nio.charset.MalformedInputException: Input length=2
  • 【Vue-Router】路由传参
  • 平板选择什么电容笔比较好?ipad手写笔推荐品牌
  • 什么是数字化车间
  • 创新零售,京东重新答题?
  • 面向对象设计与分析40讲(20)消息驱动编程和事件驱动编程模型
  • 【c语言】指针进阶(超详细)
  • C++入门篇8---vector
  • 【学会动态规划】最大子数组和(19)
  • 怎么做Tik Tok海外娱乐公会呢?新加坡市场怎么样?
  • mysql主从复制搭建
  • Java:正则表达式案例:爬数据,重复数据替换,数据分割
  • CF 765D Artsem and Saunders 构造
  • DevOps系列文章 之 SpringBoot整合GitLab-CI实现持续集成
  • K8S系列二:实战入门
  • form中表单切换,导致 relus 中的事件无法触发,原因:页面切换不要一直切换DOM,会导致问题,需要都显示出来
  • Android Ble蓝牙App(五)数据操作
  • .netcore grpc双向流方法详解
  • 【Servlet】(Servlet API HttpServlet 处理请求 HttpServletRequest 打印请求信息 前端给后端传参)
  • java中右移>>和无符号右移>>>的区别
  • 牛客周赛 Round 7
  • R语言生存分析(机器学习)(1)——GBM(梯度提升机)
  • k8s和docker简单介绍
  • Lua学习记录
  • 三分钟完美解决你的C盘内存过大爆红