qemu作为一个user mode的程序,其入口为main函数,该main函数定义在vl.c文件中. main函数比较长,其中跟KVM初始化相关的主要有两个函数: configure_accelerator()和machine->init(&args). cofigure_accelerator()函数选择运用哪一种虚拟化方案,其应用到的数据结构为accel_list,会调用accel_list[i].init函数。accel_list的初始化如下所示,当使用KVM虚拟化解决方案时,accel_list[i].init对应的函数即为kvm_init.
kvm_init函数定义在kvm-all.c文件中,其主要功能是打开/dev/kvm设备,创建一个虚拟机.
machine->init(&arg)函数主要初始化硬件设备,并且调用qemu_init_vcpu为每一个vcpu创建一个线程,线程执行的函数为qemu_kvm_cpu_thread_fn.
qemu通过调用kvm提供的一系列接口来启动kvm. qemu的入口为vl.c中的main函数, main函数通过调用kvm_init 和 machine->init来初始化kvm. 其中, machine->init会创建vcpu, 用一个线程去模拟vcpu, 该线程执行的函数为qemu_kvm_cpu_thread_fn, 并且该线程最终调用kvm_cpu_exec, 该函数调用kvm_vcpu_ioctl切换到kvm中,下次从kvm中返回时,会接着执行kvm_vcpu_ioctl之后的代码,判断exit_reason,然后进行相应处理.
int kvm_cpu_exec(CPUState *cpu) --> run_ret = kvm_vcpu_ioctl(cpu, KVM_RUN, 0);
当传入参数为KVM_RUN时,会进入到KVM中,会执行__vcpu_run
函数,最终会调用到vcpu_enter_guest
函数, vcpu_enter_guest
函数中调用了kvm_x86_ops->run(vcpu)
, 在intel处理器架构中该函数对应的实现为vmx_vcpu_run
, vmx_vcpu_run
设置好寄存器状态之后调用VM_LAUNCH或者VM_RESUME进入guest vm, 一旦发生vm exit则从此处继续执行下去.
static void vmx_vcpu_run(struct kvm_vcpu *vcpu)
{
/*vmx_vcpu_run设置好寄存器状态之后调用VM_LAUNCH或者VM_RESUME
进入guest vm, 一旦发生vm exit则从此处继续执行下去*/
asm(
/* Enter guest mode */
"jne .Llaunched \n\t"
__ex(ASM_VMX_VMLAUNCH) "\n\t"
"jmp .Lkvm_vmx_return \n\t"
".Llaunched: " __ex(ASM_VMX_VMRESUME) "\n\t"
".Lkvm_vmx_return: "
vmx->launched = 1;
/*当Guest Vm进行IO操作需要访问设备时,
就会触发vm exit 返回到vmx_vcpu_run*/
vmx_com