【QEMU系统分析之启动篇(二十一)】

系列文章目录

第二十一章 QEMU系统仿真的加速器上电后设置分析


文章目录

  • 系列文章目录
    • 第二十一章 QEMU系统仿真的加速器上电后设置分析
  • 前言
  • 一、QEMU是什么?
  • 二、QEMU系统仿真的启动分析
    • 1.系统仿真的初始化代码
    • 2.主循环
    • 3. qemu_default_main()
    • 4. qemu_main_loop()
      • main_loop_should_exit()
        • qemu_debug_requested()
        • vm_stop()
          • qemu_in_vcpu_thread()
          • qemu_system_vmstop_request_prepare()
          • qemu_system_vmstop_request()
          • cpu_stop_current()
          • qemu_suspend_requested()
        • qemu_system_suspend()
        • qemu_shutdown_requested()
        • qemu_kill_report()
        • qemu_system_shutdown(request)
        • qemu_reset_requested()
        • pause_all_vcpus()
        • qemu_system_reset(request)
        • resume_all_vcpus()
        • runstate_check()
        • runstate_set()
        • qemu_wakeup_requested()
        • notifier_list_notify()
        • qapi_event_send_wakeup()
        • qemu_powerdown_requested()
        • qemu_system_powerdown()
        • qemu_vmstop_requested()
      • main_loop_wait()
        • qemu_soonest_timeout()
        • timerlistgroup_deadline_ns()
        • os_host_main_loop_wait()
        • icount_enabled()
        • icount_start_warp_timer()
        • qemu_clock_run_all_timers()
  • 总结


前言

本文以 QEMU 8.2.2 为例,分析其作为系统仿真工具的启动过程,并为读者展示各种 QEMU 系统仿真的启动配置实例。
本文读者需要具备一定的 QEMU 系统仿真使用经验,并对 C 语言编程有一定了解。


一、QEMU是什么?

QEMU 是一个通用且开源的机器模拟器和虚拟机。
其官方主页是:https://www.qemu.org/


二、QEMU系统仿真的启动分析

1.系统仿真的初始化代码

QEMU 作为系统仿真工具,其入口代码在 system/main.c 文件中,初始化函数 qemu_init() 的实现在 system/vl.c 文件中,在完成 QEMU 虚拟机初始设置后,接下来将进入主循环,本篇文章将完成以下代码部分的分析。

2.主循环

这部分代码在 system/vl.c 文件中,实现如下:

int (*qemu_main)(void) = qemu_default_main;int main(int argc, char **argv)
{qemu_init(argc, argv);return qemu_main();
}

3. qemu_default_main()

函数 qemu_default_main() 在 /system/vl.c 文件中,定义如下:

int qemu_default_main(void)
{int status;status = qemu_main_loop();qemu_cleanup(status);return status;
}

4. qemu_main_loop()

函数 qemu_main_loop() 在 /system/runstate.c 文件中,定义如下:

int qemu_main_loop(void)
{int status = EXIT_SUCCESS;while (!main_loop_should_exit(&status)) {main_loop_wait(false);}return status;
}

main_loop_should_exit()

函数 main_loop_should_exit() 在 /system/runstate.c 文件中,定义如下:

static bool main_loop_should_exit(int *status)
{RunState r;ShutdownCause request;if (qemu_debug_requested()) {vm_stop(RUN_STATE_DEBUG);}if (qemu_suspend_requested()) {qemu_system_suspend();}request = qemu_shutdown_requested();if (request) {qemu_kill_report();qemu_system_shutdown(request);if (shutdown_action == SHUTDOWN_ACTION_PAUSE) {vm_stop(RUN_STATE_SHUTDOWN);} else {if (shutdown_exit_code != EXIT_SUCCESS) {*status = shutdown_exit_code;} else if (request == SHUTDOWN_CAUSE_GUEST_PANIC &&panic_action == PANIC_ACTION_EXIT_FAILURE) {*status = EXIT_FAILURE;}return true;}}request = qemu_reset_requested();if (request) {pause_all_vcpus();qemu_system_reset(request);resume_all_vcpus();/** runstate can change in pause_all_vcpus()* as iothread mutex is unlocked*/if (!runstate_check(RUN_STATE_RUNNING) &&!runstate_check(RUN_STATE_INMIGRATE) &&!runstate_check(RUN_STATE_FINISH_MIGRATE)) {runstate_set(RUN_STATE_PRELAUNCH);}}if (qemu_wakeup_requested()) {pause_all_vcpus();qemu_system_wakeup();notifier_list_notify(&wakeup_notifiers, &wakeup_reason);wakeup_reason = QEMU_WAKEUP_REASON_NONE;resume_all_vcpus();qapi_event_send_wakeup();}if (qemu_powerdown_requested()) {qemu_system_powerdown();}if (qemu_vmstop_requested(&r)) {vm_stop(r);}return false;
}
qemu_debug_requested()

