KVM虚拟机IO处理过程(二) ----QEMU/KVM I/O 处理过程

本文详细介绍了KVM虚拟机的IO处理过程,从guest VM的IO请求到KVM和QEMU的交互。在KVM启动时,QEMU通过调用KVM接口初始化,并创建线程模拟VCPU。当Guest VM进行IO操作时,会触发VM退出,进入kvm_vcpu_ioctl处理,依据exit_reason调用不同函数。如果使用raw磁盘格式,IO在QEMU中的处理涉及特定函数栈。完成IO操作后,通过kvm_vcpu_ioctl重新进入KVM执行。
摘要由CSDN通过智能技术生成

    接着KVM虚拟机IO处理过程中Guest Vm IO处理过程(http://blog.csdn.net/dashulu/article/details/16820281),本篇文章主要描述IO从guest vm跳转到kvm和qemu后的处理过程.

    首先回顾一下kvm的启动过程(http://blog.csdn.net/dashulu/article/details/17074675).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)
{
    struct kvm_run *run = cpu->kvm_run;
    int ret, run_ret;

    DPRINTF("kvm_cpu_exec()\n");

    if (kvm_arch_process_async_events(cpu)) {
        cpu->exit_request = 0;
        return EXCP_HLT;
    }

    do {
        if (cpu->kvm_vcpu_dirty) {
            kvm_arch_put_registers(cpu, KVM_PUT_RUNTIME_STATE);
            cpu->kvm_vcpu_dirty = false;
        }

        kvm_arch_pre_run(cpu, run);
        if (cpu->exit_request) {
            DPRINTF("interrupt exit requested\n");
            /*
             * KVM requires us to reenter the kernel after IO exits to complete
             * instruction emulation. This self-signal will ensure that we
             * leave ASAP again.
             */
            qemu_cpu_kick_self();
        }
        qemu_mutex_unlock_iothread();

        run_ret = kvm_vcpu_ioctl(cpu, KVM_RUN, 0);

        qemu_mutex_lock_iothread();
        kvm_arch_post_run(cpu, run);

        if (run_ret < 0) {
            if (run_ret == -EINTR || run_ret == -EAGAIN) {
                DPRINTF("io window exit\n");
                ret = EXCP_INTERRUPT;
                break;
            }
            fprintf(stderr, "error: kvm run failed %s\n",
                    strerror(-run_ret));
            abort();
        }

        trace_kvm_run_exit(cpu->cpu_index, run->exit_reason);
        switch (run->exit_reason) {
        case KVM_EXIT_IO:
            DPRINTF("handle_io\n");
            kvm_handle_io(run->io.port,
                          (uint8_t *)run + run->io.data_offset,
                          run->io.direction,
                          run->io.size,
                          run->io.count);
            ret = 0;
            break;
        case KVM_EXIT_MMIO:
            DPRINTF("handle_mmio\n");
            cpu_physical_memory_rw(run->mmio.phys_addr,
                                   run->mmio.data,
                                   run->mmio.len,
                                   run->mmio.is_write);
            ret = 0;
            break;
        case KVM_EXIT_IRQ_WINDOW_OPEN:
            DPRINTF("irq_window_open\n");
            ret = EXCP_INTERRUPT;
            break;
        case KVM_EXIT_SHUTDOWN:
            DPRINTF("shutdown\n");
            qemu_system_reset_request();
            ret = EXCP_INTERRUPT;
            break;
        case KVM_EXIT_UNKNOWN:
            fprintf(stderr, "KVM: unknown exit, hardware reason %" PRIx64 "\n",
                    (uint64_t)run->hw.hardware_exit_reason);
            ret = -1;
            break;
        case KVM_EXIT_INTERNAL_ERROR:
            ret = kvm_handle_internal_error(cpu, run);
            break;
        default:
            DPRINTF("kvm_arch_handle_exit\n");
            ret = kvm_arch_handle_exit(cpu, run);
            break;
        }
    } while (ret =&#
libvirt是一种用于管理虚拟化平台的开源工具,可以通过它来管理KVM虚拟机。libvirt提供了一组命令行工具和API来管理虚拟机,可以使用它来抓取KVM虚拟机的监控指标数据。 以下是使用libvirt抓取KVM虚拟机监控指标数据的步骤: 1. 安装libvirt工具及相关依赖 ```shell sudo apt-get install libvirt-bin libvirt-dev qemu-kvm ``` 2. 打开libvirt的API libvirt提供了多种语言的API,如C、Python、Java等,这里以Python为例,打开libvirt的API: ```python import libvirt conn = libvirt.open('qemu:///system') ``` 3. 获取虚拟机对象 使用`conn.listDomainsID()`函数获取所有虚拟机的ID列表,然后使用`conn.lookupByID(domain_id)`函数获取指定虚拟机的对象。 ```python domain_id_list = conn.listDomainsID() if len(domain_id_list) == 0: print('No active domains') else: for domain_id in domain_id_list: domain = conn.lookupByID(domain_id) ``` 4. 获取监控指标数据 可以使用`domain.info()`函数获取虚拟机的基本信息,如CPU使用率、内存使用情况等,也可以使用`domain.blockStats()`和`domain.interfaceStats()`函数获取虚拟机的磁盘和网络IO数据。 ```python # 获取虚拟机CPU使用率和内存使用情况 cpu_stats = domain.info()[4] mem_stats = domain.memoryStats() # 获取虚拟机磁盘IO数据 disk_stats = {} for disk in domain.listDisks(): block_info = domain.blockStats(disk) disk_stats[disk] = {'read_bytes': block_info[1], 'write_bytes': block_info[3]} # 获取虚拟机网络IO数据 net_stats = {} for interface in domain.interfaceAddresses(libvirt.VIR_DOMAIN_INTERFACE_ADDRESSES_SRC_AGENT, 0): stats = domain.interfaceStats(interface) net_stats[interface] = {'rx_bytes': stats[0], 'tx_bytes': stats[4]} ``` 5. 关闭libvirt连接 ```python conn.close() ``` 以上就是使用libvirt抓取KVM虚拟机监控指标数据的步骤,可以根据自己的需求来调整获取的监控指标数据。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值