精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

virtio I/O通信流程及設備框架的實現

openEuler ? 來源:openEuler ? 作者:openEuler ? 2022-03-10 13:37 ? 次閱讀

virtio 是一種通用的半虛擬化的 I/O 通信協議,提供了一套前后端 I/O 通信的的框架協議和編程接口。根據該協議實現的設備通過前后端的配合,相比全模擬設備可以大幅減少陷入陷出以及內存拷貝的次數,使 guest 獲得高效的 I/O 性能。作為目前虛擬化標準的通用協議規范,經歷了 0.95、1.0、1.1 三個版本的演進。根據 0.95 版本實現的稱為傳統 virtio 設備,1.0 版本修改了一些 PCI 配置空間的訪問方式和 virtioqueue 的優化和特定設備的約定,1.1 版本則增加了 packed virtqueue 的支持,詳細可以參考官方發布的 virtio 協議規范。

之所以稱 virtio 是一種半虛擬化的解決方案,是因為其首先需要在主機側通過軟件創建 virito 的后端設備,其次在 Guest 要有對應的設備前端驅動,前后端通過共享內存進行通信。virtio 規范定義了設備的控制和數據面接口,控制面接口包括設備狀態、feature 的協商等,數據面則包括共享內存的數據布局定義以及前后端的通知方式。基于 virtio 協議,目前已衍生出了 virtio-blk、virtio-net、virtio-scsi、virtio-mem 等各種各樣的半虛擬化設備。virtio 設備可以支持多種傳輸層協議,既可以掛載到 MMIO 總線,也可以作為 PCI 設備,另外還可以支持基于標準 I/O 通道機制的 S/390 設備。

4963ac88-a026-11ec-952b-dac502259ad0.png

鑒于 virtio 設備具備較好的性能及通用性,StratoVirt 自然也支持,StratoVirt 中 virtio 設備的實現架構以及 I/O 通信流程如上圖所示。下面就基于目前最新的代碼,探究一下 StratoVirt 中 virtio 設備的代碼實現框架。

VirtioDevice Trait

StratoVirt 的 virtio crate 提供了 virtio 設備的通用接口以及所有 virtio 設備的相關實現。其中,lib.rs 中定義了為所有 virtio 設備定義的 VirtioDevice Trait。每種 virtio 設備都需要實現自定義的 VirtioDevice 接口。

