QEMU源码全解析 —— PCI设备模拟(3)

接前一篇文章:

2. PCI设备的模拟

QEMU模拟的设备很多都是PCI设备,本节介绍PCI设备的模拟。与所有设备类似,PCI设备的父设备也是TYPE_DEVICE,其定义在QEMU源码根目录/hw/pci/pci.c中,代码如下:

static const TypeInfo pci_device_type_info = {.name = TYPE_PCI_DEVICE,.parent = TYPE_DEVICE,.instance_size = sizeof(PCIDevice),.abstract = true,.class_size = sizeof(PCIDeviceClass),.class_init = pci_device_class_init,.class_base_init = pci_device_class_base_init,
};static void pci_register_types(void)
{type_register_static(&pci_bus_info);type_register_static(&pcie_bus_info);type_register_static(&cxl_bus_info);type_register_static(&conventional_pci_interface_info);type_register_static(&cxl_interface_info);type_register_static(&pcie_interface_info);type_register_static(&pci_device_type_info);
}type_init(pci_register_types)

上一回讲到了PCI设备的具现化函数pci_qdev_realize(),讲解了pci_qdev_realize函数所完成的三个任务中的第一个,本回接着往下讲。为了便于理解,再次贴出该函数源码,在hw/pci/pci.c中,如下:

static void pci_qdev_realize(DeviceState *qdev, Error **errp)
{PCIDevice *pci_dev = (PCIDevice *)qdev;PCIDeviceClass *pc = PCI_DEVICE_GET_CLASS(pci_dev);ObjectClass *klass = OBJECT_CLASS(pc);Error *local_err = NULL;bool is_default_rom;uint16_t class_id;/** capped by systemd (see: udev-builtin-net_id.c)* as it's the only known user honor it to avoid users* misconfigure QEMU and then wonder why acpi-index doesn't work*/if (pci_dev->acpi_index > ONBOARD_INDEX_MAX) {error_setg(errp, "acpi-index should be less or equal to %u",ONBOARD_INDEX_MAX);return;}/** make sure that acpi-index is unique across all present PCI devices*/if (pci_dev->acpi_index) {GSequence *used_indexes = pci_acpi_index_list();if (g_sequence_lookup(used_indexes,GINT_TO_POINTER(pci_dev->acpi_index),g_cmp_uint32, NULL)) {error_setg(errp, "a PCI device with acpi-index = %" PRIu32" already exist", pci_dev->acpi_index);return;}g_sequence_insert_sorted(used_indexes,GINT_TO_POINTER(pci_dev->acpi_index),g_cmp_uint32, NULL);}if (pci_dev->romsize != -1 && !is_power_of_2(pci_dev->romsize)) {error_setg(errp, "ROM size %u is not a power of two", pci_dev->romsize);return;}/* initialize cap_present for pci_is_express() and pci_config_size(),* Note that hybrid PCIs are not set automatically and need to manage* QEMU_PCI_CAP_EXPRESS manually */if (object_class_dynamic_cast(klass, INTERFACE_PCIE_DEVICE) &&!object_class_dynamic_cast(klass, INTERFACE_CONVENTIONAL_PCI_DEVICE)) {pci_dev->cap_present |= QEMU_PCI_CAP_EXPRESS;}if (object_class_dynamic_cast(klass, INTERFACE_CXL_DEVICE)) {pci_dev->cap_present |= QEMU_PCIE_CAP_CXL;}pci_dev = do_pci_register_device(pci_dev,object_get_typename(OBJECT(qdev)),pci_dev->devfn, errp);if (pci_dev == NULL)return;if (pc->realize) {pc->realize(pci_dev, &local_err);if (local_err) {error_propagate(errp, local_err);do_pci_unregister_device(pci_dev);return;}}/** A PCIe Downstream Port that do not have ARI Forwarding enabled must* associate only Device 0 with the device attached to the bus* representing the Link from the Port (PCIe base spec rev 4.0 ver 0.3,* sec 7.3.1).* With ARI, PCI_SLOT() can return non-zero value as the traditional* 5-bit Device Number and 3-bit Function Number fields in its associated* Routing IDs, Requester IDs and Completer IDs are interpreted as a* single 8-bit Function Number. Hence, ignore ARI capable devices.*/if (pci_is_express(pci_dev) &&!pcie_find_capability(pci_dev, PCI_EXT_CAP_ID_ARI) &&pcie_has_upstream_port(pci_dev) &&PCI_SLOT(pci_dev->devfn)) {warn_report("PCI: slot %d is not valid for %s,"" parent device only allows plugging into slot 0.",PCI_SLOT(pci_dev->devfn), pci_dev->name);}if (pci_dev->failover_pair_id) {if (!pci_bus_is_express(pci_get_bus(pci_dev))) {error_setg(errp, "failover primary device must be on ""PCIExpress bus");pci_qdev_unrealize(DEVICE(pci_dev));return;}class_id = pci_get_word(pci_dev->config + PCI_CLASS_DEVICE);if (class_id != PCI_CLASS_NETWORK_ETHERNET) {error_setg(errp, "failover primary device is not an ""Ethernet device");pci_qdev_unrealize(DEVICE(pci_dev));return;}if ((pci_dev->cap_present & QEMU_PCI_CAP_MULTIFUNCTION)|| (PCI_FUNC(pci_dev->devfn) != 0)) {error_setg(errp, "failover: primary device must be in its own ""PCI slot");pci_qdev_unrealize(DEVICE(pci_dev));return;}qdev->allow_unplug_during_migration = true;}/* rom loading */is_default_rom = false;if (pci_dev->romfile == NULL && pc->romfile != NULL) {pci_dev->romfile = g_strdup(pc->romfile);is_default_rom = true;}pci_add_option_rom(pci_dev, is_default_rom, &local_err);if (local_err) {error_propagate(errp, local_err);pci_qdev_unrealize(DEVICE(pci_dev));return;}pci_set_power(pci_dev, true);pci_dev->msi_trigger = pci_msi_trigger;
}

