位置: 编程技术 - 正文

ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机(ubuntu 12.04安装)

编辑:rootadmin

推荐整理分享ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机(ubuntu 12.04安装),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:ubuntu 127.0.0.53,ubuntu 125%,ubuntu 12.04,ubuntu 20.04,ubuntu 12.04,ubuntu22.04,ubuntu12.04lts,ubuntu 12.04,内容如对您有帮助,希望把文章链接给更多的朋友!

qemu、virtual box、vmware、xen都是虚拟机,一般用户接触到的virtual box和vmware比较多,都是用来ubuntu中跑windows,或者windows中跑ubuntu的。

qemu其实是鼎鼎大名的最基础的开源模拟器,可以纯软件模拟x、arm、mips,这一点完虐其它模拟器;也可以使用硬件加速,比如linux下kvm和windows以及mac下的haxm。这些硬件加速又是基于initel VT-x, intel VT-d,以及amd对应的技术,这些技术提供了vCPU,以及硬件的影子页表(intel EPT),大大减轻了qemu软件模拟的工作量。

virtual box,qemu-kvm都使用到了qemu,但是仅仅用到了它的设备模拟功能。qemu对于gpu的模拟比较渣,所以基于qemu的Android emulator自己实现了opengles 的qemu pipe,使用host电脑上的opengl进行绘图。xen在云计算中用的比较多,在这里不做详细介绍。其它模拟器基本都是运行在普通操作系统之上的一个进程,每一个核是其中的一个线程。

本文介绍kvm的使用,在intel平台下ubuntu.中实现一个最简单的模拟器,计算2+2的结果并通过io端口输出。

内核中kvm api的介绍可以看:Documentation/virtual/kvm/api.txt,其它的一些文档:Documentation/virtual/kvm/。完整的源码: Code复制内容到剪贴板 mov $0x3f8, %dx add %bl, %al add $'0', %al out %al, (%dx) mov $'n', %al out %al, (%dx) hlt

这段代码充当了guest os,基本上算是一个裸奔的系统了。它实现了2+2,然后再加上'0',把4转为ascii的'4',并通过端口0x3f8输出。然后再输出了'n',就关机了。

我们把这段代码对应的二进制存到数组里面:

Ruby Code复制内容到剪贴 const uint8_t code[] = { 0xba, 0xf8, 0x, /* mov $0x3f8, %dx */ 0x, 0xd8, /* add %bl, %al */ 0x, '0', /* add $'0', %al */ 0xee, /* out %al, (%dx) */ 0xb0, 'n', /* mov $'n', %al */ 0xee, /* out %al, (%dx) */ 0xf4, /* hlt */ };

怎么得到这些机器码呢?

Ruby Code复制内容到剪贴板 shuyin.wsy@---:~$ cat simple_os.asm mov $0x3f8, %dx add %bl, %al add $'0', %al out %al, (%dx) mov $'n', %al out %al, (%dx) hlt shuyin.wsy@---:~$ as -o simple_os.o simple_os.asm shuyin.wsy@---:~$ objdump -d simple_os.o simple_os.o: file format elf-x- Disassembly of section .text: <.text>: 0: ba f8 mov $0x3f8,%dx 4: d8 add %bl,%al 6: add $0x,%al 8: ee out %al,(%dx) 9: b0 0a mov $0xa,%al b: ee out %al,(%dx) c: f4 hlt

可以在这个网页上查看汇编指令,以及对应的机器码: group 3

所以我们需要在simple_os.asm文件的开头添加.code,这样的话就对了,但是objdump显示的又不对了,需要这样使用才行:

Ruby Code复制内容到剪贴板 shuyin.wsy@---:~$ objdump -d -Mintel,i simple_os.o simple_os.o: file format elf-x- Disassembly of section .text: <.text>: 0: ba f8 mov dx,0x3f8 3: d8 add al,bl 5: add al,0x 7: ee out dx,al 8: b0 0a mov al,0xa a: ee out dx,al b: f4 hlt

我们会把这段代码,放到虚拟物理内存,也就是GPA(guest physical address)的第二个页面中(to avoid conflicting with a non-existent real-mode interrupt descriptor table at address 0),防止和实模式的中断向量表冲突。al和bl初始化为2,cs初始化为0,ip指向第二个页面的起始位置0x。除此之外,我们还有一个虚拟的串口设备,端口是0x3f8,8bit,用于输出字符。

为了实现一个虚拟机,我们首先需要打开/dev/kvm:

Ruby Code复制内容到剪贴板 kvm = open("/dev/kvm", O_RDWR | O_CLOEXEC);

在使用kvm之前,需要使用KVM_GET_API_VERSION ioctl()去检查下kvm的版本是否正确,看看是否为api,是才可以继续运行

Ruby Code复制内容到剪贴板 ret = ioctl(kvm, KVM_GET_API_VERSION, NULL); if (ret == -1) err(1, "KVM_GET_API_VERSION"); if (ret != ) errx(1, "KVM_GET_API_VERSION %d, expected ", ret);