///Thetraitforvirtiodeviceoperations.
pubtraitVirtioDevice:Send{
///Realizelowleveldevice.
fnrealize(&mutself)->Result<()>;

///Unrealizelowleveldevice
fnunrealize(&mutself)->Result<()>{
bail!("Unrealizeofthevirtiodeviceisnotimplemented");
}

///Getthevirtiodevicetype,refertoVirtioSpec.
fndevice_type(&self)->u32;

///Getthecountofvirtiodevicequeues.
fnqueue_num(&self)->usize;

///Getthequeuesizeofvirtiodevice.
fnqueue_size(&self)->u16;

///Getdevicefeaturesfromhost.
fnget_device_features(&self,features_select:u32)->u32;

///Setdriverfeaturesbyguest.
fnset_driver_features(&mutself,page:u32,value:u32);

///Readdataofconfigfromguest.
fnread_config(&self,offset:u64,data:&mut[u8])->Result<()>;

///Writedatatoconfigfromguest.
fnwrite_config(&mutself,offset:u64,data:&[u8])->Result<()>;

///Activatethevirtiodevice,thisfunctioniscalledbyvcputhreadwhenfrontend
///virtiodriverisreadyandwrite`DRIVER_OK`tobackend.
///
///#Arguments
///
///*`mem_space`-Systemmem.
///*`interrupt_evt`-Theeventfdusedtosendinterrupttoguest.
///*`interrupt_status`-Theinterruptstatuspresenttoguest.
///*`queues`-Thevirtioqueues.
///*`queue_evts`-Thenotifiereventsfromguest.
fnactivate(
&mutself,
mem_space:Arc,
interrupt_cb:Arc,
queues:&[Arc>],
queue_evts:Vec,
)->Result<()>;

///Deactivatevirtiodevice,thisfunctionremoveeventfd
///ofdeviceoutoftheeventloop.
fndeactivate(&mutself)->Result<()>{
bail!(
"Resetthisdeviceisnotsupported,virtiodevtypeis{}",
self.device_type()
);
}

///Resetvirtiodevice.
fnreset(&mutself)->Result<()>{
Ok(())
}

///UpdatethelowlevelconfigofMMIOdevice,
///forexample:updatetheimagesfilefdofvirtioblockdevice.
///
///#Arguments
///
///*`_file_path`-Therelatedbackendfilepath.
fnupdate_config(&mutself,_dev_config:OptiondynConfigCheck>>)->Result<()>{
bail!("Unsupportedtoupdateconfiguration")
}
}
  • realize()/unrealize(): 這一組接口用于具現化/去具現化具體的 virtio 設備。具現化做的一些具體操作包括設置支持的 features、設備特有的屬性(如網卡的 mac)、初始化連接 Host 后端設備等。
  • set_driver_features():將前端驅動支持的 features 與后端模擬設備支持的 features 進行協商后,設置最終實現的 features。
  • read_config()/write_config():virtio 協議規范為每種 virtio 設備定義了自定義的配置空間,這組接口就是用來讀寫這部分配置數據。
  • activate()/deactivate(): 激活/去激活設備,負責綁定/解綁后端、加入/移除 I/O 循環。
  • reset():虛擬機重啟時某些設備需要重置。
  • update_config():支持輕量機型下的 virtio-mmio 設備動態綁定/解綁后端,實現 virtio-mmio 設備的模擬熱插拔。

virtqueue

499e7368-a026-11ec-952b-dac502259ad0.png

virtio 設備可以有一個或多個隊列,每個隊列有描述符表、available ring、used ring 三個部分。當前 StratoVirt 的 virtio 設備均遵循 1.0 規范,隊列的內存布局僅支持 Split Vring 的方式。queue.rs 中定義了一系列針對隊列操作及查詢的接口。所有的 I/O 請求數據信息以描述符的形式存放在描述符表中,前端準備好數據后更新 available ring 告訴后端還有哪些 I/O 待發送,后端執行完 I/O 更新 used ring 通知前端。不同設備的 I/O 處理不盡相同,但是核心的 virtqueue 操作是一樣的。

pubstructSplitVring{
///Regioncacheinformation.
pubcache:Option,
///Guestphysicaladdressofthedescriptortable.
///Thetableiscomposedofdescriptors(SplitVringDesc).
pubdesc_table:GuestAddress,

///Guestphysicaladdressoftheavailablering.
///Theringiscomposedofflags(u16),idx(u16),ring[size](u16)andused_event(u16).
pubavail_ring:GuestAddress,

///Guestphysicaladdressoftheusedring.
///Theringiscomposedofflags(u16),idx(u16),used_ring[size](UsedElem)andavail_event(u16).
pubused_ring:GuestAddress,

///Hostaddresscache.
pubaddr_cache:VirtioAddrCache,

///Indicatewhetherthequeueconfigurationisfinished.
pubready:bool,

///Themaximalsizeinelementsofferedbythedevice.
pubmax_size:u16,

///Thequeuesizesetbyfrontend.
pubsize:u16,

///Interruptvectorindexofthequeueformsix
pubvector:u16,

///Thenextindexwhichcanbepoppedintheavailablevring.
next_avail:Wrapping<u16>,

///Thenextindexwhichcanbepushedintheusedvring.
next_used:Wrapping<u16>,

///Theindexoflastdescriptorusedwhichhastriggeredinterrupt.
last_signal_used:Wrapping<u16>,
}

virtio-mmio 設備