函数 qemu_debug_requested() 在 /system/runstate.c 文件中,定义如下:

static int qemu_debug_requested(void)
{int r = debug_requested;debug_requested = 0;return r;
}
vm_stop()

函数 vm_stop() 在 /system/cpus.c 文件中,定义如下:

int vm_stop(RunState state)
{if (qemu_in_vcpu_thread()) {qemu_system_vmstop_request_prepare();qemu_system_vmstop_request(state);/** FIXME: should not return to device code in case* vm_stop() has been requested.*/cpu_stop_current();return 0;}return do_vm_stop(state, true);
}
qemu_in_vcpu_thread()

函数 qemu_in_vcpu_thread() 在 /system/cpus.c 文件中,定义如下:

bool qemu_in_vcpu_thread(void)
{return current_cpu && qemu_cpu_is_self(current_cpu);
}
qemu_system_vmstop_request_prepare()

函数 qemu_system_vmstop_request_prepare() 在 /system/runstate.c 文件中,定义如下:

void qemu_system_vmstop_request_prepare(void)
{qemu_mutex_lock(&vmstop_lock);
}
qemu_system_vmstop_request()

函数 qemu_system_vmstop_request() 在 /system/runstate.c 文件中,定义如下:

void qemu_system_vmstop_request(RunState state)
{vmstop_requested = state;qemu_mutex_unlock(&vmstop_lock);qemu_notify_event();
}
cpu_stop_current()

函数 cpu_stop_current() 在 /system/cpus.c 文件中,定义如下:

void cpu_stop_current(void)
{if (current_cpu) {current_cpu->stop = true;cpu_exit(current_cpu);}
}

函数 cpu_exit() 在 /hw/core/cpu-common.c 文件中,定义如下:

void cpu_exit(CPUState *cpu)
{qatomic_set(&cpu->exit_request, 1);/* Ensure cpu_exec will see the exit request after TCG has exited.  */smp_wmb();qatomic_set(&cpu->neg.icount_decr.u16.high, -1);
}
qemu_suspend_requested()
qemu_system_suspend()
qemu_shutdown_requested()
qemu_kill_report()
qemu_system_shutdown(request)
qemu_reset_requested()
pause_all_vcpus()
qemu_system_reset(request)
resume_all_vcpus()
runstate_check()
runstate_set()
qemu_wakeup_requested()
notifier_list_notify()
qapi_event_send_wakeup()
qemu_powerdown_requested()
qemu_system_powerdown()
qemu_vmstop_requested()

main_loop_wait()

函数 qemu_default_main() 在 /util/main-loop.c 文件中,定义如下:

void main_loop_wait(int nonblocking)
{MainLoopPoll mlpoll = {.state = MAIN_LOOP_POLL_FILL,.timeout = UINT32_MAX,.pollfds = gpollfds,};int ret;int64_t timeout_ns;if (nonblocking) {mlpoll.timeout = 0;}/* poll any events */g_array_set_size(gpollfds, 0); /* reset for new iteration *//* XXX: separate device handlers from system ones */notifier_list_notify(&main_loop_poll_notifiers, &mlpoll);if (mlpoll.timeout == UINT32_MAX) {timeout_ns = -1;} else {timeout_ns = (uint64_t)mlpoll.timeout * (int64_t)(SCALE_MS);}timeout_ns = qemu_soonest_timeout(timeout_ns,timerlistgroup_deadline_ns(&main_loop_tlg));ret = os_host_main_loop_wait(timeout_ns);mlpoll.state = ret < 0 ? MAIN_LOOP_POLL_ERR : MAIN_LOOP_POLL_OK;notifier_list_notify(&main_loop_poll_notifiers, &mlpoll);if (icount_enabled()) {/** CPU thread can infinitely wait for event after* missing the warp*/icount_start_warp_timer();}qemu_clock_run_all_timers();
}
qemu_soonest_timeout()
timerlistgroup_deadline_ns()
os_host_main_loop_wait()
icount_enabled()
icount_start_warp_timer()
qemu_clock_run_all_timers()