检查完api版本后,可以使用KVM_CHECK_EXTENSION ioctl()去检查其它extensions是否可用,比如KVM_SET_USER_MEMORY_REGION,用来检查kvm是否支持硬件影子页表( Code复制内容到剪贴板 ret = ioctl(kvm, KVM_CHECK_EXTENSION, KVM_CAP_USER_MEMORY); if (ret == -1) err(1, "KVM_CHECK_EXTENSION"); if (!ret) errx(1, "Required extension KVM_CAP_USER_MEM not available"); ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机(ubuntu 12.04安装)

然后再创建一个虚拟机vm,这个vm和内存,设备,所有的vCPU相关,在host系统中对应一个进程:

Ruby Code复制内容到剪贴板 vmfd = ioctl(kvm, KVM_CREATE_VM, (unsigned long)0);

虚拟机需要一些虚拟物理内存,用来存放guest os。当guest os进行内存访问时,如果缺页,kvm会根据KVM_SET_USER_MEMORY_REGION的设置,去尝试解决缺页的问题,如果kvm无法解决,就会退出,退出原因是KVM_EXIT_MMIO,然后由qemu或者其它东西去进行设备的模拟(《android qemu-kvm内存管理和IO映射》)。

我们先在host中申请一页内存,然后把guest os裸奔的代码拷贝过去:

Ruby Code复制内容到剪贴板 mem = mmap(NULL, 0x, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANONYMOUS, -1, 0); memcpy(mem, code, sizeof(code));

然后我们需要把host 虚拟空间的内存和guest os虚拟物理内存的映射关系使用KVM_SET_USER_MEMORY_REGION ioctl()告知kvm:

Ruby Code复制内容到剪贴板 struct kvm_userspace_memory_region region = { .slot = 0, .guest_phys_addr = 0x, .memory_size = 0x, .userspace_addr = (uint_t)mem, }; ioctl(vmfd, KVM_SET_USER_MEMORY_REGION, ®ion);

这样,当guest os访问到虚拟物理内存的0x~0x之间的话,kvm会直接访问到mem所对应的真实的物理内存。

现在,我们有了一个虚拟机vm,有了一些虚拟物理内存,内存里面有guest os的代码,那么我们需要给虚拟机添加一个核(vCPU),对应一个线程。当然也可以多核(vCPUs,调用多次KVM_CREATE_VCPU):

Ruby Code复制内容到剪贴板 vcpufd = ioctl(vmfd, KVM_CREATE_VCPU, (unsigned long)0);

每一个vCPU都和一个kvm_run结构体相关,kvm_run用于内核态和用户态信息的同步,比如从用户态的虚拟机中获得内核态的kvm退出的原因,KVM_EXIT_MMIO, KVM_EXIT_IO之类的。先获得kvm_run结构体的大小,然后分配内存并和vCPU进行绑定:

Ruby Code复制内容到剪贴板 mmap_size = ioctl(kvm, KVM_GET_VCPU_MMAP_SIZE, NULL); run = mmap(NULL, mmap_size, PROT_READ | PROT_WRITE, MAP_SHARED, vcpufd, 0);

vCPU中还有处理器寄存器的状态,分为两组,struct kvm_regs和struct kvm_sregs,我们需要设置其中的cs,al,bl,ip等寄存器:

Ruby Code复制内容到剪贴板 ioctl(vcpufd, KVM_GET_SREGS, &sregs); sregs.cs.base = 0; sregs.cs.selector = 0; ioctl(vcpufd, KVM_SET_SREGS, &sregs); struct kvm_regs regs = { .rip = 0x, .rax = 2, .rbx = 2, .rflags = 0x2, }; ioctl(vcpufd, KVM_SET_REGS, ®s);

好了,东西都准备好了,我们可以开始运行vCPU了:

Ruby Code复制内容到剪贴板 while (1) { ioctl(vcpufd, KVM_RUN, NULL); switch (run->exit_reason) { /* Handle exit */ } }

我们需要根据run->exit_reason来处理kvm的退出状态,比如guest 关机:

Ruby Code复制内容到剪贴板 case KVM_EXIT_HLT: puts("KVM_EXIT_HLT"); return 0;

初始化失败:

Ruby Code复制内容到剪贴板 case KVM_EXIT_FAIL_ENTRY: errx(1, "KVM_EXIT_FAIL_ENTRY: hardware_entry_failure_reason = 0x%llx", (unsigned long long)run->fail_entry.hardware_entry_failure_reason); case KVM_EXIT_INTERNAL_ERROR: errx(1, "KVM_EXIT_INTERNAL_ERROR: suberror = 0x%x", run->internal.suberror);

以及需要进行设备的模拟器,在这里,只有一个端口为0x3f8的串口设备。模拟设备的效果就是把字符打印出来:

Ruby Code复制内容到剪贴板 case KVM_EXIT_IO: if (run->io.direction == KVM_EXIT_IO_OUT && run->io.size == 1 && run->io.port == 0x3f8 && run->io.count == 1) putchar(*(((char *)run) + run->io.data_offset)); else errx(1, "unhandled KVM_EXIT_IO"); break;

测试结果:

Ruby Code复制内容到剪贴板 tree@tree-OptiPlex-:~/Desktop$ gcc -o kvmtest kvmtest.c tree@tree-OptiPlex-:~/Desktop$ ./kvmtest KVM_EXIT_HLT

qemu-kvm中,qemu的主要任务就是KVM_EXIT_IO, KVM_EXIT_MMIO之后的虚拟设备的模拟,以及KVM_RUN之前设置好相关的设备的东西并进行初始化。

以上所述是小编给大家介绍的ubuntu.环境下使用kvm ioctl接口实现最简单的虚拟机,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对积木网网站的支持!

ubuntu怎么选择最快的更新源? ubuntu更改最快的更新源的教程 给ubuntu选择最快的更新源1、打开设置》》软件和更新,如下图2、下载自:下拉菜单,选择其他站点3、你可以选择最佳服务器,也可以指定你满意的服

Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法 接触Ubuntu系统不久,发现无线网络环境下安装UbuntuServer一个不太人性化的设计:在安装过程中选择无线网卡,即使用无线网络安装(此时需要选择Wi-Fi网

Ubuntu登录界面怎么截图? 通过编写脚本、赋予脚本可执行权限、然后执行脚本,可以实现对Ubuntu登录界面的截图。1、打开文本编辑器,输入以下代码:chvt7;sleep5s;DISPLAY=:0XAUTHORITY

标签: ubuntu 12.04安装

本文链接地址:https://www.jiuchutong.com/biancheng/352871.html 转载请保留说明!

上一篇:ubuntu开启wifi热点的图文教程(亲测16.04与14.04可用)(ubuntu20设置wifi)

下一篇:ubuntu怎么选择最快的更新源? ubuntu更改最快的更新源的教程(ubuntu20设置)

  • 哪些情形免征契税
  • 劳务公司一般纳税人可以抵扣进项税吗
  • 保险佣金个人所得税怎么算
  • 缴纳个人所得税会计分录怎么写
  • 固定资产的处理包括
  • 金税盘维护费抵税账务处理
  • 企业减免的税费账务处理
  • 发票开票人是管理员可以吗
  • 利润表中财务费用是负数怎么算营业利润
  • 挂失申请怎么写
  • 税务社保扣取300干嘛的
  • 发票密码区超出还能用吗
  • 维修费增值税怎么开
  • 学校的纳税人识别号是不是社会信用代码
  • 销售赠品怎么入账
  • 个税申报异常怎么办
  • 增值税普票丢失的最新处理
  • 补提上个月折旧会计分录
  • 支付员工补偿金计入管理费用的福利费用吗
  • 递延所得税怎么产生
  • 汽车修理费抵扣怎么做账
  • 应收利息和应计利息有什么不同
  • 代持的股份
  • win11开机蓝屏怎么解决
  • 卸载软件怎么清理干净
  • 原始股解禁后减持的条件是什么
  • windows无法验证此设备数字签名
  • blender不支持win7
  • js设置input值
  • 企业接受捐赠的税务处理
  • 划拨建设用地使用权没有使用期限的限制
  • 【深度学习】AI一键换天
  • python字符串的符号
  • 生产车间年终奖计入什么科目
  • windows安装mq
  • 生产防疫物资的厂家联系方式
  • 申报表填完后下一步是什么
  • 资产负债表固定资产清理
  • 利润表调整了资产负债表怎么调整
  • 工程施工与工程结算账务处理
  • 小规模收入怎么算
  • 主办会计的工作内容和职责
  • 无偿赠送货物如何申报增值税
  • 月末假退月初假领
  • 被代持股份的股东需要负责吗
  • 总资产报酬率的利息支出是财务费用吗
  • 股东的义务是什么意思?
  • 如何理解现金折扣
  • 什么是现金等价物
  • sql sever实验指导
  • windowsserver2003ftp服务器怎么搭建
  • mscorsvw.exe是什么进程
  • 如何关掉数据
  • 2021年win10累积更新
  • centos5.10安装
  • win8系统搜索在哪里
  • symtray.exe - symtray是什么进程 有何作用
  • w10组策略
  • xp系统怎么打开启动项
  • 电脑操作系统win8
  • 找不到config/index taro
  • linux系统开发环境
  • linux支持哪些类型的设备
  • windows10窗口设置在哪
  • android 自定义drawable
  • .android
  • 遮罩层中对象的作用是
  • JS实现的base64加密、md5加密及sha1加密详解
  • jquery点击移动div
  • 接口回调java
  • 猫的游戏视频
  • python 系统监控
  • 个人去税务局开劳务费怎么开
  • 党建阵地作用发挥不好
  • 个人所得税每年都可以退税吗
  • 税务跨区变更需要什么资料
  • 怎么在年报查询个人信息
  • 税务总局官网投诉
  • 房地产开发企业资质管理规定
  • 税务绩效管理主要内容
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设