StratoVirt 目前提供兩種機型:輕量機型和標準機型。輕量機型由于需要追求極致的啟動速度以及內存底噪開銷,因此只支持掛載數量有限的 virtio-mmio 設備。而標準機型面向傳統的標準云化場景,對于 I/O 設備的性能要求較高,且需要支持熱插拔滿足資源彈性,因此標準機型支持將 virtio 設備以 PCI 設備掛載在模擬的 PCI 總線上。目前標準機型只支持配置 virtio-pci 設備,不支持 virtio-mmio 設備。

結構體 VirtioMmioDevice 定義了一個通用的 virtio-mmio 設備,其中的 device 即為實現了 VirtioDevice 這個 trait 的具體的 virtio 設備結構,可以是網卡、磁盤等。VirtioMmioState 結構體中存放了 virtio-mmio 設備的控制寄存器,并且為其實現了對應的讀寫接口 read_common_config()/write_common_config()。virtio-mmio 設備的配置空間布局如下圖所示:

49b533a0-a026-11ec-952b-dac502259ad0.png

interrupt_evt 通過 irqfd 向虛擬機注入中斷,host_notify_info 則為每個隊列創建了一個 eventfd,虛擬機利用 ioeventfd 機制陷出到 StratoVirt 執行后端的 I/O 處理。

pubstructVirtioMmioDevice{
//Theentityoflowleveldevice.
pubdevice:ArcdynVirtioDevice>>,
//EventFdusedtosendinterrupttoVM
interrupt_evt:EventFd,
//Interruptstatus.
interrupt_status:Arc,
//HostNotifyInfousedforguestnotifier
host_notify_info:HostNotifyInfo,
//Thestateofvirtiommiodevice.
state:VirtioMmioState,
//Systemaddressspace.
mem_space:Arc,
//Virtioqueues.
queues:Vec>>,
//SystemResourceofdevice.
res:SysRes,
}

VirtioMmioDevice 實現了 realize 接口完成設備的具現化:

  1. 調用各設備實現的 VirtioDevice trait 的具現化接口。
  2. virtio-mmio 設備掛載在了系統總線上,StratoVirt 為每個設備分配 512 字節的配置空間。除此之外,需要為其注冊 irqfd 以便后續 I/O 完成后向虛擬機注入中斷。這些信息都保存在 SysRes 數據結構中。
  3. 添加內核啟動參數,通過內核啟動參數將設備的內存區間及中斷號信息直接告訴 Guest。
pubfnrealize(
mutself,
sysbus:&mutSysBus,
region_base:u64,
region_size:u64,
#[cfg(target_arch="x86_64")]bs:&Arc>,
)->ResultSelf>>>{
self.device
.lock()
.unwrap()
.realize()
.chain_err(||"Failedtorealizevirtio.")?;

ifregion_base>=sysbus.mmio_region.1{
bail!("Mmioregionspaceexhausted.");
}
self.set_sys_resource(sysbus,region_base,region_size)?;
letdev=Arc::new(self));
sysbus.attach_device(&dev,region_base,region_size)?;

#[cfg(target_arch="x86_64")]
bs.lock().unwrap().kernel_cmdline.push(Param{
param_type:"virtio_mmio.device".to_string(),
value:format!(
"{}@0x{:08x}:{}",
region_size,
region_base,
dev.lock().unwrap().res.irq
),
});
Ok(dev)
}

前端驅動加載過程中會讀寫設備的配置空間,前后端完成 feature 的協商,一切 OK 后前端驅動將向配置空間寫狀態,后端設備將會調用 activate 方法激活設備。當觸發激活時,前端已為這三個部分分配了內存空間,Guest 物理地址(GPA)已寫入設備的配置空間,后端需要將 GPA 地址轉化為 Host 虛擬地址(HVA)。隨后,就可以根據隊列配置創建隊列,并將 I/O 的 eventfd 加入事件循環激活設備開始 I/O 通信。

virtio-pci 設備

如上所述,virtio 設備也可以作為一個 PCI 類設備掛載到 PCI 總線上。類似的,在 StratoVirt 中用結構體 VirtioPciDevice 來表示一個 virtio-pci 設備。既然是作為一個 PCI 設備,virtio-pci 就需要擁有符合 PCI 規范擁有 PCI 設備的配置空間,Guest 啟動后通過 PCI 設備樹枚舉來發現設備,而不是像 virtio-mmio 設備一樣直接通過內核啟動參數告訴 Guest。

