亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Ubuntu > 正文

ubuntu12.04環境下使用kvm ioctl接口實現最簡單的虛擬機

2020-10-18 20:35:39
字體:
來源:轉載
供稿:網友

qemu、virtual box、vmware、xen都是虛擬機,一般用戶接觸到的virtual box和vmware比較多,都是用來ubuntu中跑windows,或者windows中跑ubuntu的。

qemu其實是鼎鼎大名的最基礎的開源模擬器,可以純軟件模擬x86、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平臺下ubuntu12.04中實現一個最簡單的模擬器,計算2+2的結果并通過io端口輸出。

內核中kvm api的介紹可以看:Documentation/virtual/kvm/api.txt,其它的一些文檔:Documentation/virtual/kvm/。完整的源碼:https://lwn.net/Articles/658512/。

使用kvm的真正的虛擬機,模擬了很多虛擬的設備和固件,還有復雜的初始化狀態(各個設備的初始化,CPU寄存器的初始化等),以及內存的初始化。本文所述的模擬器demo,將使用如下16bit的x86的代碼(為什么是16bit呢,因為x86一上電是實模式,工作于16bit;之后再切換到32bit的保護模式的):

Ruby Code復制內容到剪貼板
  1. mov $0x3f8, %dx     
  2. add %bl, %al     
  3. add $'0', %al     
  4. out %al, (%dx)     
  5. mov $'/n', %al     
  6. out %al, (%dx)     
  7. hlt    

這段代碼充當了guest os,基本上算是一個裸奔的系統了。它實現了2+2,然后再加上'0',把4轉為ascii的'4',并通過端口0x3f8輸出。然后再輸出了'/n',就關機了。

我們把這段代碼對應的二進制存到數組里面:

Ruby Code復制內容到剪貼
  1.   const uint8_t code[] = {     
  2. 0xba, 0xf8, 0x03, /* mov $0x3f8, %dx */     
  3. 0x00, 0xd8,       /* add %bl, %al */     
  4. 0x04, '0',        /* add $'0', %al */     
  5. 0xee,             /* out %al, (%dx) */     
  6. 0xb0, '/n',       /* mov $'/n', %al */     
  7. 0xee,             /* out %al, (%dx) */     
  8. 0xf4,             /* hlt */     
  9.    };    

怎么得到這些機器碼呢?

Ruby Code復制內容到剪貼板
  1. shuyin.wsy@10-101-175-19:~$ cat simple_os.asm     
  2.     mov $0x3f8, %dx     
  3.     add %bl, %al     
  4.     add $'0', %al     
  5.     out %al, (%dx)     
  6.     mov $'/n', %al     
  7.     out %al, (%dx)     
  8.     hlt     
  9. shuyin.wsy@10-101-175-19:~$ as -o simple_os.o simple_os.asm     
  10. shuyin.wsy@10-101-175-19:~$ objdump -d  simple_os.o     
  11. simple_os.o:     file format elf64-x86-64     
  12. Disassembly of section .text:     
  13. 0000000000000000 <.text>:     
  14.    0:   66 ba f8 03             mov    $0x3f8,%dx     
  15.    4:   00 d8                   add    %bl,%al     
  16.    6:   04 30                   add    $0x30,%al     
  17.    8:   ee                      out    %al,(%dx)     
  18.    9:   b0 0a                   mov    $0xa,%al     
  19.    b:   ee                      out    %al,(%dx)     
  20.    c:   f4                      hlt    

可以在這個網頁上查看匯編指令,以及對應的機器碼:http://x86.renejeschke.de/
注意開頭多了一個0x66,解釋如下:

http://wiki.osdev.org/X86-64_Instruction_Encoding里面的Prefix group 3

所以我們需要在simple_os.asm文件的開頭添加.code16,這樣的話就對了,但是objdump顯示的又不對了,需要這樣使用才行:

Ruby Code復制內容到剪貼板
  1. shuyin.wsy@10-101-175-19:~$ objdump -d -Mintel,i8086 simple_os.o     
  2. simple_os.o:     file format elf64-x86-64     
  3. Disassembly of section .text:     
  4. 0000000000000000 <.text>:     
  5.    0:   ba f8 03                mov    dx,0x3f8     
  6.    3:   00 d8                   add    al,bl     
  7.    5:   04 30                   add    al,0x30     
  8.    7:   ee                      out    dx,al     
  9.    8:   b0 0a                   mov    al,0xa     
  10.    a:   ee                      out    dx,al     
  11.    b:   f4                      hlt     
  12. https://sourceware.org/binutils/docs/as/i386_002d16bit.html   
  13. http://stackoverflow.com/questions/1737095/how-do-i-disassemble-raw-x86-code  

我們會把這段代碼,放到虛擬物理內存,也就是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指向第二個頁面的起始位置0x1000。
除此之外,我們還有一個虛擬的串口設備,端口是0x3f8,8bit,用于輸出字符。

為了實現一個虛擬機,我們首先需要打開/dev/kvm:

Ruby Code復制內容到剪貼板
  1. kvm = open("/dev/kvm", O_RDWR | O_CLOEXEC);    

在使用kvm之前,需要使用KVM_GET_API_VERSION ioctl()去檢查下kvm的版本是否正確,看看是否為api12,是才可以繼續運行

Ruby Code復制內容到剪貼板
  1.  ret = ioctl(kvm, KVM_GET_API_VERSION, NULL);     
  2.    if (ret == -1)     
  3. err(1, "KVM_GET_API_VERSION");     
  4.    if (ret != 12)     
  5. errx(1, "KVM_GET_API_VERSION %d, expected 12", ret);    