(2)其次,pci_qdev_realize函数调用PCI设备所属的class的realize函数,即pc->realize函数。

代码片段如下:

    if (pc->realize) {pc->realize(pci_dev, &local_err);if (local_err) {error_propagate(errp, local_err);do_pci_unregister_device(pci_dev);return;}}

pc的是PCIDeviceClass结构类型的指针,PCIDeviceClass的定义在include/hw/pci/pci_device.h中,代码如下:

#define TYPE_PCI_DEVICE "pci-device"
typedef struct PCIDeviceClass PCIDeviceClass;
DECLARE_OBJ_CHECKERS(PCIDevice, PCIDeviceClass,PCI_DEVICE, TYPE_PCI_DEVICE)

而struct PCIDeviceClass的定义也在include/hw/pci/pci_device.h中,代码如下:

struct PCIDeviceClass {DeviceClass parent_class;void (*realize)(PCIDevice *dev, Error **errp);PCIUnregisterFunc *exit;PCIConfigReadFunc *config_read;PCIConfigWriteFunc *config_write;uint16_t vendor_id;uint16_t device_id;uint8_t revision;uint16_t class_id;uint16_t subsystem_vendor_id;       /* only for header type = 0 */uint16_t subsystem_id;              /* only for header type = 0 */const char *romfile;                /* rom bar */
};

PCI_DEVICE_GET_CLASS是一个宏定义,但是在QEMU源码中搜素怎么都搜不到。笔者开始以为是在Linux内核源码中的定义,后来在Linux源码中搜索却并未搜到。后来几经周折,终于弄清楚了其具体细节。

要弄清楚PCI_DEVICE_GET_CLASS,得先从DECLARE_OBJ_CHECKERS讲起。DECLARE_OBJ_CHECKERS的相关代码在include/hw/pci/pci_device.h中:

#define TYPE_PCI_DEVICE "pci-device"
typedef struct PCIDeviceClass PCIDeviceClass;
DECLARE_OBJ_CHECKERS(PCIDevice, PCIDeviceClass,PCI_DEVICE, TYPE_PCI_DEVICE)

DECLARE_OBJ_CHECKERS的定义在include/qom/object.h中,代码如下:

/*** DECLARE_OBJ_CHECKERS:* @InstanceType: instance struct name* @ClassType: class struct name* @OBJ_NAME: the object name in uppercase with underscore separators* @TYPENAME: type name** Direct usage of this macro should be avoided, and the complete* OBJECT_DECLARE_TYPE macro is recommended instead.** This macro will provide the three standard type cast functions for a* QOM type.*/
#define DECLARE_OBJ_CHECKERS(InstanceType, ClassType, OBJ_NAME, TYPENAME) \DECLARE_INSTANCE_CHECKER(InstanceType, OBJ_NAME, TYPENAME) \\DECLARE_CLASS_CHECKERS(ClassType, OBJ_NAME, TYPENAME)