pubstructVirtioPciDevice{
///Nameofthisdevice
name:String,
///Theentityofvirtiodevice
device:ArcdynVirtioDevice>>,
///Deviceid
dev_id:Arc,
///Devfn
devfn:u8,
///Ifthisdeviceisactivatedornot.
device_activated:Arc,
///MemoryAddressSpace
sys_mem:Arc,
///Pciconfigspace.
config:PciConfig,
///VirtiocommonconfigrefertoVirtioSpec.
common_config:Arc>,
///PrimaryBus
parent_bus:Weak>,
///Eventfdsusedfornotifyingtheguest.
notify_eventfds:NotifyEventFds,
///Thefunctionforinterrupttriggering
interrupt_cb:Option>,
///Virtioqueues.ThevectorandQueuewillbesharedacrossingthread,soallwithArc>wrapper.
queues:ArcVec>>>>,
///Multi-Functionflag.
multi_func:bool,
}

VirtioPciDevice 通過實現 PciDevOps trait 的 realize()方法完成設備的具現化:

  1. 初始化 PCI 配置寄存器
  2. 將 virtio 協議規定的 common configuration、notifications、ISR status、Device-specific configuration 作為四個 PCI 設備的 capability, 對應數據的內存空間則映射到第 3 個 BAR 空間的不同部分。配置空間布局如下圖所示:

49ca4dda-a026-11ec-952b-dac502259ad0.png

  1. 前端驅動對于各空間的訪問的回調函數由 modern_mem_region_init()注冊,當前端讀寫這部分內存區間時會陷出到 StratoVirt 執行注冊的回調接口。每個隊列在 notification cap 指向的空間中占據 4 個字節,StratoVirt 為每個隊列的 4 個字節空間注冊 ioeventfd。前端驅動準備好某個隊列后,就會寫對應隊列的這 4 個字節的地址空間,后端借助 ioeventfd 機制收到通知后陷出進行 host 側的 I/O 下發。
  2. 中斷機制采用 MSI-X,向量表和 pending 位圖則位于第 2 個 BAR 空間。

審核編輯:郭婷
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • PCI
    PCI
    +關注

    關注

    4

    文章

    662

    瀏覽量

    130184
  • 虛擬化
    +關注

    關注

    1

    文章

    366

    瀏覽量

    29774

原文標題:StratoVirt 的 virtio 設備模擬是如何實現的