檢查完api版本后,可以使用KVM_CHECK_EXTENSION ioctl()去檢查其它extensions是否可用,比如KVM_SET_USER_MEMORY_REGION,用來檢查kvm是否支持硬件影子頁表(http://royluo.org/2016/03/13/kvm-mmu-virtualization/):

Ruby Code復制內容到剪貼板
  1.  ret = ioctl(kvm, KVM_CHECK_EXTENSION, KVM_CAP_USER_MEMORY);     
  2.    if (ret == -1)     
  3. err(1, "KVM_CHECK_EXTENSION");     
  4.    if (!ret)     
  5. errx(1, "Required extension KVM_CAP_USER_MEM not available");   

然后再創建一個虛擬機vm,這個vm和內存,設備,所有的vCPU相關,在host系統中對應一個進程:

Ruby Code復制內容到剪貼板
  1. 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復制內容到剪貼板
  1. mem = mmap(NULL, 0x1000, PROT_READ | PROT_WRITE, MAP_SHARED | MAP_ANONYMOUS, -1, 0);     
  2. memcpy(mem, code, sizeof(code));    

然后我們需要把host 虛擬空間的內存和guest os虛擬物理內存的映射關系使用KVM_SET_USER_MEMORY_REGION ioctl()告知kvm:

Ruby Code復制內容到剪貼板
  1. struct kvm_userspace_memory_region region = {     
  2. .slot = 0,     
  3. .guest_phys_addr = 0x1000,     
  4. .memory_size = 0x1000,     
  5. .userspace_addr = (uint64_t)mem,     
  6.    };     
  7.    ioctl(vmfd, KVM_SET_USER_MEMORY_REGION, ®ion);    

這樣,當guest os訪問到虛擬物理內存的0x1000~0x2000之間的話,kvm會直接訪問到mem所對應的真實的物理內存。

現在,我們有了一個虛擬機vm,有了一些虛擬物理內存,內存里面有guest os的代碼,那么我們需要給虛擬機添加一個核(vCPU),對應一個線程。當然也可以多核(vCPUs,調用多次KVM_CREATE_VCPU):

Ruby Code復制內容到剪貼板
  1. 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復制內容到剪貼板
  1. mmap_size = ioctl(kvm, KVM_GET_VCPU_MMAP_SIZE, NULL);     
  2. 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復制內容到剪貼板
  1. ioctl(vcpufd, KVM_GET_SREGS, &sregs);     
  2. sregs.cs.base = 0;     
  3. sregs.cs.selector = 0;     
  4. ioctl(vcpufd, KVM_SET_SREGS, &sregs);     
  5.   
  6.    struct kvm_regs regs = {     
  7. .rip = 0x1000,     
  8. .rax = 2,     
  9. .rbx = 2,     
  10. .rflags = 0x2,     
  11.    };     
  12.    ioctl(vcpufd, KVM_SET_REGS, ®s);    


好了,東西都準備好了,我們可以開始運行vCPU了:

Ruby Code復制內容到剪貼板
  1.    while (1) {     
  2. ioctl(vcpufd, KVM_RUN, NULL);     
  3. switch (run->exit_reason) {     
  4. /* Handle exit */     
  5. }     
  6.    }    

我們需要根據run->exit_reason來處理kvm的退出狀態,比如guest 關機:

Ruby Code復制內容到剪貼板
  1. case KVM_EXIT_HLT:     
  2.  puts("KVM_EXIT_HLT");     
  3.  return 0;    

初始化失?。?/p>

Ruby Code復制內容到剪貼板
  1.  case KVM_EXIT_FAIL_ENTRY:     
  2.   errx(1, "KVM_EXIT_FAIL_ENTRY: hardware_entry_failure_reason = 0x%llx",     
  3. (unsigned long long)run->fail_entry.hardware_entry_failure_reason);      
  4.  case KVM_EXIT_INTERNAL_ERROR:     
  5.   errx(1, "KVM_EXIT_INTERNAL_ERROR: suberror = 0x%x",     
  6.        run->internal.suberror);    

以及需要進行設備的模擬器,在這里,只有一個端口為0x3f8的串口設備。模擬設備的效果就是把字符打印出來:

Ruby Code復制內容到剪貼板
  1. case KVM_EXIT_IO:     
  2.         if (run->io.direction == KVM_EXIT_IO_OUT &&     
  3.             run->io.size == 1 &&     
  4.             run->io.port == 0x3f8 &&     
  5.             run->io.count == 1)     
  6.         putchar(*(((char *)run) + run->io.data_offset));     
  7.         else     
  8.         errx(1, "unhandled KVM_EXIT_IO");     
  9.         break;    

測試結果:

Ruby Code復制內容到剪貼板
  1. tree@tree-OptiPlex-7010:~/Desktop$ gcc -o kvmtest kvmtest.c     
  2. tree@tree-OptiPlex-7010:~/Desktop$ ./kvmtest      
  3. KVM_EXIT_HLT    

qemu-kvm中,qemu的主要任務就是KVM_EXIT_IO, KVM_EXIT_MMIO之后的虛擬設備的模擬,以及KVM_RUN之前設置好相關的設備的東西并進行初始化。

以上所述是小編給大家介紹的ubuntu12.04環境下使用kvm ioctl接口實現最簡單的虛擬機,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對武林網網站的支持!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲国产精彩中文乱码av| 日韩精品免费在线| 亚洲影院在线看| 午夜欧美大片免费观看| 久久av在线看| 136fldh精品导航福利| 丁香五六月婷婷久久激情| 国产精品va在线播放| 久久777国产线看观看精品| 亚洲综合av影视| 91精品久久久久久久久中文字幕| 久久久在线视频| 国产在线视频不卡| 中文字幕在线精品| 色综合天天综合网国产成人网| 日韩美女激情视频| 久久这里只有精品视频首页| 中文字幕视频在线免费欧美日韩综合在线看| www亚洲精品| 亚洲成人av在线| 在线播放日韩av| 久久精品成人动漫| 国产精品视频一区二区高潮| 亚洲欧美一区二区三区情侣bbw| 成人精品久久一区二区三区| 精品久久久久久久久久久久久| 最近2019年中文视频免费在线观看| 中文字幕久精品免费视频| 欧美激情a∨在线视频播放| 久久久久国产视频| 午夜精品久久17c| 久久夜色精品国产欧美乱| 久久久999精品免费| 欧美精品精品精品精品免费| 欧美一级黄色网| 琪琪亚洲精品午夜在线| 日韩在线观看免费高清完整版| 国产日韩欧美日韩大片| 永久免费看mv网站入口亚洲| 日本一区二区在线免费播放| 亚洲成在人线av| 国产精品人成电影在线观看| 8050国产精品久久久久久| 欧美亚洲另类在线| 欧美俄罗斯乱妇| 一本一本久久a久久精品牛牛影视| 欧美激情亚洲另类| 亚洲自拍偷拍色图| 日本成人免费在线| 日韩av影视综合网| 亚洲老头同性xxxxx| 国产精品中文字幕在线观看| 久久国产精品电影| 97精品伊人久久久大香线蕉| 亚洲综合在线做性| 亚洲综合中文字幕68页| 97精品视频在线观看| 日韩av中文字幕在线播放| 国产视频999| 国产九九精品视频| 欧美激情免费在线| 欧美大全免费观看电视剧大泉洋| 97久久久久久| 亚洲视频第一页| 亚洲女同性videos| 亚洲欧美一区二区三区在线| 久久99久久99精品中文字幕| 国产精品一区二区久久久久| 欧美性猛交xxxx乱大交3| 日韩精品一区二区三区第95| 成人国产精品免费视频| 欧美电影在线免费观看网站| 欧美激情网站在线观看| 日韩免费av在线| 国产偷亚洲偷欧美偷精品| 亚洲福利精品在线| 久久91亚洲人成电影网站| 欧美激情精品久久久| 亚洲精品小视频在线观看| 国产日韩精品综合网站| 亚洲国产精品网站| 久久夜精品va视频免费观看| 久久精品国产91精品亚洲| 97视频免费观看| 91禁外国网站| 欧美成人精品激情在线观看| 91国产精品视频在线| 欧美日韩国产精品一区二区不卡中文| 国产一区二区三区在线观看网站| 欧美电影免费观看高清完整| 成人亚洲综合色就1024| 中文字幕精品av| 欧美日韩亚洲激情| 国产精品免费观看在线| 国产精品久久久久久五月尺| 亚洲精品国产拍免费91在线| 欧美放荡办公室videos4k| 国产suv精品一区二区| 欧美日韩ab片| 国产免费亚洲高清| 91精品国产高清久久久久久久久| 日韩免费观看视频| 国产精品夜间视频香蕉| 亚洲成人精品视频在线观看| 亚洲精品自在久久| 日韩av影视综合网| 亚洲成人精品久久| 亚洲精品国偷自产在线99热| 国产伦精品免费视频| 欧美自拍视频在线| 亚洲理论电影网| 亚洲精品白浆高清久久久久久| 亚洲欧美中文另类| 欧美在线性爱视频| 热久久视久久精品18亚洲精品| 亚洲精品网址在线观看| 欧美裸体xxxx极品少妇| 欧美午夜精品伦理| 欧美极度另类性三渗透| 成人网页在线免费观看| 亚洲欧美成人在线| 一区二区成人精品| 日韩中文视频免费在线观看| 欧美亚洲免费电影| 亚洲电影在线观看| 亚洲欧美日韩综合| 精品动漫一区二区三区| 亚洲国产欧美一区二区三区同亚洲| 欧美成人精品一区二区三区| 久99久在线视频| 亚洲理论片在线观看| 亚洲国产精品久久久| 久久久精品视频在线观看| 亚洲激情视频在线播放| 日韩av一区在线观看| 色综合久久精品亚洲国产| 国产精品99久久久久久白浆小说| 尤物99国产成人精品视频| 欧美极品少妇xxxxⅹ免费视频| 55夜色66夜色国产精品视频| 热久久这里只有精品| 国产成人高潮免费观看精品| 日韩免费黄色av| 尤物九九久久国产精品的特点| 亚洲欧美中文日韩v在线观看| 亚洲欧美制服第一页| 91网在线免费观看| 日韩欧美中文字幕在线播放| 92福利视频午夜1000合集在线观看| 国产视频精品自拍| 精品久久久久久久久国产字幕| 日韩免费观看在线观看| 亚洲精品av在线| 欧美成人午夜激情视频| 日韩av在线免费看| 精品久久久久久国产| 亚洲2020天天堂在线观看| 久久精品国产一区二区电影| 色香阁99久久精品久久久| 精品亚洲一区二区三区四区五区| 日韩美女av在线| 亚洲国产小视频在线观看| 欧美综合在线第二页| 日韩电影中文字幕一区|