代入实际值并展开,得到:

    DECLARE_INSTANCE_CHECKER(PCIDevice, PCI_DEVICE, TYPE_PCI_DEVICE)DECLARE_CLASS_CHECKERS(PCIDeviceClass, PCI_DEVICE, TYPE_PCI_DEVICE)

DECLARE_INSTANCE_CHECKER宏定义也在include/qom/object.h中,代码如下:

/*** DECLARE_INSTANCE_CHECKER:* @InstanceType: instance struct name* @OBJ_NAME: the object name in uppercase with underscore separators* @TYPENAME: type name** Direct usage of this macro should be avoided, and the complete* OBJECT_DECLARE_TYPE macro is recommended instead.** This macro will provide the instance type cast functions for a* QOM type.*/
#define DECLARE_INSTANCE_CHECKER(InstanceType, OBJ_NAME, TYPENAME) \static inline G_GNUC_UNUSED InstanceType * \OBJ_NAME(const void *obj) \{ return OBJECT_CHECK(InstanceType, obj, TYPENAME); }

代入实际值并展开得:

static inline G_GNUC_UNUSED PCIDevice* PCI_DEVICE(const void *obj)
{return OBJECT_CHECK(PCIDevice, obj, TYPE_PCI_DEVICE);
}

DECLARE_CLASS_CHECKERS宏定义也在include/qom/object.h中,代码如下:

/*** DECLARE_CLASS_CHECKERS:* @ClassType: class struct name* @OBJ_NAME: the object name in uppercase with underscore separators* @TYPENAME: type name** Direct usage of this macro should be avoided, and the complete* OBJECT_DECLARE_TYPE macro is recommended instead.** This macro will provide the class type cast functions for a* QOM type.*/
#define DECLARE_CLASS_CHECKERS(ClassType, OBJ_NAME, TYPENAME) \static inline G_GNUC_UNUSED ClassType * \OBJ_NAME##_GET_CLASS(const void *obj) \{ return OBJECT_GET_CLASS(ClassType, obj, TYPENAME); } \\static inline G_GNUC_UNUSED ClassType * \OBJ_NAME##_CLASS(const void *klass) \{ return OBJECT_CLASS_CHECK(ClassType, klass, TYPENAME); }

代入实际值并展开得:

static inline G_GNUC_UNUSED PCIDeviceClass* PCI_DEVICE_GET_CLASS(const void *obj)
{return OBJECT_GET_CLASS(PCIDeviceClass, obj, TYPE_PCI_DEVICE);
}static inline G_GNUC_UNUSED PCIDeviceClass* PCI_DEVICE_CLASS(const void *klass)
{return OBJECT_CLASS_CHECK(PCIDeviceClass, klass, TYPE_PCI_DEVICE);
}

综合两式,最终得到:

static inline G_GNUC_UNUSED PCIDevice* PCI_DEVICE(const void *obj)
{return OBJECT_CHECK(PCIDevice, obj, TYPE_PCI_DEVICE);
}static inline G_GNUC_UNUSED PCIDeviceClass* PCI_DEVICE_GET_CLASS(const void *obj)
{return OBJECT_GET_CLASS(PCIDeviceClass, obj, TYPE_PCI_DEVICE);
}static inline G_GNUC_UNUSED PCIDeviceClass* PCI_DEVICE_CLASS(const void *klass)
{return OBJECT_CLASS_CHECK(PCIDeviceClass, klass, TYPE_PCI_DEVICE);
}

这样,最终就得到了也可以说生成出了PCI_DEVICE_GET_CLASS。

再深入跟紧一点,OBJECT_CHECK、OBJECT_GET_CLASS、OBJECT_CLASS_CHECK三个宏的定义都在include/qom/object.h中,代码如下:

/*** OBJECT_CHECK:* @type: The C type to use for the return value.* @obj: A derivative of @type to cast.* @name: The QOM typename of @type** A type safe version of @object_dynamic_cast_assert.  Typically each class* will define a macro based on this type to perform type safe dynamic_casts to* this object type.** If an invalid object is passed to this function, a run time assert will be* generated.*/
#define OBJECT_CHECK(type, obj, name) \((type *)object_dynamic_cast_assert(OBJECT(obj), (name), \__FILE__, __LINE__, __func__))/*** OBJECT_CLASS_CHECK:* @class_type: The C type to use for the return value.* @class: A derivative class of @class_type to cast.* @name: the QOM typename of @class_type.** A type safe version of @object_class_dynamic_cast_assert.  This macro is* typically wrapped by each type to perform type safe casts of a class to a* specific class type.*/
#define OBJECT_CLASS_CHECK(class_type, class, name) \((class_type *)object_class_dynamic_cast_assert(OBJECT_CLASS(class), (name), \__FILE__, __LINE__, __func__))/*** OBJECT_GET_CLASS:* @class: The C type to use for the return value.* @obj: The object to obtain the class for.* @name: The QOM typename of @obj.** This function will return a specific class for a given object.  Its generally* used by each type to provide a type safe macro to get a specific class type* from an object.*/
#define OBJECT_GET_CLASS(class, obj, name) \OBJECT_CLASS_CHECK(class, object_get_class(OBJECT(obj)), name)

至此,pci_qdev_realize函数所做的第二方面工作即所调用的第2个函数也就是PCI设备所属的class的realize函数,即pc->realize函数就解析完了。

欲知后事如何,且看下回分解。

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

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

相关文章

概率论与数理统计-第7章 假设检验

假设检验的基本概念 二、假设检验的基本思想 假设检验的基本思想实质上是带有某种概率性质的反证法,为了检验一个假设H0,是否正确,首先假定该假设H0正确,然后根据抽取到的样本对假设H0作出接受或拒绝的决策,如果样本观察值导致了…

二叉树的创建与遍历

对于前序遍历,首先访问当前节点,然后递归地遍历左子树和右子树。 这就是为什么前序遍历的代码中,首先是 printf("%d ", root->data);。中序遍历: 对于中序遍历,首先递归地遍历左子树,然后访问…

js 捕获 await 的报错

函数封装 func.js // 捕获 await 的错误 export const to (promise, errorExt, ) > {return promise//成功,则error返回null,result返回data.then((data) > [null, data])//错误则捕获错误内容,然后返回错误信息,result为u…

leetcode-x的平方根

69. x 的平方根 此题使用二分法做 class Solution:def mySqrt(self, x: int) -> int:left 1right xwhile left < right:mid (left right) // 2if x mid ** 2:return midelif x > mid ** 2:left mid 1else:right mid - 1return right

JavaScript 中 0.1 + 0.2 != 0.3:浮点数运算的陷阱

引言 在日常的 JavaScript 开发中&#xff0c;我们经常会面对一些看似简单的数学运算。然而&#xff0c;当我们执行 0.1 0.2 这样的操作时&#xff0c;很多人可能会感到困惑&#xff0c;因为预期的结果应该是 0.3&#xff0c;但实际上却是一个略微不同的值。这个现象是由于 J…

5、C语言:结构

结构 结构的基本知识结构与函数传递结构 结构数组、指向结构的指针自引用结构&#xff08;二叉树&#xff09;表查找类型定义&#xff08;typedef&#xff09;联合位字段 结构也是一种数据类型。类似于int、char、double、float等。 结构是一个或多个变量的集合&#xff0c;这些…

c++学习笔记-STL案例-机房预约系统1-准备工作

前言 准备工作包括&#xff1a;需求分析、项目创建、主菜单实现、退出功能实现 目录 1 机房预约系统需求 1.1 简单介绍 1.2 身份介绍 1.3 机房介绍 1.4 申请介绍 1.5 系统具体要求 1.6 预约系统-主界面思维导图 2 创建项目 2.1 创建项目 2.2 添加文件 ​编辑 3 创建…

页面跳转后,默认选中tree节点并高亮显示

1.场景 操作步骤&#xff1a; 1.点击数据连接数&#xff0c;打开弹窗 2.点击连接状态跳转到数据连接模块 3.默认选中tree的数据源id节点 2.代码 参数解释&#xff1a; 3.实现逻辑 首先将id通过组件传参的方式传过去&#xff0c;数据连接接收后&#xff0c;在tree里设置…

ubuntu 挂载新硬盘