文章出處:【微信號:openEulercommunity,微信公眾號:openEuler】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    linux設備virtio組織關系及設備初始化調用流程

    從哪里開始初始化的?要理清這些關系需要以linux設備驅動模型為背景展開討論。這篇文章,我們以linux kernel 3.10代碼為例,分析一下virtio的相關組織關系,以及設備
    的頭像 發表于 09-25 15:47 ?4645次閱讀
    linux<b class='flag-5'>設備</b>中<b class='flag-5'>virtio</b>組織關系及<b class='flag-5'>設備</b>初始化調用<b class='flag-5'>流程</b>

    請問怎樣設計并實現閃存設備I/O軟件?

    閃存設備管理技術的現狀及存在的問題是什么?閃存設備I/O軟件的分層結構是怎樣的?怎樣設計并實現閃存設備
    發表于 04-27 06:44

    怎樣去操作I/O設備呢?有哪些流程

    什么是I/O設備I/O設備是由哪些部分組成的?內存映射I
    發表于 08-13 07:04

    StratoVirt 的 virtio-blk 設備是如何實現的?

    平臺下虛擬磁盤的一種實現方式,本質上為一種半模擬技術。virtio-blk 設備中采用 io_event_fd 進行前端到后端通知,采用中斷注入方式實現后端到前端的通知,并通過 IO
    發表于 06-29 10:37

    StratoVirt 中的虛擬網卡是如何實現的?

    基于 virtio 協議的半虛擬化框架實現前后端通信Virtio 協議是一種在半虛擬化場景中使用的
    發表于 08-10 11:16

    RT-Tread設備驅動開發之I/O設備框架簡析

    獲得正確的設備驅動,然后通過這個設備驅動與底層I/O硬件設備進行數據(或控制)交互。I/
    發表于 03-15 14:44

    輸入輸出設備I/O設備總結

    I/O設備又叫輸入輸出設備。對于I/O bus 包含數據總線、控制總線、地址總線;每一個
    發表于 11-24 09:28 ?3605次閱讀
    輸入輸出<b class='flag-5'>設備</b><b class='flag-5'>I</b>/<b class='flag-5'>O</b><b class='flag-5'>設備</b>總結

    如何減少器件間通信所用的I/O引腳數實現雙向通信

    隨著嵌入式系統的小型化趨勢,市場對減少器件間通信所用的I/O 引腳數的需求與日俱增。Microchip 開發的UNI/O? 總線滿足了這一需求,這一低成本且易于
    發表于 04-27 16:14 ?8次下載
    如何減少器件間<b class='flag-5'>通信</b>所用的<b class='flag-5'>I</b>/<b class='flag-5'>O</b>引腳數<b class='flag-5'>實現</b>雙向<b class='flag-5'>通信</b>

    I/O軟件模擬虛擬化和類虛擬化

    的標準化接口。Virtio成為整個問題的焦點:不管是SPDK/vhost、還是vDPA加速,都是圍繞著Virtio接口展開。 1 I/O設備
    的頭像 發表于 10-13 11:09 ?2587次閱讀

    I/O虛擬化及Virtio接口介紹

    I/O虛擬化是計算機虛擬化最復雜的部分,因為涉及到CPU、操作系統、Hypervisor以及I/O設備的相互配合。
    的頭像 發表于 10-26 17:21 ?4044次閱讀

    探究I/O虛擬化及Virtio接口技術(上)

    I/O虛擬化是SmartNIC/DPU/IPU中最核心的部分,AWS NITRO就是從I/O硬件虛擬化開始,逐漸開啟了DPU這個新處理器類型的創新。而
    的頭像 發表于 04-04 16:54 ?3781次閱讀
    探究<b class='flag-5'>I</b>/<b class='flag-5'>O</b>虛擬化及<b class='flag-5'>Virtio</b>接口技術(上)

    探究I/O虛擬化及Virtio接口技術(下)

    I/O虛擬化是SmartNIC/DPU/IPU中最核心的部分,AWS NITRO就是從I/O硬件虛擬化開始,逐漸開啟了DPU這個新處理器類型的創新。而
    的頭像 發表于 04-04 17:03 ?2649次閱讀
    探究<b class='flag-5'>I</b>/<b class='flag-5'>O</b>虛擬化及<b class='flag-5'>Virtio</b>接口技術(下)

    VirtIO Networking虛擬網絡設備實現架構

    VirtIO 由 Rusty Russell 開發,最初是為了支持自己開發的 lguest Hypervisor,其設計目標是在虛擬化環境下提供與物理設備相近的 I/O 功能和性能
    的頭像 發表于 05-08 10:48 ?1102次閱讀
    <b class='flag-5'>VirtIO</b> Networking虛擬網絡<b class='flag-5'>設備</b><b class='flag-5'>實現</b>架構

    RT-Thread設備驅動開發之二I/O設備框架

    設備驅動層是一組驅使硬件設備工作的程序,實現了訪問硬件設備的功能,它負責創建和注冊I/O
    的頭像 發表于 10-12 10:26 ?459次閱讀

    Linux I/O 接口的類型及處理流程

    ()、lseek() 等。 網絡 I/O 接口:用于網絡通信的接口,包括 socket()、connect()、bind()、listen()、accept() 等。 設備
    的頭像 發表于 11-08 16:43 ?882次閱讀
    Linux <b class='flag-5'>I</b>/<b class='flag-5'>O</b> 接口的類型及處理<b class='flag-5'>流程</b>