qemu源码解析【05】qemu启动初始化流程

目录

  • qemu源码解析【05】qemu启动初始化流程
    • [1. qemu入口函数](#1. qemu入口函数)
    • [2. qemu_init()函数内容](#2. qemu_init()函数内容)
      • [2.1 构建参数链表](#2.1 构建参数链表)
      • [2.2 一些系统相关的交互](#2.2 一些系统相关的交互)
      • [2.3 解析启动的命令行参数](#2.3 解析启动的命令行参数)

qemu源码解析【05】qemu启动初始化流程

1. qemu入口函数

  • qemu入口main函数位于system/main.c文件,main函数内容很简单,完成启动前的初始化,进入主循环

    int qemu_default_main(void)
    {
    int status;

    复制代码
      status = qemu_main_loop();
      qemu_cleanup(status);
    
      return status;

    }

    int (*qemu_main)(void) = qemu_default_main;

    int main(int argc, char **argv)
    {
    // 初始化流程
    qemu_init(argc, argv);
    // 主循环函数
    return qemu_main();
    }

  • 这篇文章浅浅研究一下qemu_init(),也就是初始化函数里面的代码

2. qemu_init()函数内容

  • 下面开始逐段分析qemu_init()函数内容

2.1 构建参数链表

复制代码
    qemu_add_opts(&qemu_drive_opts);
    qemu_add_drive_opts(&qemu_legacy_drive_opts);
    qemu_add_drive_opts(&qemu_common_drive_opts);
    qemu_add_drive_opts(&qemu_drive_opts);
    qemu_add_drive_opts(&bdrv_runtime_opts);
    ......
    qemu_add_opts(&qemu_fw_cfg_opts);
    qemu_add_opts(&qemu_action_opts);
    qemu_add_run_with_opts();
    module_call_init(MODULE_INIT_OPTS);
  • 在util/qemu-config.c中有2个全局的QemuOptsList*数组,vm_config_groups[48]和drive_config_groups[5]
  • 以上代码就是在初始化这2个数组,将qemu_drive_opts,qemu_device_opts这些结构体填到这2个数组中,这里面存的其实就是qemu启动时我们传进去的那些参数,或者我们没有指定这些一些参数,系统帮我们给定了一个默认值
  • 然后还有个module_call_init()函数,作用跟上一章的qom链表初始化过程差不多,就是根据init_type_list[MODULE_INIT_OPTS]链表上挂的所有成员,挨个初始化

2.2 一些系统相关的交互

复制代码
    // 初始化glib的日志库
    error_init(argv[0]);

    // 找到当前执行的文件路径,存在静态变量exec_dir中
    qemu_init_exec_dir(argv[0]);

    // 保存当前机器的一些架构信息
    os_setup_limits();
    qemu_init_arch_modules();

    // 初始化子系统,例如上文说到的那个QOM链表
    qemu_init_subsystems();

2.3 解析启动的命令行参数

  • qemu自带了一个字典库,位于qobject/qdict.c文件中

  • 解析命令行参数之前,先创建了一个字典库,用来保存所有解析结果

  • 这段解析参数很多,但逻辑很简单,就是机械地解析所有参数,然后保存到字典中,以下做了删除

  • 可以看到,我们传入的cpu,hda,smp等所有的参数都在此处完成和解析和保存

    复制代码
      machine_opts_dict = qdict_new();
      if (userconfig) {
          qemu_read_default_config_file(&error_fatal);
      }
    
      /* second pass of option parsing */
      optind = 1;
      for(;;) {
          if (optind >= argc)
              break;
          if (argv[optind][0] != '-') {
              loc_set_cmdline(argv, optind, 1);
              drive_add(IF_DEFAULT, 0, argv[optind++], HD_OPTS);
          } else {
              const QEMUOption *popt;
    
              popt = lookup_opt(argc, argv, &optarg, &optind);
              if (!(popt->arch_mask & arch_type)) {
                  error_report("Option not supported for this target");
                  exit(1);
              }
              switch(popt->index) {
              case QEMU_OPTION_cpu:
              
                  /* hw initialization will check this */
                  cpu_option = optarg;
                  break;
              case QEMU_OPTION_hda:
              case QEMU_OPTION_hdb:
              case QEMU_OPTION_hdc:
              case QEMU_OPTION_hdd:
                  drive_add(IF_DEFAULT, popt->index - QEMU_OPTION_hda, optarg,
                            HD_OPTS);
                  break;
             // 此处省略
             // .......
    
              default:
                  error_report("Option not supported in this build");
                  exit(1);
              }
          }
      }
相关推荐
Thera7775 小时前
状态机(State Machine)详解:原理、优缺点与 C++ 实战示例
开发语言·c++
linux开发之路5 小时前
C++高性能日志库开发实践
c++·c++项目·后端开发·c++新特性·c++校招
刻BITTER6 小时前
在TRAE 上安装PlatformIO
c++·单片机·嵌入式硬件·arduino
永远都不秃头的程序员(互关)6 小时前
C++动态数组实战:从手写到vector优化
c++·算法
水力魔方7 小时前
武理排水管网模拟分析系统应用专题5:模型克隆与并行计算
数据库·c++·算法·swmm
OliverH-yishuihan7 小时前
在win10上借助WSL用VS2019开发跨平台项目实例
linux·c++·windows
汉克老师8 小时前
GESP2025年12月认证C++二级真题与解析(编程题1 (环保能量球))
c++·gesp二级·gesp2级
郝学胜-神的一滴8 小时前
Linux进程与线程控制原语对比:双刃出鞘,各显锋芒
linux·服务器·开发语言·数据结构·c++·程序人生
青岛少儿编程-王老师9 小时前
CCF编程能力等级认证GESP—C++2级—20251227
java·开发语言·c++
javachen__9 小时前
341-十道经典程序设计题目
数据结构·c++·算法