1、检测新硬盘 新增加硬盘&#xff0c;检测硬盘识别情况。 命令检查&#xff1a;sudo fdisk -l 3、格式化磁盘 格式化&#xff1a;sudo mkfs.ext4 /dev/sdb 其中&#xff0c;/dev/sdb是新分区的设备文件名&#xff0c;ext4是要使用的文件系统类型。 4、挂载新分区 sudo mk…

【银行测试】24年金融银行项目,10道高频测试面试题汇总...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 面试题1&#xff…

使用AUTOSAR来开发汽车基础软件的优点

1、高质量。以前我们采用手写代码的方式&#xff0c;是几个工程师在战斗。现在我们采用平台&#xff0c;BSW代码都是供应商提供的&#xff0c;我们相当于后面还有一个团队陪着我们在战斗。 2、低成本。大家都说采用AUTOSAR平台好贵&#xff0c;但是从长远来看是值得的&#xff…

计算机毕业设计----ssm开发的Java快递代拿系统

使用技术 采用 Spring SpringMVC MyBatisPlus&#xff0c;连接池采用 Druid&#xff0c;安全框架使用 Shiro&#xff0c;前端采用 Bootstrap layer 实现。 支付采用支付宝沙箱环境&#xff0c;支付APP下载链接&#xff0c;[点击这里](https://sandbox.alipaydev.com/user/…

SpringBoot中使用SpringEvent业务解耦神器实现监听发布事件同步异步执行任务

场景 SpringBoot中使用单例模式ScheduledExecutorService实现异步多线程任务(若依源码学习)&#xff1a; SpringBoot中使用单例模式ScheduledExecutorService实现异步多线程任务(若依源码学习)-CSDN博客 设计模式-观察者模式在Java中的使用示例-环境监测系统&#xff1a; 设…

GPT实战系列-简单聊聊LangChain

GPT实战系列-简单聊聊LangChain LLM大模型相关文章&#xff1a; GPT实战系列-ChatGLM3本地部署CUDA111080Ti显卡24G实战方案 GPT实战系列-Baichuan2本地化部署实战方案 GPT实战系列-大话LLM大模型训练 GPT实战系列-探究GPT等大模型的文本生成 GPT实战系列-Baichuan2等大模…

SpringBoot中使用LocalDateTime踩坑记录

文章目录 前言一、为什么推荐使用java.time包的LocalDateTime而不是java.util的Date&#xff1f;二、使用LocalDateTime和LocalDate时遇到了哪些坑&#xff1f;2.1 Redis序列化报错2.1.1 问题现象2.1.2 问题分析2.1.3 解决方案 2.2 LocalDateTime和LocalDate类型的属性返回给前…

分享一个实现侧滑菜单的Flutter页面所遇到的问题与解决思路

最近做了一个需要实现侧滑菜单相关的Flutter新页面&#xff0c;页面布局结构稍微比较复杂。因此&#xff0c;做完之后就对研发的过程做出一些整理。 以下主要整理跟侧滑菜单相关的内容。直奔主题&#xff0c;首先&#xff0c;要实现侧滑菜单&#xff0c;有以下几个方案。而本次…

类型特质和静态断言

static_assert( constant-expression, string-literal );static_assert( constant-expression ); // C17 (Visual Studio 2017 and later) constant-expression 可以转换为布尔值的整型常量表达式。 如果计算出的表达式为零 (false)&#xff0c;则显示 string-literal 参数&…

PyTorch|view(),改变张量维度

在构建自己的网络时&#xff0c;了解数据经过每个层后的形状变化是必须的&#xff0c;否则&#xff0c;网络大概率会出现问题。PyToch张量有一个方法&#xff0c;叫做view(),使用这个方法&#xff0c;我们可以很容易的对张量的形状进行改变&#xff0c;从而符合网络的输入要求。…

React 18中新钩子 useDeferredValue 使用

React是一个流行的用于构建用户界面的JavaScript库,它不断发展以为开发人员提供优化性能的工具。 React 18中引入的此类工具之一是useDeferredValue钩子,它旨在通过优先渲染更新来提高应用程序的性能。 useDeferredValue钩子是什么? useDeferredValue钩子是React性能优化工…

SAP PP配置学习(五)

查找 四、 其它 设置 MM 过帐号码范围 定义凭证号码范围 OB52 打开期间 MMPV 开帐 &#xff08;下篇见&#xff09;