总结

以上分析了 QEMU 系统仿真在启动过程中,QEMU系统仿真通知字符设备后端驱动开始工作的代码部分。

在完成此动作后,QEMU 程序回到主循环,并开始等待轮询的事件处理。

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

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

相关文章

BPE、Wordpiece、Unigram、SpanBERT等Tokenizer细节总结

BPE(Byte Pair Encoding) GPT-2和Roberta用的是这种&#xff0c;不会产生[UNK]这个unknown字符 这部分部分摘录自https://martinlwx.github.io/zh-cn/the-bpe-tokenizer/ 看以下code例子就足够理解了&#xff0c;核心是维护self.merges&#xff08;维护一个pair->str的字…

[蓝桥杯2024]-Reverse:rc4解析(对称密码rc4)

无壳 查看ida 这里应该运行就可以得flag&#xff0c;但是这个程序不能直接点击运行 按照伪代码写exp 完整exp&#xff1a; keylist(gamelab) content[0xB6,0x42,0xB7,0xFC,0xF0,0xA2,0x5E,0xA9,0x3D,0x29,0x36,0x1F,0x54,0x29,0x72,0xA8, 0x63,0x32,0xF2,0x44,0x8B,0x85,0x…

如何在 Visual Studio 中通过 NuGet 添加包

在安装之前要先确定Nuget的包源是否有问题。 Visual Studio中怎样更改Nuget程序包源-CSDN博客 1.图形界面安装 打开您的项目&#xff0c;并在解决方案资源管理器中选择您的项目。单击“项目”菜单&#xff0c;然后选择“管理 NuGet 程序包”选项。在“NuGet 包管理器”窗口中…

详解如何品味品深茶的精髓

在众多的茶品牌中&#xff0c;品深茶以其独特的韵味和深厚的文化底蕴&#xff0c;赢得了众多茶友的喜爱。今天&#xff0c;让我们一同探寻品深茶的精髓&#xff0c;品味其独特的魅力。 品深茶&#xff0c;源自中国传统茶文化的精髓&#xff0c;承载着世代茶人的智慧与匠心。这…

理解JavaScript对象[Bom]:属性、方法

对象是什么 在JavaScript中&#xff0c;对象&#xff08;Object&#xff09;是一种复合的数据类型&#xff0c;它允许你将数据和功能&#xff08;方法&#xff09;组合在一起。 创建对象 var person { firstName: John, lastName: Doe, age: 30, greet: function() …

03-MVC执行流程-参数解析与Model

重要组件 准备Model&#xff0c;Controller Configuration public class WebConfig {ControllerAdvicestatic class MyControllerAdvice {ModelAttribute("b")public String bar() {return "bar";}}Controllerstatic class Controller1 {ResponseStatus(H…

windows环境下安装Apache

首先apache官网下载地址&#xff1a;http://www.apachelounge.com/download/按照自己的电脑操作系统来安装 这里我安装的是win64 主版本是2.4的apache。 然后解压压缩包到一个全英文的路径下&#xff01;&#xff01;&#xff01;一定一定不要有中文 中文符号也不要有&#xff…

ansible-copy用法

目录 概述实践不带目录拷贝带目录拷贝 概述 ansible copy 常用用法举例 不带目录拷贝&#xff0c;拷贝的地址要写全 带目录拷贝&#xff0c;拷贝路径不要写在 dest 路径中 实践 不带目录拷贝 # with_fileglob 是 Ansible 中的一个循环关键字&#xff0c;用于处理文件通配符匹…

【Vue3+Tres 三维开发】02-Debug

