QEMU源码全解析15 —— QOM介绍(4)

88 篇文章 19 订阅
本文深入解析QEMU中的QOM(对象模型)类型注册过程,包括pci_edu_register_types函数和type_register系列函数。通过分析TypeInfo结构和TypeImpl的创建与哈希表插入,阐述QEMU如何保存和管理对象类型信息。QOM在C语言中实现面向对象,提供了类似C++、Java中类的功能。
摘要由CSDN通过智能技术生成

接前一篇文章: QEMU源码全解析14 —— QOM介绍(3)

本文内容参考:

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

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

特此致谢!

上一回讲到pci_edu_register_types函数,本回接着往下深入分析,并且完成对QOM第一部分——类型的注册的解析。qom/object.c

为了便于理解,再次贴出pci_edu_register_types函数代码,在在hw/misc/edu.c中,如下:

  1. static void pci_edu_register_types(void)
  2. {
  3. static InterfaceInfo interfaces[] = {
  4. { INTERFACE_CONVENTIONAL_PCI_DEVICE },
  5. { },
  6. };
  7. static const TypeInfo edu_info = {
  8. .name = TYPE_PCI_EDU_DEVICE,
  9. .parent = TYPE_PCI_DEVICE,
  10. .instance_size = sizeof(EduState),
  11. .instance_init = edu_instance_init,
  12. .class_init = edu_class_init,
  13. .interfaces = interfaces,
  14. };
  15. type_register_static(&edu_info);
  16. }
  17. type_init(pci_edu_register_types)

pci_edu_register_types函数唯一的工作是构造了一个TypeInfo类型的edu_info,并将其作为参数调用了type_register_static。实际上在之前回目中已经对于type_register_static函数进行过详细分析,可以参看: QEMU源码全解析10 —— 定义一个QEMU模块(2) 。还是那句话,这里才从QOM的角度再次讲解一下,权当是加深理解了。

type_register_static函数调用type_register函数,最终到达type_register_internal函数,核心工作在此函数中进行(可以参阅: QEMU源码全解析11 —— 定义一个QEMU模块(3) )。

以上几个函数都是在qom/object.c中,代码如下:

  1. static TypeImpl *type_register_internal(const TypeInfo *info)
  2. {
  3. TypeImpl *ti;
  4. ti = type_new(info);
  5. type_table_add(ti);
  6. return ti;
  7. }
  8. TypeImpl *type_register(const TypeInfo *info)
  9. {
  10. assert(info->parent);
  11. return type_register_internal(info);
  12. }
  13. TypeImpl *type_register_static(const TypeInfo *info)
  14. {
  15. return type_register(info);
  16. }

可见,这几个函数是按照调用关系一个在一个上边。

type_register_internal函数的参数TypeInfo表示的是类型信息。其中的parent成员表示的是父类型的名字;inistance_size和instance_init成员分别表示该类型对应的实例大小以及实例的初始化函数;class_init成员表示该类型的类初始化函数。

type_register_internal函数很简单,type_new函数首先通过一个TypeInfo结构构造出一个TypeImpl,然后通过type_table_add函数将这个TypeImpl加入到一个哈希表中。此哈希表的key是TypeImpl的名字,value为TypeImpl本身的值。这一过程完成了从TypeInfo到TypeImpl的转变,并且将其插入到了一个哈希表中。TypeImpl的数据基本上都是从TypeInfo复制过来的,表示的是一个类型的基本信息。

在C++、Java中,可以使用class关键字定义一个类型。 QEMU 使用C语言实现面向对象时,也必须保存对象的类型信息。因此,在TypeInfo里面指定了类型的基本信息,然后在初始化的时候复制到TypeImpl的哈希表中。

TypeImpl中存放了类型的所有信息,其定义如下(qom/object.c中):

  1. typedef struct InterfaceImpl InterfaceImpl;
  2. typedef struct TypeImpl TypeImpl;
  3. struct InterfaceImpl
  4. {
  5. const char *typename;
  6. };
  7. struct TypeImpl
  8. {
  9. const char *name;
  10. size_t class_size;
  11. size_t instance_size;
  12. size_t instance_align;
  13. void (*class_init)(ObjectClass *klass, void *data);
  14. void (*class_base_init)(ObjectClass *klass, void *data);
  15. void *class_data;
  16. void (*instance_init)(Object *obj);
  17. void (*instance_post_init)(Object *obj);
  18. void (*instance_finalize)(Object *obj);
  19. bool abstract;
  20. const char *parent;
  21. TypeImpl *parent_type;
  22. ObjectClass *class;
  23. int num_interfaces;
  24. InterfaceImpl interfaces[MAX_INTERFACES];
  25. };

下面对TypeImpl结构中的成员进行基本介绍:

  • name —— 表示类型的名字,如"edu"、"isa-i8259"等。
  • class_size —— 表示所属类的大小。
  • instance_size —— 表示该类实例的大小。
  • class_init、class_base_init —— 函数指针,表示类的初始化函数。这类函数只会在类初始化的时候进行调用。
  • instance_init、instance_post_init、instance_finalize:函数指针,表示该类所属实例相关的初始化与销毁函数。
  • abstract —— 表示类型是否是抽象的,与C++中的abstract类型相似,抽象类型不能直接创建实例,只能创建其子类所属实例。
  • parent —— 表示父类型的名字。
  • parent_type —— 表示父类型对应的类型信息。这是一个TypeImpl。
  • class —— 是一个指向ObjectClass的指针,保存了该类型的基本信息。
  • num_interfaces —— 表示接口数量。
  • interfaces —— 描述类型的接口信息。与Java语言中的接口类似,接口是一种特殊的抽象类型。

至此,QOM的第一部分 —— 类型的注册就介绍完了。

举报

选择你想要举报的内容(必选)
  • 内容涉黄
  • 政治相关
  • 内容抄袭
  • 涉嫌广告
  • 内容侵权
  • 侮辱谩骂
  • 样式问题
  • 其他