预览 介绍 Debug 这里主要是讲在三维中的调试,同以前threejs中使用的lil-gui类似,TRESJS也提供了一套可视化参数调试的插件。使用方式和之前的组件相似。 使用 通过导入useTweakPane 即可 import { useTweakPane, OrbitControls } from "@tresjs/cientos"const {…

数字文旅重塑旅游发展新格局:以数字化转型为突破口,提升旅游服务的智能化水平,为游客带来全新的旅游体验

随着信息技术的迅猛发展&#xff0c;数字化已成为推动各行各业创新发展的重要力量。在旅游业领域&#xff0c;数字文旅的兴起正以其强大的驱动力&#xff0c;重塑旅游发展的新格局。数字文旅以数字化转型为突破口&#xff0c;通过提升旅游服务的智能化水平&#xff0c;为游客带…

HarmonyOS Next从入门到精通实战精品课

第一阶段&#xff1a;HarmonyOS Next星河版从入门到精通该阶段由HarmonyOS Next星河版本出发&#xff0c;介绍HarmonyOS Next版本应用开发基础概念&#xff0c;辅助学员快速上手新版本开发范式&#xff0c;共计42课时 第一天鸿蒙NEXT Mac版、Windows版【编辑器】和【模拟器】&a…

java 一个方法里面的 所有方法并行执行 用多线程

java 一个方法里面的 所有方法并行执行 用线程 package com.example.springbootycw.schedule;import java.util.Arrays; import java.util.HashSet; import java.util.List; import java.util.Set; import java.util.concurrent.ExecutorService; import java.util.concurrent…

android 内部保存数据

在Android中&#xff0c;内部序列化对象通常指的是将对象的状态转换为字节流&#xff0c;以便可以将对象保存到文件、通过网络发送&#xff0c;或者在应用的不同部分之间传递。在Java中&#xff0c;序列化是通过实现java.io.Serializable接口来完成的。Android继承了Java的序列…

BootStrap详解

Bootstrap简介 什么是BootStrap&#xff1f; BootStrap来自Twitter&#xff0c;是目前最受欢迎的响应式前端框Bootstrap是基于HTML、CSS、JavaScript的&#xff0c;它简洁灵活&#xff0c;使得Web开发更加快捷 为什么使用Bootstrap&#xff1f; 移动设备优先&#xff1a;自…

Kafka 3.x.x 入门到精通(07)——Java应用场景——SpringBoot集成

Kafka 3.x.x 入门到精通&#xff08;07&#xff09;——Java应用场景——SpringBoot集成 4. Java应用场景——SpringBoot集成4.1 创建SpringBoot项目4.1.1 创建SpringBoot项目4.1.2 修改pom.xml文件4.1.3 在resources中增加application.yml文件 4.2 编写功能代码4.2.1 创建配置…

如何维护 Oracle B*tree 索引。 多列index是合并一起指向rowid的

尽管这是一份较旧的文档&#xff0c;但以下信息仍与更高版本相关。 Although this is an older document, the information below is still relevant to later versions. 范围 它旨在帮助试图了解如何维护 Oracle B*tree 索引的用户。 详 Oracle 版本 8 提供了五种索引方案…

机器人-轨迹规划

旋转矩阵 旋转矩阵--R--一个3*3的矩阵&#xff0c;其每列的值时B坐标系在A坐标系上的投影值。 代表B坐标系相对于A坐标系的姿态。 旋转矩阵的转置矩阵 其实A相对于B的旋转矩阵就相当于把B的列放到行上就行。 视频 &#xff08;将矩阵的行列互换得到的新矩阵称为转置矩阵。&…

MySQL的SQL文件转换为适用于SQLite的SQL文件

最近了解到一个较小众的数据库SQLite&#xff0c;打算拿来玩一玩&#xff0c;但手上目前只有MySQL的一些库表文件无法直接导入使用&#xff0c;所以出一期记录&#xff0c;手动进行二者转换 分析SQL语句 首先要明确&#xff0c;MySQL和SQLite两者格式差别不止限于字段类型、特…

C++二维数组arr[3][4]与arr(3, vector<int>(4))的差异

int arr[3][4] 和 vector<vector<int>> arr(3, vector<int>(4)) 都是用于存储二维数组的数据结构&#xff0c;但它们之间有以下几个差异&#xff1a; 1. 内存管理&#xff1a;int arr[3][4] 是一个静态数组&#xff0c;它在编译时就分配了连续的内存空间&am…

flask+uwsgi+nginx+cerbot配置

配置步骤 安装flask和uwsgi pip install Flask uwsgi 创建一个简单的flask应用&#xff08;app.py&#xff09;或者是自己的flask项目 from flask import Flask app Flask(__name__)app.route(/) def hello_world():return Hello, World! 配置uwsgi&#xff0c;这里我给出…