Hypervisor Functional Specification

January 9, 2009 – Draft Version 2.0

Abstract

This document is the top-level functional specification (TLFS) of the second-generation Microsoft hypervisor. It specifies the externally-visible behavior. The document assumes familiarity with the goals of the project and the high-level hypervisor architecture.

This specification is provided under the Microsoft Open Specification Promise. For further details on the Microsoft Open Specification Promise, please refer to: <http://www.microsoft.com/interop/osp/default.mspx>. Microsoft may have patents, patent applications, trademarks, copyrights, or other intellectual property rights covering subject matter in these materials. Except as expressly provided in the Microsoft Open Specification Promise, the furnishing of these materials does not give you any license to these patents, trademarks, copyrights, or other intellectual property.

**Disclaimer:** This is a preliminary document and may be changed substantially prior to final commercial release of the software described herein. This document is provided entirely "AS IS." To the extent permitted by law, MICROSOFT MAKES NO WARRANTY OF ANY KIND, DISCLAIMS ALL EXPRESS, IMPLIED AND STATUTORY WARRANTIES, AND ASSUMES NO LIABILITY TO YOU FOR ANY DAMAGES OF ANY TYPE IN CONNECTION WITH THESE MATERIALS OR ANY INTELLECTUAL PROPERTY IN THEM.

The information contained in this document represents the point-in-time view of Microsoft Corporation on the issues discussed as of the date of publication. Because Microsoft must respond to changing market conditions, it should not be interpreted to be a commitment on the part of Microsoft, and Microsoft cannot guarantee the accuracy of any information presented after the date of authoring.

Unless otherwise noted, the example companies, organizations, products, domain names, e-mail addresses, logos, people, places and events depicted herein are fictitious, and no association with any real company, organization, product, domain name, email address, logo, person, place or event is intended or should be inferred.

©2009 Microsoft Corporation. All rights reserved.

Microsoft, Windows, Windows NT, Windows Server, and Windows Vista are either registered trademarks or trademarks of Microsoft Corporation in the United States and/or other countries.

The names of actual companies and products mentioned herein may be the trademarks of their respective owners.

Contents

[1 Introduction 1](#_Toc230067740)

[1.1 Specification Style 1](#_Toc230067741)

[1.2 Interface Requirements and Goals 1](#_Toc230067742)

[1.3 Reserved Values 1](#_Toc230067743)

[2 Basic Data Types, Concepts and Notation 3](#_Toc230067744)

[2.1 Simple Scalar Types 3](#_Toc230067745)

[2.2 Hypercall Status Code 3](#_Toc230067746)

[2.3 Memory Types 3](#_Toc230067747)

[2.4 Structures, Enumerations and Bit Fields 4](#_Toc230067748)

[2.5 Endianness 4](#_Toc230067749)

[2.6 Pointer Naming Convention 4](#_Toc230067750)

[3 Feature and Interface Discovery 5](#_Toc230067751)

[3.1 Interface Mechanisms 5](#_Toc230067752)

[3.2 Hypervisor Discovery 5](#_Toc230067753)

[3.3 Standard Hypervisor CPUID Leafs 5](#_Toc230067754)

[3.4 Microsoft Hypervisor CPUID Leafs 6](#_Toc230067755)

[3.5 Versioning 8](#_Toc230067756)

[3.6 Reporting the Guest OS Identity 9](#_Toc230067757)

[4 Hypercall Interface 11](#_Toc230067758)

[4.1 Hypercall Overview 11](#_Toc230067759)

[4.2 Hypercall Classes 11](#_Toc230067760)

[4.3 Hypercall Continuation 11](#_Toc230067761)

[4.4 Hypercall Atomicity and Ordering 12](#_Toc230067762)

[4.5 Legal Hypercall Environments 12](#_Toc230067763)

[4.6 Alignment Requirements 12](#_Toc230067764)

[4.7 Hypercall Inputs 13](#_Toc230067765)

[4.7.1 Extended Fast Hypercalls 14](#_Toc230067766)

[4.8 Hypercall Outputs 15](#_Toc230067767)

[4.9 Hypercall Details 15](#_Toc230067768)

[4.10 Hypercall Restrictions 16](#_Toc230067769)

[4.11 Hypercall Status Codes 16](#_Toc230067770)

[4.11.1 Output Parameter Validity on Failed Hypercalls 17](#_Toc230067771)

[4.11.2 Ordering of Error Conditions 17](#_Toc230067772)

[4.11.3 Common Hypercall Status Codes 17](#_Toc230067773)

[4.12 Establishing the Hypercall Interface 17](#_Toc230067774)

[5 Partition Management 21](#_Toc230067775)

[5.1 Overview 21](#_Toc230067776)

[5.2 Partition Management Data Types 21](#_Toc230067777)

[5.2.1 Partition IDs 21](#_Toc230067778)

[5.2.2 Partition Properties 21](#_Toc230067779)

[5.2.3 Partition Privilege Flags 23](#_Toc230067780)

[5.2.4 Partition Creation Flags 25](#_Toc230067781)

[5.2.5 Partition State 25](#_Toc230067782)

[5.2.6 Partition Virtual TLB Page Count 26](#_Toc230067783)

[5.2.7 Partition Processor Vendor 26](#_Toc230067784)

[5.2.8 Partition Processor Features 27](#_Toc230067785)

[5.2.9 Partition Processor XSAVE Features 27](#_Toc230067786)

[5.2.10 Partition Cache Line Flush Size 28](#_Toc230067787)

[5.2.11 Partition Compatibility Mode 28](#_Toc230067788)

[5.3 Partition Creation 29](#_Toc230067789)

[5.4 Partition Destruction 29](#_Toc230067790)

[5.4.1 Partition Finalization 29](#_Toc230067791)

[5.4.2 Partition Deletion 30](#_Toc230067792)

[5.4.3 Partition Destruction 30](#_Toc230067793)

[5.5 Partition Enumeration 30](#_Toc230067794)

[5.6 Partition Management Interfaces 30](#_Toc230067795)

[5.6.1 HvCreatePartition 30](#_Toc230067796)

[5.6.2 HvInitializePartition 32](#_Toc230067797)

[5.6.3 HvFinalizePartition 33](#_Toc230067798)

[5.6.4 HvDeletePartition 34](#_Toc230067799)

[5.6.5 HvGetPartitionProperty 35](#_Toc230067800)

[5.6.6 HvSetPartitionProperty 36](#_Toc230067801)

[5.6.7 HvGetPartitionId 37](#_Toc230067802)

[5.6.8 HvGetNextChildPartition 38](#_Toc230067803)

[6 Physical Hardware Management 41](#_Toc230067804)

[6.1 Overview 41](#_Toc230067805)

[6.1.1 System Physical Address Space 41](#_Toc230067806)

[6.1.2 Logical Processors 42](#_Toc230067807)

[6.1.3 Dynamic Addition of Logical Processors 42](#_Toc230067808)

[6.1.4 Physical Nodes 42](#_Toc230067809)

[6.1.5 System Reset 42](#_Toc230067810)

[6.2 Hardware Information 42](#_Toc230067811)

[6.2.1 Boot-Time Hardware Properties 42](#_Toc230067812)

[6.2.2 Discovered Hardware Properties 43](#_Toc230067813)

[6.2.3 Root Partition Hardware Properties 43](#_Toc230067814)

[6.3 Hardware Management Data Types 43](#_Toc230067815)

[6.3.1 Logical Processors 43](#_Toc230067816)

[6.3.2 Power States 43](#_Toc230067817)

[6.3.2.1 Power State MSRs 43](#_Toc230067818)

[6.3.2.1.1 Power State Configuration Register 43](#_Toc230067819)

[6.3.2.1.2 Power State Trigger Register 45](#_Toc230067820)

[6.3.3 Logical Processor Run Time 45](#_Toc230067821)

[6.3.4 Global Run Time 45](#_Toc230067823)

[6.3.5 System Reset MSR 45](#_Toc230067824)

[6.4 Hardware Management Interfaces 46](#_Toc230067825)

[6.4.1 HvGetLogicalProcessorRunTime 46](#_Toc230067828)

[6.4.2 HvParkLogicalProcessors 47](#_Toc230067829)

[7 Resource Management 49](#_Toc230067830)

[7.1 Overview 49](#_Toc230067831)

[7.1.1 Memory Pools 49](#_Toc230067832)

[7.1.2 NUMA Proximity Domains 50](#_Toc230067833)

[7.2 Resource Management Data Types 50](#_Toc230067834)

[7.2.1 Proximity Domains 50](#_Toc230067835)

[7.3 Resource Management Interfaces 50](#_Toc230067836)

[7.3.1 HvDepositMemory 50](#_Toc230067837)

[7.3.2 HvWithdrawMemory 52](#_Toc230067838)

[7.3.3 HvGetMemoryBalance 53](#_Toc230067839)

[8 Guest Physical Address Spaces 55](#_Toc230067840)

[8.1 Overview 55](#_Toc230067841)

[8.1.1 GPA Space 55](#_Toc230067842)

[8.1.2 Page Access Rights 55](#_Toc230067843)

[8.1.3 GPA Overlay Pages 56](#_Toc230067844)

[8.2 GPA Data Types 57](#_Toc230067845)

[8.2.1 Map Page Flags 57](#_Toc230067846)

[8.3 GPA Interfaces 57](#_Toc230067847)

[8.3.1 HvMapGpaPages 57](#_Toc230067848)

[8.3.2 HvUnmapGpaPages 59](#_Toc230067849)

[8.3.3 HvMapSparseGpaPages 61](#_Toc230067850)

[9 Intercepts 65](#_Toc230067851)

[9.1 Overview 65](#_Toc230067852)

[9.1.1 Programmable Intercept Types 65](#_Toc230067853)

[9.1.2 Unsolicited Intercept Types 66](#_Toc230067854)

[9.2 Intercept Data Types 67](#_Toc230067855)

[9.2.1 Intercept Types 67](#_Toc230067856)

[9.2.2 Intercept Parameters 67](#_Toc230067857)

[9.2.3 Intercept Access Types 67](#_Toc230067858)

[9.2.4 Unsupported Feature Codes 68](#_Toc230067859)

[9.3 Intercept Interfaces 68](#_Toc230067860)

[9.3.1 HvInstallIntercept 68](#_Toc230067861)

[9.4 Intercept Messages and Message Formats 69](#_Toc230067862)

[10 Virtual Processor Management 71](#_Toc230067863)

[10.1 Overview 71](#_Toc230067864)

[10.1.1 Virtual Processor Indices 71](#_Toc230067865)

[10.1.2 Virtual Processor Registers 71](#_Toc230067866)

[10.1.3 Virtual Processor States 71](#_Toc230067867)

[10.1.4 Virtual Processor Idle Sleep State 71](#_Toc230067868)

[10.1.5 Virtual Boot Processor 72](#_Toc230067869)

[10.1.6 Virtual Processor APIC IDs 72](#_Toc230067870)

[10.2 Virtual Processor Data Types 72](#_Toc230067871)

[10.2.1 Virtual Processor Index 72](#_Toc230067872)

[10.2.2 Virtual Processor Register Names 72](#_Toc230067873)

[10.2.3 Virtual Processor Register Values 76](#_Toc230067874)

[10.3 Virtual Processor Register Formats 77](#_Toc230067875)

[10.3.1 Virtual Processor Suspend Registers 77](#_Toc230067876)

[10.3.2 Virtual Processor Run Time Register 77](#_Toc230067877)

[10.3.3 Virtual Processor Interrupt State Register 77](#_Toc230067878)

[10.3.4 Virtual Processor Pending Interruption Register 78](#_Toc230067879)

[10.3.5 Virtual Processor Floating-point and Vector Registers 79](#_Toc230067880)

[10.3.6 Virtual Processor Segment Registers 80](#_Toc230067881)

[10.3.7 Virtual Processor Table Registers 80](#_Toc230067882)

[10.4 Virtual Processor Interfaces 80](#_Toc230067883)

[10.4.1 HvCreateVp 80](#_Toc230067884)

[10.4.2 HvDeleteVp 82](#_Toc230067885)

[10.4.3 HvGetVpRegisters 83](#_Toc230067886)

[10.4.4 HvSetVpRegisters 84](#_Toc230067887)

[11 Virtual Processor Execution 87](#_Toc230067888)

[11.1 Processor Features and CPUID 87](#_Toc230067889)

[11.2 Family, Model and Stepping Reported by CPUID 87](#_Toc230067890)

[11.3 Platform ID Reported by MSR 87](#_Toc230067891)

[11.4 Real Mode 88](#_Toc230067892)

[11.5 MONITOR / MWAIT 88](#_Toc230067893)

[11.6 System Management Mode 88](#_Toc230067894)

[11.7 Time Stamp Counter 88](#_Toc230067895)

[11.8 Memory Accesses 88](#_Toc230067896)

[11.9 I/O Port Accesses 89](#_Toc230067897)

[11.10 MSR Accesses 90](#_Toc230067898)

[11.10.1 Modified MSR Behavior 91](#_Toc230067899)

[11.11 CPUID Execution 91](#_Toc230067900)

[11.12 Exceptions 91](#_Toc230067901)

[12 Virtual MMU and Caching 93](#_Toc230067902)

[12.1 Virtual MMU Overview 93](#_Toc230067903)

[12.1.1 Compatibility 93](#_Toc230067904)

[12.1.2 Legacy TLB Management Operations 94](#_Toc230067905)

[12.1.3 Virtual TLB Enhancements 94](#_Toc230067906)

[12.1.4 Restrictions on TLB Flushes 94](#_Toc230067907)

[12.2 Memory Cache Control Overview 95](#_Toc230067908)

[12.2.1 Cacheability Settings 95](#_Toc230067909)

[12.2.2 Mixing Cache Types between a Partition and the Hypervisor 95](#_Toc230067910)

[12.3 Virtual MMU Data Types 95](#_Toc230067911)

[12.3.1 Virtual Address Spaces 95](#_Toc230067912)

[12.3.2 Virtual Address Flush Flags 96](#_Toc230067913)

[12.3.3 Cache Types 96](#_Toc230067914)

[12.3.4 Virtual Address Translation Types 96](#_Toc230067915)

[12.3.5 Gpa Access Types 97](#_Toc230067916)

[12.4 Virtual MMU Interfaces 98](#_Toc230067917)

[12.4.1 HvSwitchVirtualAddressSpace 98](#_Toc230067918)

[12.4.2 HvFlushVirtualAddressSpace 99](#_Toc230067919)

[12.4.3 HvFlushVirtualAddressList 100](#_Toc230067920)

[12.4.4 HvTranslateVirtualAddress 102](#_Toc230067921)

[12.4.5 HvReadGpa 104](#_Toc230067922)

[12.4.6 HvWriteGpa 106](#_Toc230067923)

[13 Virtual Interrupt Control 109](#_Toc230067924)

[13.1 Overview 109](#_Toc230067925)

[13.2 Local APIC 109](#_Toc230067926)

[13.2.1 Local APIC Virtualization 109](#_Toc230067927)

[13.2.2 Local APIC Memory-mapped Accesses 110](#_Toc230067928)

[13.2.3 Local APIC MSR Accesses 110](#_Toc230067929)

[13.2.3.1 EOI Register 110](#_Toc230067930)

[13.2.3.2 ICR Register 111](#_Toc230067931)

[13.2.3.3 TPR Register 111](#_Toc230067932)

[13.3 Virtual Interrupts 111](#_Toc230067933)

[13.3.1 Virtual Interrupt Overview 111](#_Toc230067934)

[13.3.2 Virtual Interrupt Types 111](#_Toc230067935)

[13.3.3 Trigger Types 112](#_Toc230067936)

[13.3.4 EOI Intercepts 112](#_Toc230067937)

[13.3.4.1 APIC Assist Page Register 112](#_Toc230067938)

[13.4 Virtual Interrupt Data Types 114](#_Toc230067939)

[13.4.1 Interrupt Types 114](#_Toc230067940)

[13.4.2 Interrupt Control 114](#_Toc230067941)

[13.4.3 Interrupt Vectors 114](#_Toc230067942)

[13.5 Virtual Interrupt Interfaces 114](#_Toc230067943)

[13.5.1 HvAssertVirtualInterrupt 114](#_Toc230067944)

[13.5.2 HvClearVirtualInterrupt 116](#_Toc230067945)

[14 Inter-Partition Communication 119](#_Toc230067946)

[14.1 Overview 119](#_Toc230067947)

[14.2 SynIC Messages 119](#_Toc230067948)

[14.2.1 Message Buffers 119](#_Toc230067949)

[14.2.2 Message Buffer Queues 119](#_Toc230067950)

[14.2.3 Reliability and Sequencing of Guest Message Buffers 119](#_Toc230067951)

[14.2.4 Messages 120](#_Toc230067952)

[14.2.5 Recommended Message Handling 121](#_Toc230067953)

[14.2.6 Message Sources 121](#_Toc230067954)

[14.3 SynIC Event Flags 121](#_Toc230067955)

[14.3.1 Event Flag Delivery 121](#_Toc230067956)

[14.3.2 Recommended Event Flag Handling 122](#_Toc230067957)

[14.3.3 Event Flags versus Messages 122](#_Toc230067958)

[14.4 Ports and Connections 122](#_Toc230067959)

[14.5 Monitored Notifications 122](#_Toc230067960)

[14.5.1 Monitored Notification Trigger 123](#_Toc230067961)

[14.5.2 Monitored Notification Latency Hint 123](#_Toc230067962)

[14.5.3 Monitored Notification Parameters 123](#_Toc230067963)

[14.5.4 Monitored Notification Page 123](#_Toc230067964)

[14.6 SynIC MSRs 123](#_Toc230067965)

[14.6.1 SCONTROL Register 124](#_Toc230067966)

[14.6.2 SVERSION Register 124](#_Toc230067967)

[14.6.3 SIEFP Register 124](#_Toc230067968)

[14.6.4 SIMP Register 125](#_Toc230067969)

[14.6.5 SINTx Registers 125](#_Toc230067970)

[14.6.6 EOM Register 126](#_Toc230067971)

[14.7 SIM and SIEF Pages 126](#_Toc230067972)

[14.8 Inter-Partition Communication Data Types 127](#_Toc230067973)

[14.8.1 Synthetic Interrupt Sources 127](#_Toc230067974)

[14.8.2 SynIC Message Types 127](#_Toc230067975)

[14.8.3 SynIC Message Flags 128](#_Toc230067976)

[14.8.4 SynIC Message Format 128](#_Toc230067977)

[14.8.5 SynIC Event Flags 129](#_Toc230067978)

[14.8.6 Ports 129](#_Toc230067979)

[14.8.7 Port Properties 130](#_Toc230067980)

[14.8.8 Connections 131](#_Toc230067981)

[14.8.9 Connection Information 131](#_Toc230067982)

[14.8.9.1 Monitored Notification Trigger Group 131](#_Toc230067983)

[14.8.9.2 Monitored Notification Parameters 132](#_Toc230067984)

[14.8.10 Monitored Notification Page 132](#_Toc230067985)

[14.9 Inter-Partition Communication Interfaces 133](#_Toc230067986)

[14.9.1 HvCreatePort 133](#_Toc230067987)

[14.9.2 HvDeletePort 136](#_Toc230067988)

[14.9.3 HvConnectPort 137](#_Toc230067989)

[14.9.4 HvGetPortProperty 139](#_Toc230067990)

[14.9.5 HvSetPortProperty 140](#_Toc230067991)

[14.9.6 HvDisconnectPort 141](#_Toc230067992)

[14.9.7 HvPostMessage 142](#_Toc230067993)

[14.9.8 HvSignalEvent 143](#_Toc230067994)

[15 Timers 147](#_Toc230067995)

[15.1 Overview 147](#_Toc230067996)

[15.1.1 Timer Services 147](#_Toc230067997)

[15.1.2 Reference Counter 147](#_Toc230067998)

[15.1.3 Synthetic Timers 147](#_Toc230067999)

[15.1.4 Periodic Timers 147](#_Toc230068000)

[15.1.5 Periodic Timer Assist 148](#_Toc230068001)

[15.1.6 PM Timer Assist 149](#_Toc230068002)

[15.1.7 Ordering of Timer Expirations 150](#_Toc230068003)

[15.1.8 Timer Expiration Messages 150](#_Toc230068004)

[15.1.9 Partition Reference Time Enlightenment 150](#_Toc230068005)

[15.2 Reference Counter MSR 150](#_Toc230068006)

[15.2.1 Reference Counter MSR 151](#_Toc230068007)

[15.3 Synthetic Timer MSRs 151](#_Toc230068008)

[15.3.1 Synthetic Timer Configuration Register 151](#_Toc230068009)

[15.3.2 Synthetic Timer Count Register 152](#_Toc230068010)

[15.4 Partition Reference Time Enlightenment 152](#_Toc230068011)

[15.4.1 Reference Time Stamp Counter (TSC) Page MSR 153](#_Toc230068012)

[15.4.2 Format of the Reference TSC Page 153](#_Toc230068013)

[15.4.3 Partition Reference TSC Mechanism 153](#_Toc230068014)

[15.4.3.1 TscScale 154](#_Toc230068015)

[15.4.3.2 TscSequence 154](#_Toc230068016)

[15.4.3.3 Reference TSC During Save/Restore and Migration 154](#_Toc230068017)

[16 Message Formats 155](#_Toc230068018)

[16.1 Overview 155](#_Toc230068019)

[16.2 Message Data Types 155](#_Toc230068020)

[16.2.1 Message Header 155](#_Toc230068021)

[16.2.2 Intercept Message Header 155](#_Toc230068022)

[16.2.3 VP Execution State 156](#_Toc230068023)

[16.2.4 I/O Port Access Information 156](#_Toc230068024)

[16.2.5 Exception Information 157](#_Toc230068025)

[16.2.6 Memory Access Flags 157](#_Toc230068026)

[16.3 Memory Access Messages 157](#_Toc230068027)

[16.3.1 Unmapped GPA Message 159](#_Toc230068028)

[16.3.2 GPA Access Violation Message 159](#_Toc230068029)

[16.4 Timer Messages 159](#_Toc230068030)

[16.4.1 Timer Expiration Message 159](#_Toc230068031)

[16.5 Processor Event Messages 159](#_Toc230068032)

[16.5.1 CPUID Intercept Message 159](#_Toc230068033)

[16.5.2 MSR Intercept Message 160](#_Toc230068034)

[16.5.3 I/O Port Intercept Message 161](#_Toc230068035)

[16.5.4 Exception Intercept Message 162](#_Toc230068036)

[16.5.5 APIC EOI Message 164](#_Toc230068037)

[16.5.6 FERR Asserted Message 164](#_Toc230068038)

[16.5.7 Invalid VP Register Value Message 164](#_Toc230068039)

[16.5.8 Unrecoverable Exception Message 165](#_Toc230068040)

[16.5.9 Unsupported Feature Message 165](#_Toc230068041)

[16.5.10 Event Log Buffers Ready Message 166](#_Toc230068042)

[17 Partition Save and Restore 167](#_Toc230068043)

[17.1 Overview 167](#_Toc230068044)

[17.1.1 Saved State 167](#_Toc230068045)

[17.1.2 Summary State 167](#_Toc230068046)

[17.1.3 Saved State Compatibility and Versioning 167](#_Toc230068047)

[17.1.4 State That Is Not Saved by the Hypervisor 167](#_Toc230068048)

[17.1.5 State That Is Saved by the Hypervisor 168](#_Toc230068049)

[17.1.6 Partition State Streams 168](#_Toc230068050)

[17.1.7 Recommended Save Process 169](#_Toc230068051)

[17.1.8 Recommended Restore Process 169](#_Toc230068052)

[17.2 Partition Save and Restore Data Types 170](#_Toc230068053)

[17.2.1 Partition Save and Restore State 170](#_Toc230068054)

[17.2.2 Save and Restore Partition State Flags 172](#_Toc230068055)

[17.3 Partition Save and Restore Interfaces 172](#_Toc230068056)

[17.3.1 HvSavePartitionState 172](#_Toc230068057)

[17.3.2 HvRestorePartitionState 173](#_Toc230068058)

[18 Scheduler 177](#_Toc230068059)

[18.1 Scheduling Concepts 177](#_Toc230068060)

[18.2 Scheduling Policy Settings 177](#_Toc230068061)

[18.2.1 CPU Reserve 177](#_Toc230068062)

[18.2.2 CPU Cap 177](#_Toc230068063)

[18.2.3 CPU Weight 178](#_Toc230068064)

[18.3 Other Scheduling Considerations 178](#_Toc230068065)

[18.3.1 Hyperthreading 178](#_Toc230068066)

[18.3.2 NUMA and Affinity 178](#_Toc230068067)

[18.3.3 Guest Spinlocks 178](#_Toc230068068)

[18.4 Scheduler Data Types 179](#_Toc230068069)

[18.5 Scheduler Interfaces 179](#_Toc230068070)

[18.5.1 HvNotifyLongSpinWait 179](#_Toc230068071)

[19 Event Logging 180](#_Toc230068072)

[19.1 Overview 180](#_Toc230068073)

[19.1.1 Event Log Buffers 180](#_Toc230068074)

[19.1.2 Event Log Buffer Groups 180](#_Toc230068075)

[19.1.3 Local and Global Buffer Classes 180](#_Toc230068076)

[19.1.4 Event Log Buffer Indices 181](#_Toc230068077)

[19.1.5 Event Log Buffer States 181](#_Toc230068078)

[19.1.6 Accessing Event Log Buffers 183](#_Toc230068079)

[19.1.7 Preparing for Event Logging 183](#_Toc230068080)

[19.1.8 Enabling and Disabling Event Logging 184](#_Toc230068081)

[19.1.9 Logging Events into Buffers 184](#_Toc230068082)

[19.1.10 Event Log Buffers Ready Notification 184](#_Toc230068083)

[19.1.11 Completed Buffer Lists 185](#_Toc230068084)

[19.1.12 Buffer Access Restrictions 185](#_Toc230068085)

[19.1.13 Adding and Removing Buffers While Event Logging is Active 186](#_Toc230068086)

[19.1.14 Concluding Event Logging 186](#_Toc230068087)

[19.2 Event Logging Data Types 187](#_Toc230068088)

[19.2.1 Event Log Types 187](#_Toc230068089)

[19.2.2 Event Enable Flags 187](#_Toc230068090)

[19.2.3 Event Log Buffer State 187](#_Toc230068091)

[19.2.4 Event Log Buffer Index 188](#_Toc230068092)

[19.2.5 Event Log Buffer Header 188](#_Toc230068093)

[19.2.6 Event Log Entry Header 189](#_Toc230068094)

[19.3 Event Logging Interfaces 190](#_Toc230068095)

[19.3.1 HvInitializeEventLogBufferGroup 190](#_Toc230068096)

[19.3.2 HvFinalizeEventLogBufferGroup 191](#_Toc230068097)

[19.3.3 HvCreateEventLogBuffer 192](#_Toc230068098)

[19.3.4 HvDeleteEventLogBuffer 193](#_Toc230068099)

[19.3.5 HvMapEventLogBuffer 194](#_Toc230068100)

[19.3.6 HvUnmapEventLogBuffer 195](#_Toc230068101)

[19.3.7 HvSetEventLogGroupSources 195](#_Toc230068102)

[19.3.8 HvReleaseEventLogBuffer 196](#_Toc230068103)

[19.3.9 HvFlushEventLogBuffer 197](#_Toc230068104)

[20 Guest Debugging Support 199](#_Toc230068105)

[20.1 Overview 199](#_Toc230068106)

[20.1.1 Debug Sessions 199](#_Toc230068107)

[20.1.2 Transmitting Data on a Session 200](#_Toc230068108)

[20.1.3 Retrieving Data from a Session 200](#_Toc230068109)

[20.1.4 Purging Data from a Session 200](#_Toc230068110)

[20.1.5 Sensing Activity on the Physical Connection 200](#_Toc230068111)

[20.1.6 Hypercall Loops 200](#_Toc230068112)

[20.2 Debugging Data Types 201](#_Toc230068113)

[20.2.1 Debug Types 201](#_Toc230068114)

[20.3 Debugging Interfaces 201](#_Toc230068115)

[20.3.1 HvPostDebugData 201](#_Toc230068116)

[20.3.2 HvRetrieveDebugData 203](#_Toc230068117)

[20.3.3 HvResetDebugSession 204](#_Toc230068118)

[21 Statistics 207](#_Toc230068119)

[21.1 Overview 207](#_Toc230068120)

[21.1.1 Global Versus Local Statistics 207](#_Toc230068121)

[21.1.2 Statistics Page Mappings 207](#_Toc230068122)

[21.1.3 Format of Statistics Pages 207](#_Toc230068123)

[21.1.4 Statistics Group Compatibility 208](#_Toc230068124)

[21.2 Statistics Data Types 209](#_Toc230068125)

[21.2.1 Specifying Statistics Objects 209](#_Toc230068126)

[21.3 Statistics Interfaces 210](#_Toc230068127)

[21.3.1 HvMapStatsPage 210](#_Toc230068128)

[21.3.2 HvUnmapStatsPage 212](#_Toc230068129)

[21.3.3 Statistics Page Mapping MSRs 213](#_Toc230068130)

[22 Booting 215](#_Toc230068131)

[22.1 Overview 215](#_Toc230068132)

[22.2 Pre-boot Requirements 215](#_Toc230068133)

[22.3 Post-boot Conditions 215](#_Toc230068134)

[22.4 Root Partition 216](#_Toc230068135)

[23 System Properties 217](#_Toc230068136)

[23.1 Overview 217](#_Toc230068137)

[23.2 System Property Data Types 217](#_Toc230068138)

[23.3 Performance Counter Configuration 217](#_Toc230068139)

[23.4 System Property Interfaces 218](#_Toc230068140)

[23.4.1 HvSetSystemProperty 218](#_Toc230068141)

[23.4.2 HvGetSystemProperty 218](#_Toc230068142)

[24 Appendix A: Interface Guidelines 220](#_Toc230068143)

[25 Appendix B: Hypercall Code Reference 221](#_Toc230068144)

[26 Appendix C: Hypercall Status Code Reference 223](#_Toc230068145)

[27 Appendix D: Architectural CPUID 227](#_Toc230068146)

[28 Appendix E: Boot-time CPUID Feature Requirements 243](#_Toc230068147)

[29 Appendix F: Architectural MSRs 249](#_Toc230068148)

[30 Appendix G: Vendor-Specific MSRs 251](#_Toc230068149)

[31 Appendix H: Hypervisor Synthetic MSRs 252](#_Toc230068150)

[32 Appendix I: Event Log Entries 257](#_Toc230068151)

[33 Appendix J: Statistics Counter Definitions 261](#_Toc230068152)

# Introduction

This document is the top-level functional specification (TLFS) of the second-generation Microsoft hypervisor. It specifies the externally-visible behavior. The document assumes familiarity with the goals of the project and the high-level hypervisor architecture.

This document is intended to be sufficiently complete and precise to allow a developer to implement a compatible hypervisor from scratch.

## Specification Style

This specification is informal; that is, the interfaces are not specified in a formal language. Nevertheless, it is a goal to be precise. It is also a goal to specify which behaviors are architectural and which are implementation-specific. Callers should not rely on behaviors that fall into the latter category because they may change in future implementations.

Segments of code and algorithms are presented with a grey background.

## Interface Requirements and Goals

The hypervisor interfaces are designed with the following requirements and goals in mind:

The amount of time spent in the hypervisor in response to a hypercall should be bounded to 50μs. This restriction requires that operations acting on large lists or ranges must be piece-wise interruptible.

To the extent possible, the interfaces should be independent of hardware architecture. Although the first-generation hypervisor will run only on x64 processors, it is highly likely that future versions will be ported to other architectures. It may not be possible to implement certain aspects of an architecture in a transparent manner. The x64 implementation exposes architecture-specifics with the handling of the CPUID instruction (the specifics are detailed in this specification).

The interfaces should provide significant “future proofing” and therefore should not impose short-sighted restrictions on memory size, address space size, processor count, and so on.

The interfaces should allow a conforming hypervisor implementation to correctly virtualize all aspects of the underlying system architecture. In some cases, the architecture may allow implementations the freedom not to virtualize all aspects correctly, but it should not constrain an implementation in a way that it cannot virtualize all aspects correctly.

## Reserved Values

This specification documents some fields as “reserved.” These fields may be given specific meaning in future versions of the hypervisor architecture. For maximum forward compatibility, clients of the hypervisor interface should follow the guidance provided within this document. In general, two forms of guidance are provided.

Preserve value (documented as *RsvdP* in diagrams and ReservedP in code segments): For maximum forward compatibility, clients should preserve the value within this field. This is typically done by reading the current value, modifying the values of the non-reserved fields, and writing the value back.

Zero value (documented as *RsvdZ* in diagrams and ReservedZ in code segments): For maximum forward compatibility, clients should zero the value within this field.

Reserved fields within read-only structures are simply documented as *Rsvd* in diagrams and simply as Reserved in code segments. For maximum forward compatibility, the values within these fields should be ignored. Clients should not assume these values will always be zero.

# Basic Data Types, Concepts and Notation

## Simple Scalar Types

Hypervisor data types are built up from simple scalar types UINT8, UINT16, UINT32, UINT64 and UINT128. Each of these represents a simple unsigned integer scalar with the specified bit count. Several corresponding signed integer scalars are also defined: INT8, INT16, INT32, and INT64.

The hypervisor uses neither floating point instructions nor floating point types.

## Hypercall Status Code

Every hypercall returns a 16-bit status code of type HV\_STATUS.

typedef UINT16 HV\_STATUS;

## Memory Types

The hypervisor architecture defines three independent address spaces:

*System physical addresses* (SPAs) define the physical address space of the underlying hardware as seen by the CPUs. There is only one system physical address space for the entire machine.

*Guest physical addresses* (GPAs) define the guest’s view of physical memory. GPAs can be mapped to underlying SPAs. There is one guest physical address space per partition.

*Guest virtual addresses* (GVAs) are used within the guest when it enables address translation and provides a valid guest page table.

All three of these address spaces are up to 264 bytes in size. The following types are thus defined:

typedef UINT64 HV\_SPA;

typedef UINT64 HV\_GPA;

typedef UINT64 HV\_GVA;

Many hypervisor interfaces act on *pages* of memory rather than single bytes. The minimum page size is architecture-dependent. For x64, it is defined as 4 K.

#define X64\_PAGE\_SIZE 0x1000

#define HV\_X64\_MAX\_PAGE\_NUMBER (MAXUINT64/X64\_PAGE\_SIZE)

#define HV\_PAGE\_SIZE X64\_PAGE\_SIZE

#define HV\_LARGE\_PAGE\_SIZE X64\_LARGE\_PAGE\_SIZE

#define HV\_PAGE\_MASK (HV\_PAGE\_SIZE - 1)

typedef UINT64 HV\_SPA\_PAGE\_NUMBER;

typedef UINT64 HV\_GPA\_PAGE\_NUMBER;

typedef UINT64 HV\_GVA\_PAGE\_NUMBER;

typedef HV\_GPA\_PAGE\_NUMBER \*PHV\_GPA\_PAGE\_NUMBER;

To convert an HV\_SPA to an HV\_SPA\_PAGE\_NUMBER, simply divide by HV\_PAGE\_SIZE.

## Structures, Enumerations and Bit Fields

Many data structures and constant values defined later in this specification are defined in terms of C-style enumerations and structures. The C language purposely avoids defining certain implementation details. However, this document assumes the following:

All enumerations declared with the “enum” keyword define 32-bit signed integer values.

All structures are padded in such a way that fields are aligned naturally (that is, an 8-byte field is aligned to an offset of 8 bytes and so on).

All bit fields are packed from low-order to high-order bits with no padding.

## Endianness

The hypervisor interface is designed to be endian-neutral (that is, it should be possible to port the hypervisor to a big-endian or little-endian system), but some of the data structures defined later in this specification assume little-endian layout. Such data structures will need to be amended if and when a big-endian port is attempted.

## Pointer Naming Convention

The document uses a naming convention for pointer types. In particular, a “P” prepended to a defined type indicates a pointer to that type. A “PC” prepended to a defined type indicates a pointer to a constant value of that type.

# 

# Feature and Interface Discovery

## Interface Mechanisms

Guest software interacts with the hypervisor through a variety of mechanisms. Many of these mirror the traditional mechanisms used by software to interact with the underlying processor. As such, these mechanisms are architecture-specific. On the x64 architecture, the following mechanisms are used:

CPUID instruction: Used for static feature and version information.

MSRs (model-specific registers): Used for status and control values.

Memory-mapped registers: Used for status and control values.

Processor interrupts: Used for asynchronous events, notifications and messages.

In addition to these architecture-specific interfaces, the hypervisor provides a simple procedural interface implemented with *hypercalls*. For information about the hypercall mechanism, see chapter .

## Hypervisor Discovery

Before using any hypervisor interfaces, software should first determine whether it’s running within a virtualized environment. On x64 platforms that conform to this specification, this is done by executing the CPUID instruction with an input (EAX) value of 1. Upon execution, code should check bit 31 of register ECX (the “hypervisor present bit”). If this bit is set, a hypervisor is present. In a non-virtualized environment, the bit will be clear.

If the “hypervisor present bit” is set, additional CPUID leafs can be queried for more information about the conformant hypervisor and its capabilities. Two such leafs are guaranteed to be available: 0x40000000 and 0x40000001. Subsequently-numbered leafs may also be available.

## Standard Hypervisor CPUID Leafs

When the leaf at 0x40000000 is queried, the hypervisor will return information that provides the maximum hypervisor CPUID leaf number and a vendor ID signature.

|  |  |  |
| --- | --- | --- |
| Leaf | Information Provided | |
| 0x40000000 | Hypervisor CPUID leaf range and vendor ID signature. | |
| EAX | The maximum input value for hypervisor CPUID information. |
| EBX | Hypervisor Vendor ID Signature |
| ECX | Hypervisor Vendor ID Signature |
| EDX | Hypervisor Vendor ID Signature |

If the leaf at 0x40000001 is queried, it will return a value that identifies the hypervisor interface.

|  |  |  |
| --- | --- | --- |
| Leaf | Information Provided | |
| 0x40000001 | Hypervisor vendor-neutral interface identification. This determines the semantics of the leaves from 0x40000002 through 0x400000FF | |
| EAX | Hypervisor Interface Signature |
| EBX | Reserved |
| ECX | Reserved |
| EDX | Reserved |

These two leafs allow the guest to query the hypervisor vendor ID and interface independently. The vendor ID is provided only for informational and diagnostic purposes. It is recommended that software only base compatibility decisions on the interface signature reported through leaf 0x40000001.

## Microsoft Hypervisor CPUID Leafs

On hypervisors conforming to the Microsoft hypervisor CPUID interface, the 0x40000000 and 0x40000001 leaf registers will have the following values:

| Leaf | Information Provided | |
| --- | --- | --- |
| 0x40000000 | Hypervisor CPUID leaf range and vendor ID signature. | |
| EAX | The maximum input value for hypervisor CPUID information. On Microsoft hypervisors, this will be at least 0x40000005. The vendor ID signature should be used only for reporting and diagnostic purposes. |
| EBX | 0x7263694D—“Micr” |
| ECX | 0x666F736F—“osof” |
| EDX | 0x76482074—“t Hv” |
| 0x40000001 | Hypervisor vendor-neutral interface identification. This determines the semantics of the leaves from 0x40000002 through 0x400000FF. | |
| EAX | 0x31237648—“Hv#1” |
| EBX | Reserved |
| ECX | Reserved |
| EDX | Reserved |

Hypervisors conforming to the “Hv#1” interface also provide at least the following leaves:

| Leaf | Information Provided | |
| --- | --- | --- |
| 0x40000002 | Hypervisor system identity. This value will be zero until the OS identity MSR is set (see section ); after that, they have the following definitions: | |
| EAX | Build Number |
| EBX | Bits 31-16: Major Version  Bits 15-0: Minor Version |
| ECX | Service Pack |
| EDX | Bits 31-24: Service Branch  Bits 23-0: Service Number |
| 0x40000003 | Feature identification. EAX indicates which features are available to the partition based upon the current partition privileges. | |
| EAX | Bit 0:VP Runtime (HV\_X64\_MSR\_VP\_RUNTIME) available  Bit 1: Partition Reference Counter (HV\_X64\_MSR\_TIME\_REF\_COUNT) available  Bit 2: Basic SynIC MSRs (HV\_X64\_MSR\_SCONTROL through HV\_X64\_MSR\_EOM and HV\_X64\_MSR\_SINT0 through HV\_X64\_MSR\_SINT15) available  Bit 3: Synthetic Timer MSRs (HV\_X64\_MSR\_STIMER0\_CONFIG through HV\_X64\_MSR\_STIMER3\_COUNT) available  Bit 4: APIC access MSRs (HV\_X64\_MSR\_EOI, HV\_X64\_MSR\_ICR and HV\_X64\_MSR\_TPR) are available  Bit 5: Hypercall MSRs (HV\_X64\_MSR\_GUEST\_OS\_ID and HV\_X64\_MSR\_HYPERCALL) available  Bit 6: Access virtual processor index MSR (HV\_X64\_MSR\_VP\_INDEX) available  Bit 7: Virtual system reset MSR (HV\_X64\_MSR\_RESET) is available.  Bit 8:Access statistics pages MSRs (HV\_X64\_MSR\_STATS\_PARTITION\_RETAIL\_PAGE, HV\_X64\_MSR\_STATS\_PARTITION\_INTERNAL\_PAGE, HV\_X64\_MSR\_STATS\_VP\_RETAIL\_PAGE, HV\_X64\_MSR\_STATS\_VP\_INTERNAL\_PAGE) available.  Bits 9-31: Reserved |
| Feature identification: EBX indicates which flags were specified at partition creation. The format is the same as the partition creation flag structure defined in section 5.2.4. | |
| EBX | Bit 0: CreatePartitions  Bit 1: AccessPartitionId  Bit 2: AccessMemoryPool  Bit 3: AdjustMessageBuffers  Bit 4: PostMessages  Bit 5: SignalEvents  Bit 6: CreatePort  Bit 7: ConnectPort  Bit 8: AccessStats  Bit 9-10: RsvdZ  Bit 11: Debugging  Bit 12: CpuPowerManagement  Bit 13: ConfigureProfiler  Bit 14-31: Reserved |
| Feature identification. ECX contains power management related information. | |
| ECX | Bit 0-3: Maximum Processor Power State. 0 is C0, 1 is C1, 2 is C2, 3 is C3.  Bits 8-31: Reserved |
| Feature identification. EDX indicates which miscellaneous features are available to the partition. | |
| EDX | Bit 0: The MWAIT instruction is available (per section )  Bit 1: Guest debugging support is available  Bit 2: Performance Monitor support is available  Bit 3: Support for physical CPU dynamic partitioning events is available  Bit 4: Support for passing hypercall input parameter block via XMM registers is available  Bit 5: Support for a virtual guest idle state is available Bits 6-31: Reserved |
| 0x40000004 | Implementation recommendations. Indicates which behaviors the hypervisor recommends the OS implement for optimal performance. | |
| EAX | Bit 0: Recommend using hypercall for address space switches rather than MOV to CR3 instruction  Bit 1: Recommend using hypercall for local TLB flushes rather than INVLPG or MOV to CR3 instructions  Bit 2: Recommend using hypercall for remote TLB flushes rather than inter-processor interrupts  Bit 3: Recommend using MSRs for accessing APIC registers EOI, ICR and TPR rather than their memory-mapped counterparts.  Bit 4: Recommend using the hypervisor-provided MSR to initiate a system RESET.  Bit 5: Recommend using relaxed timing for this partition. If used, the VM should disable any watchdog timeouts that rely on the timely delivery of external interrupts.  Bit 6-31: Reserved |
| EBX | Recommended number of attempts to retry a spinlock failure before notifying the hypervisor about the failures.  0xFFFFFFFF indicates never to retry. |
| ECX | Reserved |
| EDX | Reserved |
| 0x40000005 | Implementation limits. If any value is zero, the hypervisor does not expose the corresponding information; otherwise, they have these meanings: | |
| EAX | The maximum number of virtual processors supported. |
| EBX | The maximum number of logical processors supported. |
| ECX | Reserved |
| EDX | Reserved |
| 0x40000006 | Implementation hardware features. Indicates which hardware-specific features have been detected and are currently in use by the hypervisor. | |
| EAX | Bit 0: Support for APIC overlay assist is detected and in use.  Bit 1: Support for MSR bitmaps is detected and in use.  Bit 2: Support for architectural performance counters is detected and in use.  Bits 3-31: Reserved for future Intel-specific features. |
| EBX | Reserved |
| ECX | Reserved |
| EDX | Reserved for future AMD-specific features. |

## Versioning

The hypervisor version information is encoded in leaf 0x40000002. Two version numbers are provided: the main version and the service version.

The main version includes a major and minor version number and a build number. These correspond to Microsoft Windows® release numbers. The service version describes changes made to the main version.

For maximum forward compatibility, clients should use the hypervisor version information with extreme care. When checking main versions, clients should use greater-than-or-equal tests, not equality tests. The following pseudo-code demonstrates the method that should be employed when comparing entire version numbers (consisting of both the main and service versions):

if <your-main-version> greater than <hypervisor-main-version>

{

your version is compatible

}

else if <your-main-version> equal to <hypervisor-main-version>

and

<your-service-version> greater than or equal to

<hypervisor-service-version>

{

your version is compatible

}

else

{

your version is NOT compatible

}

Clients are strongly encouraged to check for hypervisor features by using CPUID leaves 0x40000003 through 0x40000005 rather than by comparing against version ranges.

## Reporting the Guest OS Identity

The guest OS running within the partition must identify itself to the hypervisor by writing its signature and version to an MSR (HV\_X64\_MSR\_GUEST\_OS\_ID). This MSR is partition-wide and is shared among all virtual processors (virtual processors are described in chapter 10).

This register’s value is initially zero. A non-zero value must be written to it before the hypercall code page can be enabled (see section ). If this register is subsequently zeroed, the hypercall code page will be disabled.

The value written to this MSR is ignored by the hypervisor but may be used in future hypervisor implementations to maintain compatibility with existing guest OSs.

#define HV\_X64\_MSR\_GUEST\_OS\_ID 0x40000000

The following is the recommended encoding for this MSR. Some fields may not apply for some guest OSs.

|  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- |
| 63:48 | 47:40 | 39:32 | 31:24 | 23:16 | 15:0 |
| Vendor ID | OS ID | Major Version | Minor Version | Service Version | Build Number |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:48 | Vendor ID  Indicates theguest OS vendor. A value of 0 is reserved. A value of 1 indicates Microsoft. | Read/write |
| 47:40 | OS ID  Indicates the OS variant. Encoding is unique to the vendor. Microsoft operating systems are encoded as follows:  0=Undefined, 1=MS-DOS®, 2=Windows® 3.x, 3=Windows® 9x, 4=Windows® NT (and derivatives), 5=Windows® CE. | Read/write |
| 39:32 | Major Version  Indicates the major version of the OS. | Read/write |
| 31:24 | Minor Version  Indicates the minor version of the OS. | Read/write |
| 23:16 | Service Version  Indicates the service version (for example, "service pack" number). | Read/write |
| 15:0 | Build Number  Indicates the build number of the OS. | Read/write |

# 

# Hypercall Interface

## Hypercall Overview

The hypervisor provides a calling mechanism for guests. Such calls are referred to as *hypercalls*. Each hypercall defines a set of input and/or output parameters. These parameters are specified in terms of a memory-based data structure. All elements of the input and output data structures are padded to natural boundaries up to 8 bytes (that is, two-byte elements must be on two-byte boundaries and so on).

A second hypercall calling convention can optionally be used for a subset of hypercalls – in particular, those that have two or fewer input parameters and no output parameters. When using this calling convention, the input parameters are passed in registers.

A third hypercall calling convention can optionally be used for a subset of hypercalls where the input parameter block is up to 112 bytes. When using this calling convention, the input parameters are passed in registers, including the volatile XMM registers.

Input and output data structures must both be placed in memory on an 8-byte boundary and padded to a multiple of 8 bytes in size. The values within the padding regions are ignored by the hypervisor.

For output, the hypervisor is allowed to (but not guaranteed to) overwrite padding regions. If it overwrites padding regions, it will write zeros.

## Hypercall Classes

There are two classes of hypercalls: *simple* and *rep* (short for “repeat”). A *simple hypercall* performs a single operation and has a fixed-size set of input and output parameters. A *rep hypercall* acts like a series of simple hypercalls. In addition to a fixed-size set of input and output parameters, rep hypercalls involve a list of fixed-size input and/or output elements.

When a caller initially invokes a rep hypercall, it specifies a *rep count* that indicates the number of elements in the input and/or output parameter list. Callers also specify a *rep start index* that indicates the next input and/or output element that should be consumed. The hypervisor processes rep parameters in list order – that is, by increasing element index.

For subsequent invocations of the rep hypercall, the *rep start index* indicates how many elements have been completed – and, in conjunction with the *rep count* value – how many elements are left. For example, if a caller specifies a rep count of 25, and only 20 iterations are completed within the 50μs window (described in section 4.3), the hypercall returns control back to the calling virtual processor after updating the *rep start index* to 20. (See section 4.7 for more information about the *rep start index*.) When the hypercall is re-executed, the hypervisor will resume at element 20 and complete the remaining 5 elements.

If an error is encountered when processing an element, an appropriate status code is provided along with a *reps completed count*, indicating the number of elements that were successfully processed before the error was encountered. Assuming the specified hypercall control word is valid (see the following) and the input / output parameter lists are accessible, the hypervisor is guaranteed to attempt at least one rep, but it is not required to process the entire list before returning control back to the caller.

## Hypercall Continuation

A hypercall can be thought of as a complex instruction that takes many cycles. The hypervisor attempts to limit hypercall execution to 50μs or less before returning control to the virtual processor that invoked the hypercall. Some hypercall operations are sufficiently complex that a 50μs guarantee is difficult to make. The hypervisor therefore relies on a *hypercall* *continuation* mechanism for some hypercalls – including all rep hypercall forms.

The hypercall continuation mechanism is mostly transparent to the caller. If a hypercall is not able to complete within the prescribed time limit, control is returned back to the caller, but the instruction pointer is not advanced past the instruction that invoked the hypercall. This allows pending interrupts to be handled and other virtual processors to be scheduled. When the original calling thread resumes execution, it will re-execute the hypercall instruction and make forward progress toward completing the operation.

Most simple hypercalls are guaranteed to complete within the prescribed time limit. However, a small number of simple hypercalls might require more time. These hypercalls use hypercall continuation in a similar manner to rep hypercalls. In such cases, the operation involves two or more internal states. The first invocation places the object (for example, the partition or virtual processor) into one state, and after repeated invocations, the state finally transitions to a terminal state. For each hypercall that follows this pattern, the visible side effects of intermediate internal states is described.

## Hypercall Atomicity and Ordering

Except where noted, the action performed by a hypercall is atomic both with respect to all other guest operations (for example, instructions executed within a guest) and all other hypercalls being executed on the system. A simple hypercall performs a single atomic action; a rep hypercall performs multiple, independent atomic actions.

Simple hypercalls that use hypercall continuation may involve multiple internal states that are externally visible. Such calls comprise multiple atomic operations.

Each hypercall action may read input parameters and/or write results. The inputs to each action can be read at any granularity and at any time after the hypercall is made and before the action is executed. The results (that is, the output parameters) associated with each action may be written at any granularity and at any time after the action is executed and before the hypercall returns.

The guest must avoid the examination and/or manipulation of any input or output parameters related to an executing hypercall. While a virtual processor executing a hypercall will be incapable of doing so (as its guest execution is suspended until the hypercall returns), there is nothing to prevent other virtual processors from doing so. Guests behaving in this manner may crash or cause corruption within their partition.

## Legal Hypercall Environments

Hypercalls can be invoked only from the most privileged guest processor mode. In the case of x64, this means protected mode with a current privilege level (CPL) of zero. Although real-mode code runs with an effective CPL of zero, hypercalls are *not* allowed in real mode. An attempt to invoke a hypercall within an illegal processor mode will generate a #UD (undefined operation) exception.

All hypercalls should be invoked through the architecturally-defined hypercall interface. (See the following sections for instructions on discovering and establishing this interface.) An attempt to invoke a hypercall by any other means (for example, copying the code from the hypercall code page to an alternate location and executing it from there) *might* result in an undefined operation (#UD) exception. The hypervisor is not guaranteed to deliver this exception.

## Alignment Requirements

Callers must specify the 64-bit guest physical address (GPA) of the input and/or output parameters. If the hypercall involves no input or output parameters, the hypervisor ignores the corresponding GPA pointer.

The input and output parameter lists cannot overlap or cross page boundaries. Hypercall input and output pages are expected to be GPA pages and not “overlay” pages (for a discussion of overlay pages, see section ). If the virtual processor writes the input parameters to an overlay page and specifies a GPA within this page, hypervisor access to the input parameter list is undefined.

The hypervisor will validate that the calling partition can read from the input page before executing the requested hypercall. This validation consists of two checks: the specified GPA is mapped and the GPA is marked *readable*. If either of these tests fails, the hypervisor generates a memory intercept message. For more information on memory intercepts, see section .

For hypercalls that have output parameters, the hypervisor will validate that the partition can be write to the output page. This validation consists of two checks: the specified GPA is mapped and the GPA is marked *writable*. If either of these tests fails, the hypervisor attempts to generate a memory intercept message. If the validation succeeds, the hypervisor “locks” the output GPA for the duration of the operation. Any attempt to remap or unmap this GPA will be deferred until after the hypercall is complete.

## Hypercall Inputs

Callers specify a hypercall by a 64-bit value called a *hypercall input value*. It is formatted as follows:

Hypercall input value:

|  |  |  |
| --- | --- | --- |
| Call code | 16 bits | Specifies which hypercall is requested |
| Fast | 1 bit | Specifies whether the hypercall uses the register-based calling convention.  0: Use the memory-based calling convention  1: Use the register-based calling convention  If this calling convention is used, the Rep fields must be zero. |
| RsvdZ | 15 bits | Must be zero |
| Rep Count | 12 bits | Total number of reps (for rep call, must be zero otherwise) |
| RsvdZ | 4 bits | Must be zero |
| Rep Start Index | 12 bits | Starting index (for rep call, must be zero otherwise) |
| RsvdZ | 4 bits | Must be zero |

|  |  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- | --- |
| 63:60 | 59:48 | 47:44 | 43:32 | 31:17 | 16 | 15:0 |
| RsvdZ  (4 bits) | Rep start index  (12 bits) | RsvdZ  (4 bits) | Rep count  (12 bits) | RsvdZ  (15 bits) | Fast  (1 bit) | Call Code  (16 bits) |

For rep hypercalls, the *rep count* field indicates the total number of reps. The *rep start index* indicates the particular repetition relative to the start of the list (zero indicates that the first element in the list is to be processed). Therefore, the *rep count* value must always be greater than the *rep start index*.

Register mapping for hypercall inputs when the Fast flag is zero:

|  |  |  |
| --- | --- | --- |
| x64 | x86 | Contents |
| RCX | EDX:EAX | Hypercall Input Value |
| RDX | EBX:ECX | Input Parameters GPA |
| R8 | EDI:ESI | Output Parameters GPA |

The hypercall input value is passed in registers along with a GPA that points to the input and output parameters. The register mappings depend on whether the caller is running in 32-bit (x86) or 64-bit (x64) mode. The hypervisor determines the caller’s mode based on the value of EFER.LMA and CS.L. If both of these flags are set, the caller is assumed to be a 64-bit caller.

Some register values are considered volatile and can be modified by a hypercall. This list includes the condition code registers SF, ZF, OF, CF, AF, and PF. It also includes EAX, ECX, and EDX for x86 callers and RCX, RDX, R8, R9, R10, R11, and XMM0 through XMM5 for x64 callers.

RAX (on x64) and EDX:EAX (on x86) are always overwritten with the hypercall result value (discussed in section 4.8).

Register mapping for hypercall inputs when the Fast flag is one:

|  |  |  |
| --- | --- | --- |
| x64 | x86 | Contents |
| RCX | EDX:EAX | Hypercall Input Value |
| RDX | EBX:ECX | Input Parameter |
| R8 | EDI:ESI | Input Parameter |

The hypercall input value is passed in registers along with the input parameters. The register mappings depend on whether the caller is running in 32-bit (x86) or 64-bit (x64) mode. The hypervisor determines the caller’s mode based on the value of EFER.LMA and CS.L. If both of these flags are set, the caller is assumed to be a 64-bit caller.

Some register values are considered volatile and can be modified by a hypercall. This list includes the condition code registers SF, ZF, OF, CF, AF, and PF. It also includes EAX, ECX, EDX, EDI and ESI for x86 callers and RCX, RDX, R8, R9, R10, R11, and XMM0 through XMM5 for x64 callers.

RAX, RDX and R8 (on x64) and EDX:EAX, EBX:ECX and EDI:ESI (on x86) are always overwritten with the hypercall result value and the output parameters, if any (discussed in section 4.8).

### Extended Fast Hypercalls

The hypervisor supports the use of *extended fast hypercalls*, which allows some hypercalls to take advantage of the improved performance of the fast hypercall interface even though they require more than two input parameters. The extended fast hypercall interface uses six volatile XMM registers to allow the caller to pass an input parameter block up to 112 bytes in size. Availability of the extended fast hypercall interface is indicated by the presence of a flag returned by the CPUID instruction for a hypervisor leaf (see section 3.4). Any attempt to use this interface when the hypervisor does not indicate availability will result in a #UD fault.

Register mapping for extended fast hypercall inputs when the Fast flag is one:

|  |  |  |
| --- | --- | --- |
| x64 | x86 | Contents |
| RCX | EDX:EAX | Hypercall Input Value |
| RDX | EBX:ECX | Input Parameter Block (Byte 0-7) |
| R8 | EDI:ESI | Input Parameter Block (Byte 8-15) |
| XMM0 | XMM0 | Input Parameter Block (Byte 16-31) |
| XMM1 | XMM1 | Input Parameter Block (Byte 32-47) |
| XMM2 | XMM2 | Input Parameter Block (Byte 48-63) |
| XMM3 | XMM3 | Input Parameter Block (Byte 64-79) |
| XMM4 | XMM4 | Input Parameter Block (Byte 80-95) |
| XMM5 | XMM5 | Input Parameter Block (Byte 96-111) |

The hypercall input value is passed in registers along with the input parameters. The register mappings depend on whether the caller is running in 32-bit (x86) or 64-bit (x64) mode. The hypervisor determines the caller’s mode based on the value of EFER.LMA and CS.L. If both of these flags are set, the caller is assumed to be a 64-bit caller. If the input parameter block is smaller than 112 bytes, any extra bytes in the registers are ignored.

Some register values are considered volatile and can be modified by a hypercall. This list includes the condition code registers SF, ZF, OF, CF, AF, and PF. It also includes EAX, ECX, EDX, EDI and ESI for x86 callers and RCX, RDX, R8, R9, R10, R11, and XMM0 through XMM5 for x64 callers.

RAX, RDX and R8 (on x64) and EDX:EAX, EBX:ECX and EDI:ESI (on x86) are always overwritten with the hypercall result value and the output parameters, if any (discussed in section 4.8).

## Hypercall Outputs

All hypercalls return a 64-bit value called a *hypercall result value*. It is formatted as follows:

**Hypercall output value:**   
Result: 16 bits HV\_STATUS code indicating success or failure  
Rsvd: 16 bits Callers should ignore the value in these bits  
Reps completed: 12 bits Number of reps successfully completed  
Rsvd: 20 bits Callers should ignore the value in these bits

|  |  |  |  |
| --- | --- | --- | --- |
| 63:40 | 43:32 | 31:16 | 15:0 |
| Rsvd  (20 bits) | Reps complete  (12 bits) | Rsvd  (16 bits) | Result  (16 bits) |

For rep hypercalls, the *reps complete* field is the total number of reps complete and not relative to the *rep start index*. For example, if the caller specified a *rep start index* of 5, and a *rep count* of 10, the *reps complete* field would indicate 10 upon successful completion.

The hypercall result value is passed back in registers. The register mapping depends on whether the caller is running in 32-bit (x86) or 64-bit (x64) mode (see above).

The register mapping for hypercall outputs is as follows:

|  |  |  |
| --- | --- | --- |
| x64 | x86 | Content |
| RAX | EDX:EAX | Hypercall Result Value |

## Hypercall Details

Each hypercall in this document is described in two ways: a *wrapper interface* and a *native interface*. The wrapper interface is the recommended high-level (C-style) calling convention typically provided by a “wrapper library” that runs within the guest (for example, WinHv.sys on Microsoft Windows®). The native interface is the one actually provided by the hypervisor.

The recommended wrapper interface is described using standard C-style notation. The following is an example of a wrapper interface for the hypothetical HvAssignWidgets hypercall:

HV\_STATUS

HvAssignWidgets(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in UINT64 Flags,

\_\_inout PUINT32 RepCount,

\_\_in PCHV\_WIDGET WidgetList

);

The native interface is defined in terms of memory-based data structures. Up to four data structures may be defined:

Input parameter header

Input list element (for rep hypercalls)

Output parameter header

Output list element (for rep hypercalls)

The following is an example of the native interface documentation for the hypothetical HvAssignWidgets hypercall:

|  |  |  |
| --- | --- | --- |
| HvAssignWidgets [rep] | | |
|  | Call Code = 0xBADD | |
| 🡆 InputParameter Header | | |
| 0 | PartitionId (8 bytes) | |
| 8 | Flags (8 bytes) | |
| 🡆 Input List Element | | |
| 0 | WidgetId (8 bytes) | |
| 8 | WidgetType (4 bytes) | Padding (4 bytes) |

The above is an example of a rep (repeating) hypercall. As input, it has two fixed parameters and an input list consisting of one or more elements. The first list element can be found at offset 16. The list element is described using offsets within the element itself, starting with 0.

## Hypercall Restrictions

Hypercalls may have restrictions associated with them in order for them to perform their intended function. If all restrictions are not met, the hypercall will terminate with an appropriate error. The following restrictions will be listed, if any apply:

* The calling partition must possess a particular privilege (see 5.2.3 for information regarding privilege flags)
* The calling partition must have been created with a particular creation flag (see section 5.2.4 for information regarding creation flags)
* The partition being acted upon must be in a particular state (see section 5.2.5 for information regarding partition states)
* The partition must be the root
* The partition must be either a parent or child
* The virtual processor must be in a particular state (see section 10.1.3 for information regarding virtual processor states).

## Hypercall Status Codes

Each hypercall is documented as returning an output value that contains several fields. A status value field (of type HV\_STATUS) is used to indicate whether the call succeeded or failed. The hypercall status value field is discussed in section .

### Output Parameter Validity on Failed Hypercalls

When a hypercall fails (that is, the result field of the hypercall result value contains a value other than HV\_STATUS\_SUCCESS), the content of all output parameters are indeterminate and should not be examined by the caller. Only when the hypercall succeeds, will all appropriate output parameters contain valid, expected results.

### Ordering of Error Conditions

Error conditions are not presented in this document in any particular sequence. The order in which error conditions are detected and reported by the hypervisor is undefined. In other words, if multiple errors exist, the hypervisor must choose which error condition to report. Priority should be given to those error codes offering greater security, the intent being to prevent the hypervisor from revealing information to callers lacking sufficient privilege. For example, the status code HV\_STATUS\_ACCESS\_DENIED is the preferred status code over one that would reveal some context or state information purely based upon privilege.

### Common Hypercall Status Codes

Several result codes are common to all hypercalls and are therefore not documented for each hypercall individually. These include the following:

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_SUCCESS | The call succeeded. |
| HV\_STATUS\_INVALID\_HYPERCALL\_CODE | The hypercall code is not recognized. |
| HV\_STATUS\_INVALID\_HYPERCALL\_INPUT | The rep count is incorrect (for example, a non-zero rep count is passed to a non-rep call or a zero rep count is passed to a rep call). |
| The rep start index is not less than the rep count. |
| A reserved bit in the specified hypercall input value is non-zero. |
| HV\_STATUS\_INVALID\_ALIGNMENT | The specified input or output GPA pointer is not aligned to 8 bytes. |
| The specified input or output parameter lists spans pages. |
| The input or output GPA pointer is not within the bounds of the GPA space. |

The return code HV\_STATUS\_SUCCESS indicates that no error condition was detected.

## Establishing the Hypercall Interface

Hypercalls are invoked by using a special opcode. Because this opcode differs among virtualization implementations, it is necessary for the hypervisor to abstract this difference. This is done through a special *hypercall page*. This page is provided by the hypervisor and appears within the guest’s GPA space. The guest is required to specify the location of the page by programming an MSR.

#define HV\_X64\_MSR\_HYPERCALL 0x40000001

|  |  |  |
| --- | --- | --- |
| 63:12 | 11:1 | 0 |
| Guest Physical Page Number of Hypercall Page | RsvdP | Enable |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:12 | Hypercall GPFN  Indicates the **Guest Physical Page Number** of the hypercall page | Read/write |
| 11:1 | RsvdP  Guest should ignore on reads and preserve on writes | Reserved |
| 0 | Enable Hypercall Page  Enables the hypercall page | Read/write |

The hypercall page can be placed anywhere within the guest’s GPA space, but must be page-aligned. If the guest attempts to move the hypercall page beyond the bounds of the GPA space, a #GP fault will result when the MSR is written.

This MSR is a partition-wide MSR. In other words, it is shared by all virtual processors in the partition. If one virtual processor successfully writes to the MSR, another virtual processor will read the same value.

Before the hypercall page is enabled, the guest OS must report its identity by writing its version signature to a separate MSR (HV\_X64\_MSR\_GUEST\_OS\_ID). If no guest OS identity has been specified, attempts to enable the hypercall will fail. The enable bit will remain zero even if a one is written to it. Furthermore, if the guest OS identity is cleared to zero after the hypercall page has been enabled, it will become disabled.

The hypercall page appears as an “overlay” to the GPA space; that is, it covers whatever else is mapped to the GPA range. Its contents are readable and executable by the guest. Attempts to write to the hypercall page will result in a protection (#GP) exception.

After the hypercall page has been enabled, invoking a hypercall simply involves a call to the start of the page.

The following is a detailed list of the steps involved in establishing the hypercall page:

The guest reads CPUID leaf 1 and determines whether a hypervisor is present by checking bit 31 of register ECX.

The guest reads CPUID leaf 0x40000000 to determine the maximum hypervisor CPUID leaf (returned in register EAX) and CPUID leaf 0x40000001 to determine the interface signature (returned in register EAX). It verifies that the maximum leaf value is at least 0x40000005 and that the interface signature is equal to “Hv#1”. This signature implies that HV\_X64\_MSR\_GUEST\_OS\_ID, HV\_X64\_MSR\_HYPERCALL and HV\_X64\_MSR\_VP\_INDEX are implemented.

The guest writes its OS identity into the MSR HV\_X64\_MSR\_GUEST\_OS\_ID if that register is zero.

The guest reads the Hypercall MSR (HV\_X64\_MSR\_HYPERCALL).

The guest checks the Enable Hypercall Page bit. If it is set, the interface is already active, and steps 6 and 7 should be omitted.

The guest finds a page within its GPA space, preferably one that is not occupied by RAM, MMIO, and so on. If the page is occupied, the guest should avoid using the underlying page for other purposes.

The guest writes a new value to the Hypercall MSR (HV\_X64\_MSR\_HYPERCALL) that includes the GPA from step 6 and sets the Enable Hypercall Page bit to enable the interface.

The guest creates an executable VA mapping to the hypercall page GPA.

The guest consults CPUID leaf 0x40000003 to determine which hypervisor facilities are available to it.

After the interface has been established, the guest can initiate a hypercall. To do so, it populates the registers per the hypercall protocol and issues a CALL to the beginning of the hypercall page. The guest should assume the hypercall page performs the equivalent of a near ret (0xC3) to return to the caller. As such, the hypercall must be invoked with a valid stack.

# 

# Partition Management

## Overview

This section defines interfaces that allow guests to create and delete partitions, to enumerate existing partitions and to manipulate their state.

## Partition Management Data Types

### Partition IDs

Partitions are identified by using a partition ID. This 64-bit number is allocated by the hypervisor. All partitions are guaranteed by the hypervisor to have unique IDs. Note that these are not “globally unique” in that the same ID may be generated across a power cycle (that is, a reboot of the hypervisor). However, the hypervisor guarantees that IDs created within a single power cycle are unique.

typedef UINT64 HV\_PARTITION\_ID;

typedef HV\_PARTITION\_ID \*PHV\_PARTITION\_ID;

The guest should not ascribe any meaning to the value of a partition ID. The “invalid” partition ID is used in several interfaces to indicate an invalid partition.

#define HV\_PARTITION\_ID\_INVALID 0x0000000000000000UI64

### Partition Properties

Partition properties provide a generic way for a parent partition to control aspects of its child partitions. After a partition property is set, its value is constant unless and until it is again modified by a caller possessing sufficient privilege. Properties are identified by a 32-bit code. Property values are each 64 bits in size.

typedef UINT64 HV\_PARTITION\_PROPERTY;

typedef HV\_PARTITION\_PROPERTY \*PHV\_PARTITION\_PROPERTY;

typedef enum

{

// Privilege properties

HvPartitionPropertyPrivilegeFlags = 0x00010000,

// Scheduling properties

HvPartitionPropertyCpuReserve = 0x00020001,

HvPartitionPropertyCpuCap = 0x00020002,

HvPartitionPropertyCpuWeight = 0x00020003,

// Timer assist properties

HvPartitionPropertyEmulatedTimerPeriod = 0x00030000,

HvPartitionPropertyEmulatedTimerControl = 0x00030001,

HvPartitionPropertyPmTimerAssist = 0x00030002,

// Debugging properties

HvPartitionPropertyDebugChannelId = 0x00040000,

// Resource properties

HvPartitionPropertyVirtualTlbPageCount = 0x00050000,

// Compatibility properties

HvPartitionPropertyProcessorVendor = 0x00060000,

HvPartitionPropertyProcessorFeatures = 0x00060001,

HvPartitionPropertyProcessorXsaveFeatures = 0x00060002,

HvPartitionPropertyProcessorCLFlushSize = 0x00060003

} HV\_PARTITION\_PROPERTY\_CODE;

The following table explains the meaning of each property. The table also lists the default values.

| Partition property | Meaning | | Default value |
| --- | --- | --- | --- |
| HvPartitionPropertyPrivilegeFlags | For details about the privilege flags, see section 5.2.3. | | |
|  |  | | |
| HvPartitionPropertyCpuReserve | The amount of per-VP CPU time reserved for this partition. For details, see section . | | 0x0000000000000000 |
| HvPartitionPropertyCpuCap | The maximum amount of per-VP CPU time the partition is allowed to use. For details, see section . | | 0x0000000000010000 |
| HvPartitionPropertyCpuWeight | The relative weight for the virtual processors in the partition. For details, see section . | | 0x0000000000000064 |
| HvPartitionPropertyEmulatedTimerPeriod | The periodic timer assist period. For details, see section . | | 0x0000000000000000 |
| HvPartitionPropertyEmulatedTimerControl | The periodic timer assist control. For details, see section . | | 0x0000000000000000 |
| HvPartitionPropertyPmTimerAssist | The PM timer assist definition. For details, see section . | | 0x0000000000000000 |
| HvPartitionPropertyDebugChannelId | The debug session channel identifier. For details, see section . | | 0x0000000000000000 |
| HvPartitionPropertyVirtualTlbPageCount | The number of pool pages reserved for the virtual TLB. For details, see section 5.2.6. | | |
| HvPartitionPropertyProcessorVendor | | The processor vendor. For details, see section 5.2.7. Read-only. | The vendor of the platform’s processors. |
| HvPartitionPropertyProcessorFeatures | | The bitmask representing supported processor features. For details, see section 5.2.8. Default value is the value for the root partition. Read-only for the root partition. | The guest visible features supported and enabled by the platform. |
| HvPartitionPropertyProcessorXsaveFeatures | | The bitmask representing supported processor XSAVE related features. For details, see section . Default value is the value for the root partition. Read-only for the root partition. | The guest visible XSAVE features supported and enabled by the platform. |
| HvPartitionPropertyProcessorCLFlushSize | | The size of the processor cache line flush size. For details, see section . Default value is the value for the root partition. Read-only for the root partition. | The minimum cache line flush size of all processors on the platform. |

### Partition Privilege Flags

One of the partition properties (HvPartitionPropertyPrivilegeFlags) defines the hypervisor facilities that the partition is allowed to access. This enables the parent to control which synthetic MSRs and hypercalls a child partition can access.

The property is defined with the following structure:

typedef struct

{

// Access to virtual MSRs

UINT64 AccessVpRunTimeMsr:1;

UINT64 AccessPartitionReferenceCounter:1;

UINT64 AccessSynicMsrs:1;

UINT64 AccessSyntheticTimerMsrs:1;

UINT64 AccessApicMsrs:1;

UINT64 AccessHypercallMsrs:1;

UINT64 AccessVpIndex:1;

UINT64 AccessResetMsr:1;

UINT64 AccessStatsMsr:1;

UINT64 AccessPartitionReferenceTsc:1;

UINT64 AccessGuestIdleMsr:1;

UINT64 Reserved1:`21;

// Access to hypercalls

UINT64 CreatePartitions:1;

UINT64 AccessPartitionId:1;

UINT64 AccessMemoryPool:1;

UINT64 AdjustMessageBuffers:1;

UINT64 PostMessages:1;

UINT64 SignalEvents:1;

UINT64 CreatePort:1;

UINT64 ConnectPort:1;

UINT64 AccessStats:1;

UINT64 Reserved2:2;

UINT64 Debugging:1;

UINT64 CpuManagement:1;

UINT64 ConfigureProfiler:1;

UINT64 Reserved3:18;

} HV\_PARTITION\_PRIVILEGE\_MASK;

The following table explains what each of these flags controls. The table also lists the default value for non-root partitions. The root partition has access to all capabilities.

| Privilege Flag | Meaning | Default Value |
| --- | --- | --- |
| AccessVpRunTimeMsr | The partition has access to the synthetic MSR HV\_X64\_MSR\_VP\_RUNTIME. If this flag is cleared, accesses to this MSR results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessPartitionReferenceCounter | The partition has access to the partition-wide reference count MSR, HV\_X64\_MSR\_TIME\_REF\_COUNT. If this flag is cleared, accesses to this MSR results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessSynicMsrs | The partition has access to the synthetic MSRs associated with the Synic (HV\_X64\_MSR\_SCONTROL through HV\_X64\_MSR\_EOM and HV\_X64\_MSR\_SINT0 through HV\_X64\_MSR\_SINT15). If this flag is cleared, accesses to these MSRs results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessSyntheticTimerMsrs | The partition has access to the synthetic MSRs associated with the Synic (HV\_X64\_MSR\_STIMER0\_CONFIG through HV\_X64\_MSR\_STIMER3\_COUNT). If this flag is cleared, accesses to these MSRs results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessApicMsrs | The partition has access to the synthetic MSRs associated with the APIC (HV\_X64\_MSR\_EOI, HV\_X64\_MSR\_ICR and HV\_X64\_MSR\_TPR). If this flag is cleared, accesses to these MSRs results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessHypercallMsrs | The partition has access to the synthetic MSRs related to the hypercall interface (HV\_X64\_MSR\_GUEST\_OS\_ID and HV\_X64\_MSR\_HYPERCALL). If this flag is cleared, accesses to these MSRs result in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessVpIndex | The partition has access to the synthetic MSR that returns the virtual processor index. If this flag is cleared, accesses to this MSR results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessResetMsr | This partition has access to the synthetic MSR that resets the system. . If this flag is cleared, accesses to this MSR results in a #GP fault if the MSR intercept is not installed. | 1 |
| AccessStatsMsr | This partition has access to the synthetic MSRs that allows the guest to map and unmap its own statistics pages. | 1 |
| AccessPartitionReferenceTsc | The partition has access to the reference TSC. | 0 |
| AccessGuestIdleMsr | The partition has access to the synthetic MSR that allows the guest to enter the guest idle state. | 1 |
| CreatePartitions | The partition can invoke the hypercall HvCreatePartition. The partition also can make any other hypercall that is restricted to operating on children. | 0 |
| AccessPartitionId | The partition can invoke the hypercall HvGetPartitionId to obtain its own partition ID. | 0 |
| AccessMemoryPool | The partition can invoke the hypercalls HvDepositMemory, HvWithdrawMemory and HvGetMemoryBalance. | 0 |
| PostMessages | The partition can invoke the hypercall HvPostMessage. | 0 |
| SignalEvents | The partition can invoke the hypercall HvSignalEvent. | 0 |
| CreatePort | The partition can invoke the hypercall HvCreatePort. | 0 |
| ConnectPort | The partition can invoke the hypercall HvConnectPort. | 0 |
| AccessStats | The partition can invoke the hypercalls HvMapStatsPage and HvUnmapStatsPage. | 0 |
| Debugging | The partition can invoke the hypercalls HvPostDebugData, HvRetrieveDebugData and HvResetDebugSession. | 0 |
| CpuPowerManagement [[1]](#footnote-1) | The partition can invoke the hypercalls  HvGetLogicalProcessorRunTime  andHvCallParkLogicalProcessors .  This partition also has access to the power management MSRs. | 0 |

### Partition Creation Flags

When a partition is created, the parent has the option of specifying a *creation flags* mask*.* Creation flags are reserved for future use and should be set to zero.

### Partition State

Each partition is in one of four states:

**Uninitialized** – The state of the partition after it has been created by a call to HvCreatePartition. Only the HvDepositMemory, HvQueryMemoryBalance, HvGetNextChildPartition and HvInitializePartition hypercalls are permitted on *uninitialized* partitions.

**Active** – The state of the partition after it has been successfully initialized by a call to HvInitializePartition. All hypercalls are permitted on *active* partitions, with the exception of HvDeletePartition.

**Finalizing** – The transitory state of the partition after HvFinalizePartition has been invoked but not yet completed. Only the HvQueryMemoryBalance, HvWithdrawMemory, HvGetNextChildPartition and HvFinalizePartition hypercalls are permitted on *finalizing* partitions.

**Finalized** – The state of the partition after it has been finalized by a successfully completed call to HvFinalizePartition. Only the HvQueryMemoryBalance, HvWithdrawMemory, HvGetNextChildPartition and HvDeletePartition hypercalls are permitted on *finalized* partitions.

Figure is a state diagram that shows the valid states and actions that can result in state transitions. All state transitions appear to be atomic from the perspective of hypervisor callers.

![](data:image/x-emf;base64,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AAAAAP//////////BgAAAIUTaAuKE2YLjhNnC5ITaAuVE20LkxNxCzYAAAAQAAAAhRMAAJ8LAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAhBOjC4ATpgt7E6QLdxOjC3UTnwt2E5oLPQAAAAgAAAAbAAAAEAAAAJUTAAA+CwAANgAAABAAAACsEwAAFAsAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAACuExALsxMPC7cTEQu7ExMLvBMYC7oTHAs2AAAAEAAAAKMTAABGCwAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAKETSgucE0sLmBNJC5UTRwuTE0ILlRM+Cz0AAAAIAAAAGwAAABAAAADDEwAA6goAADYAAAAQAAAAwxMAAOoKAABYAAAAKAAAAAAAAAAAAAAA//////////8DAAAAxBPpCsQT6QrEE+gKNgAAABAAAADjEwAAxAoAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADlE8AK6xPACu4TwgrxE8UK8hPKCu8TzgpZAAAAKAAAAAAAAAAAAAAA//////////8DAAAA0RPzCtET8QrRE/IKWAAAADQAAAAAAAAAAAAAAP//////////BgAAAM8T9QrKE/cKxhP1CsMT8wrBE+4KwxPqCj0AAAAIAAAAGwAAABAAAAABFAAAnwoAADYAAAAQAAAACxQAAJIKAABYAAAAKAAAAAAAAAAAAAAA//////////8DAAAADBSSCgwUkgoMFJEKNgAAABAAAAAlFAAAfQoAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAAApFHoKLhR7CjAUfgozFIEKMxSGCi8UiQpZAAAAKAAAAAAAAAAAAAAA//////////8DAAAAFxSeChgUnQoOFKkKWAAAADQAAAAAAAAAAAAAAP//////////BgAAAAsUrAoGFK0KAhSqCv8Tpwr+E6IKARSfCj0AAAAIAAAAGwAAABAAAABKFAAAXgoAADYAAAAQAAAAYRQAAEsKAABYAAAAKAAAAAAAAAAAAAAA//////////8DAAAAYhRLCmIUSwpjFEoKNgAAABAAAABzFAAAQgoAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAAB2FEAKexRBCn0URQqAFEkKfhROCnoUUApZAAAAKAAAAAAAAAAAAAAA//////////8DAAAAahRZCmwUWApUFGsKWAAAADQAAAAAAAAAAAAAAP//////////BgAAAFEUbQpMFG0KSRRqCkYUZgpHFGEKShReCj0AAAAIAAAAGwAAABAAAACdFAAAKwoAADYAAAAQAAAAxRQAABUKAABYAAAAKAAAAAAAAAAAAAAA//////////8DAAAAxRQVCsYUFQrGFBUKNgAAABAAAADJFAAAFAoAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADNFBMK0RQVCtMUGQrUFB4K0hQiCs0UIwpZAAAAKAAAAAAAAAAAAAAA//////////8DAAAAyxQkCswUJAqkFDkKWAAAADQAAAAAAAAAAAAAAP//////////BgAAAKEUOwqcFDoKmhQ2CpcUMgqZFC0KnRQrCj0AAAAIAAAAGwAAABAAAAD3FAAABgoAADYAAAAQAAAAJBUAAPgJAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAKRX3CS0V+QkuFf0JMBUCCi0VBgopFQcKNgAAABAAAAD7FAAAFQoAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAAD3FBcK8xQUCvEUEArwFAwK8hQHCvcUBgo9AAAACAAAABsAAAAQAAAAVRUAAO4JAABZAAAAJAAAAAAAAAAAAAAA//////////8CAAAAbBXrCYUV6QlYAAAANAAAAAAAAAAAAAAA//////////8GAAAAiRXpCY0V7AmNFfEJjhX1CYoV+QmGFfkJWQAAACQAAAAAAAAAAAAAAP//////////AgAAAG8V+glXFf4JWAAAADQAAAAAAAAAAAAAAP//////////BgAAAFMV/wlPFfwJThX3CU0V8wlQFe8JVRXuCT0AAAAIAAAAGwAAABAAAAC2FQAA6AkAAFkAAAAkAAAAAAAAAAAAAAD//////////wIAAADkFeoJ5xXrCVgAAAA0AAAAAAAAAAAAAAD//////////wYAAADrFewJ7hXwCe0V9AntFfgJ6BX7CeQV+wlZAAAAJAAAAAAAAAAAAAAA//////////8CAAAA4xX6CbUV+AlYAAAANAAAAAAAAAAAAAAA//////////8GAAAAsRX4Ca0V9AmtFfAJrhXrCbEV6Am2FegJPQAAAAgAAAAbAAAAEAAAABYWAADyCQAAWQAAACQAAAAAAAAAAAAAAP//////////AgAAAB4W9AlFFgAKWAAAADQAAAAAAAAAAAAAAP//////////BgAAAEkWAQpMFgUKSxYKCkkWDgpFFhAKQRYPClkAAAAkAAAAAAAAAAAAAAD//////////wIAAAAbFgMKExYCClgAAAA0AAAAAAAAAAAAAAD//////////wYAAAAPFgEKDBb9CQ0W+QkOFvUJEhbyCRYW8gk9AAAACAAAABsAAAAQAAAAcxYAAA4KAAA2AAAAEAAAAIsWAAAVCgAAWAAAACgAAAAAAAAAAAAAAP//////////AwAAAIsWFQqMFhUKjBYVCjYAAAAQAAAAoRYAACAKAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAApRYjCqYWJwqkFisKohYvCp0WMQqZFi8KWQAAACgAAAAAAAAAAAAAAP//////////AwAAAIUWJAqGFiQKbhYdClgAAAA0AAAAAAAAAAAAAAD//////////wYAAABqFhwKaBYXCmkWEwpqFg8KbxYMCnMWDgo9AAAACAAAABsAAAAQAAAAyxYAADcKAAA2AAAAEAAAAO4WAABKCgAAWAAAACgAAAAAAAAAAAAAAP//////////AwAAAO8WSwrvFksK8BZLCjYAAAAQAAAA9hYAAFAKAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAA+RZTCvoWWAr3FlwK9BZfCu8WXwrrFl0KWQAAACgAAAAAAAAAAAAAAP//////////AwAAAOUWWArnFlkKwxZFClgAAAA0AAAAAAAAAAAAAAD//////////wYAAAC/FkMKvhY+CsAWOwrCFjcKxxY1CssWNwo9AAAACAAAABsAAAAQAAAAGxcAAG8KAAA2AAAAEAAAAEAXAACNCgAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAEMXkApEF5UKQReZCj4XnAo5F5wKNheaCjYAAAAQAAAAEBcAAHsKAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAADRd4Cg0XcwoPF3AKEhdsChcXbAobF28KPQAAAAgAAAAbAAAAEAAAAGAXAACzCgAANgAAABAAAAB/FwAA2AoAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAACCF9sKgRfgCn4X4wp6F+YKdRflCnIX4go2AAAAEAAAAFQXAAC9CgAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAFEXuQpSF7QKVRexClgXrwpdF68KYBezCj0AAAAIAAAAGwAAABAAAACaFwAAAQsAADYAAAAQAAAAsRcAACsLAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAsxcvC7IXMwuuFzYLqhc4C6UXNgujFzILNgAAABAAAACMFwAACAsAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAACKFwQLixcAC48X/QqTF/sKmBf9CpoXAQs9AAAACAAAABsAAAAQAAAAxxcAAFcLAAA2AAAAEAAAANQXAACFCwAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAANYXiQvTF44LzxePC8sXkQvGF44LxReKCzYAAAAQAAAAtxcAAFwLAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAthdYC7gXUwu9F1ILwRdRC8UXUwvHF1cLPQAAAAgAAAAbAAAAEAAAAOIXAACzCwAAWQAAACQAAAAAAAAAAAAAAP//////////AgAAAOQXuQvrF+MLWAAAADQAAAAAAAAAAAAAAP//////////BgAAAOwX6AvpF+wL5BftC+AX7QvcF+oL2xfmC1kAAAAkAAAAAAAAAAAAAAD//////////wIAAADVF74L0xe4C1gAAAA0AAAAAAAAAAAAAAD//////////wYAAADSF7QL1BevC9gXrgvdF60L4RevC+IXsws9AAAACAAAABsAAAAQAAAA7xcAABQMAAA2AAAAEAAAAPAXAAAvDAAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAPEXMwztFzcM6Rc3DOQXOAzhFzQM4BcwDDYAAAAQAAAA3xcAABUMAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAA3xcQDOIXDQznFw0M6xcMDO8XEAzvFxQMPQAAAAgAAAAbAAAAEAAAAPAXAAAwDAAANgAAABAAAADvFwAARQwAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADvF0kM6xdNDOcXTAzjF0wM3xdIDN8XRAw2AAAAEAAAAOAXAAAvDAAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAOEXKwzkFycM6RcnDO0XKAzxFysM8BcwDD0AAAAIAAAAGwAAABAAAADsFwAAdgwAAFkAAAAkAAAAAAAAAAAAAAD//////////wIAAADkF6UM5BemDFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADjF6oM3hetDNoXqwzWF6oM0xemDNUXoQxZAAAAKAAAAAAAAAAAAAAA//////////8DAAAA1ReiDNUXogzcF3MMWAAAADQAAAAAAAAAAAAAAP//////////BgAAAN0XbwzhF2wM5RdsDOoXbQztF3EM7Bd2DD0AAAAIAAAAGwAAABAAAADWFwAA1AwAADYAAAAQAAAAyBcAAAINAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAxxcGDcIXCA2+FwcNuhcGDbcXAQ25F/0MNgAAABAAAADHFwAAzwwAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADIF8sMzBfJDNEXygzVF8sM1xfQDNYX1Aw9AAAACAAAABsAAAAQAAAAtBcAAC8NAAA2AAAAEAAAAJ0XAABZDQAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAJsXXQ2WF14NkhdcDY4XWg2NF1UNjxdRDTYAAAAQAAAAphcAACcNAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAqBcjDa0XIg2wFyQNtBcmDbYXKw20Fy8NPQAAAAgAAAAbAAAAEAAAAIIXAACDDQAANgAAABAAAABkFwAAqA0AAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAABhF6sNXBerDVgXqQ1VF6YNVRehDVcXnQ02AAAAEAAAAHYXAAB5DQAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAHkXdQ1+F3UNgRd4DYUXeg2FF38NgheDDT0AAAAIAAAAGwAAABAAAABEFwAAzQ0AADYAAAAQAAAAHxcAAOwNAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAHBfvDRYX7g0UF+sNERfoDREX4w0VF+ANNgAAABAAAAA5FwAAwQ0AAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAAA9F74NQhe/DUUXwg1IF8UNRxfKDUQXzQ09AAAACAAAABsAAAAQAAAA+hYAAAsOAAA2AAAAEAAAAPAWAAAUDgAAWAAAACgAAAAAAAAAAAAAAP//////////AwAAAO8WFA7vFhQO7hYVDjYAAAAQAAAA0BYAACUOAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAzBYnDscWJg7FFiIOwxYeDsUWGQ7IFhcOWQAAACgAAAAAAAAAAAAAAP//////////AwAAAOcWBg7lFgcO8Bb/DVgAAAA0AAAAAAAAAAAAAAD//////////wYAAADzFvwN+Bb8DfsWAA7+FgMO/RYIDvoWCw49AAAACAAAABsAAAAQAAAAphYAADwOAAA2AAAAEAAAAIwWAABKDgAAWAAAACgAAAAAAAAAAAAAAP//////////AwAAAIwWSg6LFkoOixZKDjYAAAAQAAAAeRYAAFAOAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAdBZRDnAWTw5vFkoObRZGDnAWQg50FkAOWQAAACgAAAAAAAAAAAAAAP//////////AwAAAIYWOw6FFjsOnhYuDlgAAAA0AAAAAAAAAAAAAAD//////////wYAAACiFiwOpxYtDqkWMQ6rFjUOqhY6DqYWPA49AAAACAAAABsAAAAQAAAASxYAAF4OAABZAAAAJAAAAAAAAAAAAAAA//////////8CAAAAHxZrDhwWbA5YAAAANAAAAAAAAAAAAAAA//////////8GAAAAFxZsDhMWaQ4TFmUOEhZhDhUWXQ4ZFlwOWQAAACQAAAAAAAAAAAAAAP//////////AgAAABoWXA5GFk4OWAAAADQAAAAAAAAAAAAAAP//////////BgAAAEoWTQ5PFk8OUBZUDlEWWA5PFlwOSxZeDj0AAAAIAAAAGwAAABAAAADsFQAAcw4AAFkAAAAkAAAAAAAAAAAAAAD//////////wIAAADlFXQOvBV3DlgAAAA0AAAAAAAAAAAAAAD//////////wYAAAC3FXcOsxVzDrMVbw6zFWsOthVnDrsVZw5ZAAAAJAAAAAAAAAAAAAAA//////////8CAAAA4hVlDuoVYw5YAAAANAAAAAAAAAAAAAAA//////////8GAAAA7hVjDvIVZg7zFWoO9BVuDvEVcg7sFXMOPQAAAAgAAAAbAAAAEAAAAIsVAAB2DgAAWQAAACQAAAAAAAAAAAAAAP//////////AgAAAG0VdA5aFXIOWAAAADQAAAAAAAAAAAAAAP//////////BgAAAFYVcQ5TFW0OVBVoDlQVZA5YFWEOXRViDlkAAAAkAAAAAAAAAAAAAAD//////////wIAAABuFWQOjBVmDlgAAAA0AAAAAAAAAAAAAAD//////////wYAAACQFWYOkxVqDpMVbg6TFXMOjxV2DosVdg49AAAACAAAABsAAAAQAAAAKhUAAGkOAAA2AAAAEAAAAPwUAABbDgAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAPgUWQ72FFUO9xRRDvgUTA79FEoOARVLDjYAAAAQAAAALxUAAFkOAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAMxVbDjUVXw40FWMOMxVoDi4Vag4qFWkOPQAAAAgAAAAbAAAAEAAAAM4UAABNDgAANgAAABAAAADGFAAASg4AAFgAAAAoAAAAAAAAAAAAAAD//////////wMAAADGFEoOxRRKDsUUSg42AAAAEAAAAKIUAAA3DgAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAJ4UNQ6dFDAOnxQsDqEUKA6mFCcOqhQpDlkAAAAoAAAAAAAAAAAAAAD//////////wMAAADMFDsOyxQ7DtMUPQ5YAAAANAAAAAAAAAAAAAAA//////////8GAAAA1xQ/DtoUQw7YFEcO1xRMDtIUTg7OFE0OPQAAAAgAAAAbAAAAEAAAAHgUAAAgDgAANgAAABAAAABjFAAAFQ4AAFgAAAAoAAAAAAAAAAAAAAD//////////wMAAABiFBQOYhQUDmEUFA42AAAAEAAAAE8UAAAEDgAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAEsUAQ5LFPwNThT5DVEU9Q1WFPUNWRT4DVkAAAAoAAAAAAAAAAAAAAD//////////wMAAABsFAcOahQGDn8UEg5YAAAANAAAAAAAAAAAAAAA//////////8GAAAAgxQUDoUUGQ6DFB0OgBQhDnwUIg54FCAOPQAAAAgAAAAbAAAAEAAAACoUAADlDQAANgAAABAAAAAMFAAAzQ0AAFgAAAAoAAAAAAAAAAAAAAD//////////wMAAAAMFMwNDBTMDQsUzA02AAAAEAAAAAUUAADEDQAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAAIUwQ0DFLwNBhS5DQoUtg0PFLcNEhS6DVkAAAAoAAAAAAAAAAAAAAD//////////wMAAAAYFMENFxTADTQU2Q1YAAAANAAAAAAAAAAAAAAA//////////8GAAAAOBTcDTgU4Q01FOQNMhToDS0U6A0qFOUNPQAAAAgAAAAbAAAAEAAAAOcTAACgDQAANgAAABAAAADIEwAAew0AAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADFE3cNxRNyDckTbw3ME20N0RNtDdQTcA02AAAAEAAAAPMTAACVDQAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAPYTmQ31E54N8hOhDe4Tow3pE6MN5xOgDT0AAAAIAAAAGwAAABAAAACvEwAAUA0AADYAAAAQAAAAmBMAACYNAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAlhMiDZgTHQ2cExsNnxMZDaQTGg2mEx4NNgAAABAAAAC9EwAASA0AAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAAC/E0wNvhNRDboTUw22E1UNsRNUDa8TUA09AAAACAAAABsAAAAQAAAAhhMAAPgMAAA2AAAAEAAAAHgTAADKDAAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAHcTxgx5E8EMfRPADIETvwyGE8EMhxPFDDYAAAAQAAAAlRMAAPMMAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAAlhP3DJQT/AyQE/0MjBP+DIcT/AyGE/gMPQAAAAgAAAAbAAAAEAAAAGsTAACbDAAANgAAABAAAABkEwAAbAwAAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAABjE2cMZhNjDGoTYwxvE2IMcxNlDHMTaQw2AAAAEAAAAHsTAACYDAAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAHsTnAx4E6EMdBOhDHATogxsE58MaxObDD0AAAAIAAAAPAAAAAgAAAA/AAAAGAAAADQBAACdAAAAgQEAAOkAAAATAAAADAAAAAEAAAAlAAAADAAAAAAAAIAlAAAADAAAAAcAAIAkAAAAJAAAAAAAgEEAAAAAAAAAAAAAgEEAAAAAAAAAAAIAAAAoAAAADAAAAAEAAABGAAAA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AAAAAP//////////AgAAAJIVyAp2FRsLWAAAADQAAAAAAAAAAAAAAP//////////BgAAAHUVHwtwFSELbBUgC2gVHgtmFRoLZxUWC1kAAAAkAAAAAAAAAAAAAAD//////////wIAAACDFcUKiBWsClgAAAA0AAAAAAAAAAAAAAD//////////wYAAACJFacKjRWlCpEVpgqWFaYKmRWrCpgVrwo9AAAACAAAABsAAAAQAAAAWRUAAGYLAABZAAAAJAAAAAAAAAAAAAAA//////////8CAAAAMRW/CyoVzAtYAAAANAAAAAAAAAAAAAAA//////////8GAAAAJxXQCyMV0QsfFc8LGxXNCxkVyAsbFcQLWQAAACQAAAAAAAAAAAAAAP//////////AgAAACIVuAtKFV8LWAAAADQAAAAAAAAAAAAAAP//////////BgAAAEwVWwtRFVkLVRVbC1kVXQtaFWILWRVmCz0AAAAIAAAAGwAAABAAAAADFQAAEgwAAFkAAAAkAAAAAAAAAAAAAAD//////////wIAAADsFDsMyBRyDFgAAAA0AAAAAAAAAAAAAAD//////////wYAAADGFHYMwRR3DL0UdAy5FHIMuBRtDLsUaQxZAAAAJAAAAAAAAAAAAAAA//////////8CAAAA3hQ0DPUUCgxYAAAANAAAAAAAAAAAAAAA//////////8GAAAA9xQHDPwUBQwAFQcMBBUJDAUVDgwDFRIMPQAAAAgAAAAbAAAAEAAAAJwUAAC1DAAANgAAABAAAABXFAAADg0AAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAABUFBENTxQSDUwUDw1IFAwNRxQHDUoUBA02AAAAEAAAAI8UAACsDAAAWAAAADQAAAAAAAAAAAAAAP//////////BgAAAJIUqAyXFKcMmhSqDJ4UrQyeFLIMnBS1DD0AAAAIAAAAGwAAABAAAAAiFAAASg0AADYAAAAQAAAA4RMAAI4NAABYAAAANAAAAAAAAAAAAAAA//////////8GAAAA3hORDdkTkQ3WE44N0xOLDdMThg3WE4MNNgAAABAAAAAWFAAAPw0AAFgAAAA0AAAAAAAAAAAAAAD//////////wYAAAAZFDwNHhQ8DSEUPw0lFEINJRRHDSIUSg09AAAACAAAADwAAAAIAAAAPwAAABgAAAA7AQAAnAAAAF0BAADbAAAAEwAAAAwAAAABAAAAJQAAAAwAAAAAAACAJQAAAAwAAAAHAACAJAAAACQAAAAAAIBBAAAAAAAAAAAAAIBBAAAAAAAAAAACAAAAKAAAAAwAAAADAAAARgAAAJwAAACQAAAARU1GKwhAAQJAAAAANAAAAAIQwNsAAAAAzgAAAAAAAAAK16M+AgAAAAIAAAACAAAAAgAAAAAAAAACEMDbAAAAAAAAAP8IQAADPAAAADAAAAACEMDbBAAAAAAAAAB2KadDGsbEQgg/rENz0NtCRfiuQ7f9CEPLPq1DcOEeQwADAwMVQAAAEAAAAAQAAAABAAAAJAAAACQAAAAAAIA9AAAAAAAAAAAAAIA9AAAAAAAAAAACAAAAXwAAADgAAAADAAAAOAAAAAAAAAA4AAAAAAAAAAAAAQAFAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAlAAAADAAAAAMAAAAlAAAADAAAAAUAAIBVAAAALAAAAE0BAABhAAAAYAEAAKEAAAAEAAAA5hQnBogV3wbgFZAIqBXvCSUAAAAMAAAABwAAgCUAAAAMAAAAAAAAgCQAAAAkAAAAAACAQQAAAAAAAAAAAACAQQAAAAAAAAAAAgAAACgAAAAMAAAAAwAAAEYAAAA0AAAAKAAAAEVNRisKQACAJAAAABgAAAD/////AQAAAH5pj0MbLgxDVFXbQlBVVUEoAAAADAAAAAEAAAAkAAAAJAAAAAAAgD0AAAAAAAAAAAAAgD0AAAAAAAAAAAIAAAAnAAAAGAAAAAEAAAAAAAAA////AAAAAAAlAAAADAAAAAEAAAAlAAAADAAAAAgAAIBWAAAAMAAAAB4BAACMAAAAjQEAAJoAAAAFAAAA7hHDCO4RmQnIGJkJyBjDCO4RwwglAAAADAAAAAcAAIAlAAAADAAAAAAAAIAkAAAAJAAAAAAAgEEAAAAAAAAAAAAAgEEAAAAAAAAAAAIAAABGAAAAKAEAABwBAABFTUYrKkAAACQAAAAYAAAAAADAQgAAAAAAAAAAAADAQn5pj0MbLgxDNkACgPQAAADoAAAAAAAA/wEAAAABAAAAEwAAAEgAdgBGAGkAbgBhAGwAaQB6AGUAUABhAHIAdABpAHQAaQBvAG4AAEjdPQB80j0Az0A+AHzSPQCzeT4AfNI9gJmfPgB80j0APaw+AHzSPYDgyz4AfNI9AITrPgB80j2AJ/g+AHzSPYBlAj8AfNI9gJ4QPwB80j1AcCA/AHzSPQBpMz8AfNI9wDpDPwB80j3As0w/AHzSPcCaVD8AfNI9gOxaPwB80j2A02I/AHzSPUAlaT8AfNI9APd4PwB80j0AAIA/AAAAAAAAAAAAAIA/AAAAAAAAAAAAABIAAAAMAAAAAQAAABgAAAAMAAAAAAAAABYAAAAMAAAAGAAAACUAAAAMAAAAAgAAAFQAAADAAAAAKQEAAIsAAAB/AQAAmAAAAAEAAAAA0ARCAKAMQikBAACWAAAAEwAAAEwAAAAAAAAAAAAAAAAAAAD//////////3QAAABIAHYARgBpAG4AYQBsAGkAegBlAFAAYQByAHQAaQB0AGkAbwBuAAAACAAAAAUAAAAHAAAAAgAAAAYAAAAGAAAAAgAAAAIAAAAFAAAABgAAAAcAAAAGAAAABAAAAAMAAAACAAAAAwAAAAIAAAAGAAAAAAAAACUAAAAMAAAADQAAgEYAAAAcAAAAEAAAAEVNRisrQAAADAAAAAAAAABMAAAAZAAAAAIAAAACAAAArgEAACMBAAACAAAAAgAAAK0BAAAiAQAAKQCqAAAAAAAAAAAAAACAPwAAAAAAAAAAAACAPwAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAIgAAAAwAAAD/////RgAAABwAAAAQAAAARU1GKwJAAAAMAAAAAAAAAA4AAAAUAAAAAAAAABAAAAAUAAAA)

Figure 1. Partition management state diagram

### Partition Virtual TLB Page Count

The hypervisor maintains a virtual TLB for each virtual processor (see section 12.1). The memory available for the virtual TLB is reserved and allocated from the partition’s memory pool. The count of pages reserved is a partition property that can be modified dynamically. The hypervisor establishes a default value, which may be different for the root and other (non-root) partitions. It also enforces minimum and maximum page count restrictions. Attempts to set the page count outside of the restrictions will result in the value being set to the appropriate minimum or maximum value.

### Partition Processor Vendor

One of the partition properties (HvPartitionProcessorVendor) defines the processor vendor of the platform the partition was created on.

The codes are defined as follows.

typedef enum \_HV\_PROCESSOR\_VENDOR

HvProcessorVendorAmd = 0x0000,

HvProcessorVendorIntel = 0x0001

} HV\_PROCESSOR\_VENDOR, \*PHV\_PROCESSOR\_VENDOR;

### Partition Processor Features

One of the partition properties (HvPartitionProcessorFeatures) defines the processor features that are verified for migration compatibility when restoring a VM[[2]](#footnote-2). This allows a partition’s processor features to be enumerated or de-featured, to enable platform migration.

The property is defined with the following structure:

typedef \_HV\_PARTITION\_PROCESSOR\_FEATURES

{

UINT64 AsUINT64;

struct

{

UINT64 Sse3Support:1;

UINT64 LahfSahfSupport:1;

UINT64 Ssse3Support:1;

UINT64 Sse4\_1Support:1;

UINT64 Sse4\_2Support:1;

UINT64 Sse4aSupport:1;

UINT64 Sse5Support:1;

UINT64 PopCntSupport:1;

UINT64 Cmpxchg16bSupport:1;

UINT64 Altmovcr8Support:1;

UINT64 LzcntSupport:1;

UINT64 MisAlignSseSupport:1;

UINT64 MmxExtSupport:1;

UINT64 Amd3DNowSupport:1;

UINT64 ExtendedAmd3DNowSupport:1;

UINT64 Page1GBSupport:1;

UINT64 Reserved1:47;

};

} HV\_PARTITION\_PROCESSOR\_FEATURES, \*PHV\_PARTITION\_PROCESSOR\_FEATURES;

### Partition Processor XSAVE Features

Similar to HvPartitionProcessorFeatures, HvPartitionProcessorXsaveFeatures defines the processor XSAVE related features that are verified for migration compatibility when restoring a VM processor. This allows a partition’s processor XSAVE features to be enumerated or defeatured, to enable VM migration.

The property is defined with the following structure:

typedef union \_HV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES

{

UINT64 AsUINT64;

struct

{

UINT64 XsaveSupport:1;

UINT64 XsaveoptSupport:1;

UINT64 AvxSupport:1;

UINT64 Reserved1:60;

};

HV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES, \*PHV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES;

### Partition Cache Line Flush Size

This property defines the cache line flush size of the partition. This allows a partition’s processor cache line flush size to be enumerated or specified, to enable VM migration.

This property is a UINT64,

### Partition Compatibility Mode

To allow a VM to migrate between platforms of the same processor architecture but with differing processor feature sets and characteristics, the following defines the partition properties for compatibility VM.

#define HV\_PARTITION\_PROCESSOR\_FEATURES\_INTEL\_COMPATIBILITY\_MODE \

{ 1, /\* Sse3Support \*/ \

1, /\* LahfSahfSupport \*/ \

0, /\* Ssse3Support \*/ \

0, /\* Sse4\_1Support \*/ \

0, /\* Sse4\_2Support \*/ \

0, /\* Sse4aSupport \*/ \

0, /\* Sse5Support \*/ \

0, /\* PopCntSupport \*/ \

1, /\* Cmpxchg16bSupport \*/ \

0, /\* Altmovcr8Support \*/ \

0, /\* LzcntSupport \*/ \

0, /\* MisAlignSseSupport \*/ \

0, /\* MmxExtSupport \*/ \

0, /\* Amd3DNowSupport \*/ \

0, /\* ExtendedAmd3DNowSupport \*/ \

0, /\* Page1GBSupport \*/ \

0 /\* Reserved1 \*/ \

}

#define HV\_PARTITION\_PROCESSOR\_FEATURES\_AMD\_COMPATIBILITY\_MODE \

{ \

1, /\* Sse3Support \*/ \

1, /\* LahfSahfSupport \*/ \

0, /\* Ssse3Support \*/ \

0, /\* Sse4\_1Support \*/ \

0, /\* Sse4\_2Support \*/ \

0, /\* Sse4aSupport \*/ \

0, /\* Sse5Support \*/ \

0, /\* PopCntSupport \*/ \

1, /\* Cmpxchg16bSupport \*/ \

1, /\* Altmovcr8Support \*/ \

0, /\* LzcntSupport \*/ \

0, /\* MisAlignSseSupport \*/ \

1, /\* MmxExtSupport \*/ \

0, /\* Amd3DNowSupport \*/ \

0, /\* ExtendedAmd3DNowSupport \*/ \

0, /\* Page1GBSupport \*/ \

0 /\* Reserved1 \*/ \

}

typedef union \_HV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES

{

struct

{

UINT64 XsaveSupport:1;

UINT64 XsaveoptSupport:1;

UINT64 AvxSupport:1;

UINT64 Reserved1:60;

};

UINT64 AsUINT64;

} HV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES, \*PHV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES;

#define HV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES\_INTEL\_COMPATIBILITY\_MODE \

{ \

0, /\* XsaveSupport \*/ \

0, /\* XsaveoptSupport \*/ \

0, /\* AvxSupport \*/ \

0 /\* Reserved1 \*/ \

}

#define HV\_PARTITION\_PROCESSOR\_XSAVE\_FEATURES\_AMD\_COMPATIBILITY\_MODE \

{ \

0, /\* XsaveSupport \*/ \

0, /\* XsaveoptSupport \*/ \

0, /\* AvxSupport \*/ \

0 /\* Reserved1 \*/ \

}

#define HV\_PARTITION\_PROCESSOR\_CL\_FLUSHSIZE\_INTEL\_COMPATIBILITY\_MODE (8)

#define HV\_PARTITION\_PROCESSOR\_CL\_FLUSHSIZE\_AMD\_COMPATIBILITY\_MODE (8)

## Partition Creation

Partitions creation involves several steps:

HvCreatePartition allocates a new child partition using the minimal amount of memory from the parent’s memory pool. This memory is used to create the basis for a partition that can subsequently be provisioned and initialized. As part of creation, the new partition is assigned a partition ID. This ID is used to identify the partition during all subsequent steps.

Memory is added to the newly-created partition’s memory pool by calling HvDepositMemory.

HvInitializePartition initializes the partition. After successfully initialized, the partition enters the “active” state and additional operations can be performed on the partition.

Additional resources are added to the partition. For example, one or more virtual processors can be allocated, memory can be mapped to the partition’s GPA space, and scheduling policies can be defined.

## Partition Destruction

Partition destruction is complicated by the fact that memory must be reclaimed before the partition disappears. Furthermore, a parent partition might not have the rights to access some of the memory while its child is running, such as the memory used for various internal hypervisor data structures. Destruction is thus a two-part operation, finalization followed by deletion.

### Partition Finalization

Finalization of a partition causes an irreversible change in the partition state. A parent can request this state change for any of its children by calling HvFinalizePartition.

Partition finalization takes an indeterminate amount of time depending on the amount of internal cleanup that is required to bring the partition to the *finalized* state. Cleanup de-allocates the partition’s internal hypervisor data structures — including those structures associated with virtual processors, intercepts, ports, connections, GPA mappings, and memory buffers — and returns them to the pool from which they were allocated. Once the finalization process has begun, very few operations can be performed on the partition.

A finalized partition remains in that state until it is *deleted*. A partition in the finalized state has the following characteristics:

All virtual processors belonging to the partition are completely deleted. A finalized partition is unable to execute any guest instructions.

The partition does not receive messages or interrupts. Attempts to send messages to the partition will fail.

Any hypervisor call that has the partition identifier as one of its parameters will fail, except for HvQueryMemoryBalance, HvWithdrawMemory, HvDeletePartition and HvGetNextChildPartition.

### Partition Deletion

After a partition is finalized, the parent must reclaim the resources associated with the target partition. In general, the parent might not be authorized to perform these actions *before* the partition is finalized. Note that resources associated with descendents of the target partition have been implicitly withdrawn during finalization.

Resources are reclaimed by calling HvWithdrawMemory, which should be called as many times as required to reclaim all resources. After the resources have been reclaimed, the target can be deleted by calling HvDeletePartition.

### Partition Destruction

The full algorithm for destroying a partition is as follows:

Initiate the finalization process by calling HvFinalizePartition for the target partition. This brings the target partition to the finalized state.

Reclaim all resources by calling HvWithdrawMemory for the target partition as many times as necessary.

Delete the partition by calling HvDeletePartition for the target partition.

## Partition Enumeration

The hypervisor provides a hypercall interface to enumerate the partition IDs of children of a particular parent partition. In order for the enumeration process to produce predictable results, the hypervisor will return the partition IDs of each child for a specified parent partition in the order of its creation (that is, the more recently that a partition was created, the later it will be returned by an enumeration process).

## Partition Management Interfaces

### HvCreatePartition

The HvCreatePartition hypercall allows an authorized guest to create a new partition.

**Wrapper Interface**

HV\_STATUS

HvCreatePartition(

\_\_in UINT64 Flags,

\_\_in HV\_PROXIMITY\_DOMAIN\_INFO ProximityDomainInfo,

\_\_out PHV\_PARTITION\_ID NewPartitionId

);

Native Interface

|  |  |
| --- | --- |
| HvCreatePartition | |
|  | Call Code = 0x0040 |
| 🡆 Input Parameters | |
| 0 | Flags (8 bytes) |
| 8 | ProximityDomainInfo (8 bytes) |
| 🡄 Output Parameters | |
| 0 | NewPartitionId (8 bytes) |

Description

Partition creation requires the hypervisor to allocate a new partition data structure. The memory required for the partition’s initial data structure comes from the caller’s memory pool. For information on memory pools, see section .

If a partition is successfully created, it begins in the “uninitialized” state. The new partition’s memory pool is initially empty and the caller must populate it with sufficient pages to allow a subsequent HvInitializePartition hypercall to succeed.

The hypervisor restricts the number maximum depth of the partition hierarchy. All hypervisor implementations must support a depth of at least two partitions (that is, the root partition and one descendent).

Input Parameters

*Flags* is a mask of capabilities that are permanently assigned to the new partition. Creation flags are described in section . They are reserved for future use and should be set to zero.

*ProximityDomainInfo* specifies the ACPI proximity domain information of the NUMA node where the partition’s initial hypervisor data structures will reside. If there are no pages in the caller’s pool for the specified ID, then the call will fail. The Proximity Domain specification is described in section 7.2.1.

Output Parameters

*NewPartitionId* is the identifier for the new partition. This value is guaranteed to be unique within the current power cycle.

Restrictions

The caller must possess the *CreatePartitions* privilege.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition privilege flag CreatePartitions is cleared. |
| HV\_STATUS\_INVALID\_PARAMETER | The *flags* parameter includes bits for capabilities that are either undefined or reserved in the implementation.. |
|  | The *ProximityDomainInfo* parameter specifies an invalid flag bit or an invalid domain ID. |
| HV\_STATUS\_PARTITION\_TOO\_DEEP | Creating a new partition would exceed the maximum depth in the partition hierarchy. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the caller is insufficient to perform the operation. |
| HV\_STATUS\_NO\_RESOURCES | A required system resource is unavailable or an implementation limit has been reached. |

### HvInitializePartition

The HvInitializePartition hypercall allows a parent to initialize a child partition.

**Wrapper Interface**

HV\_STATUS

HvInitializePartition(

\_\_in HV\_PARTITION\_ID PartitionId

);

Native Interface

|  |  |
| --- | --- |
| HvInitializePartition [fast] | |
|  | Call Code = 0x0041 |
| 🡆 Input Parameters | |
| 0 | PartitionId (8 bytes) |

Description

The HvInitializePartition call is used to bring a partition from the “uninitialized” to the “active” state. The child partition must be in the “uninitialized” state; that is, it must have been allocated with a call to HvCreatePartition. Prior to using the HvInitializePartition hypercall, the caller may be required to initially populate the partition’s memory pool (see section ). The internal data structures allocated by this hypercall are allocated preferentially from the proximity domain specified by the caller in the related HvCreatePartition hypercall.

Input Parameters

*PartitionId* is the partition ID of the child partition that is to be initialized.

Output Parameters

None.

Restrictions

The partition specified by *PartitionId* must be in the “uninitialized” state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The specified partition is not a child of the caller. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "uninitialized" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the specified partition is insufficient to perform the operation. |
| HV\_STATUS\_NO\_RESOURCES | A required system resource is unavailable or an implementation limit has been reached. |

### HvFinalizePartition

The HvFinalizePartition hypercall begins the partition destruction operation by freeing all internal hypervisor resources.

**Wrapper Interface**

HV\_STATUS

HvFinalizePartition(

\_\_in HV\_PARTITION\_ID PartitionId

);

Native Interface

|  |  |
| --- | --- |
| HvFinalizePartition [fast] | |
|  | Call Code = 0x0042 |
| 🡆 Input Parameters | |
| 0 | PartitionId (8 bytes) |

Description

This call begins the irreversible destruction process of a partition and its children. It frees internal hypervisor data structures, returning the memory to the partition’s memory pool so the pages can be subsequently withdrawn.

Finalization processing for a partition and its descendents takes an indeterminate amount of time. The architectural state of the partition(s) involved will be in transition until the process completes.

The caller must be the parent of the specified partition. In addition, the partition must be in either the “active” or “finalizing” state. For more information on partition destruction, see section .

Input Parameters

*PartitionId* specifies the partition to be finalized.

Output Parameters

None.

Restrictions

The specified partition must be in the “active” or “finalizing” state.

The caller must be the parent of the specified partition.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | *PartitionId* is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | The specified partition has one or more children. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |

### HvDeletePartition

The HvDeletePartition hypercall completes the partition destruction operation by deleting a finalized or uninitialized partition.

**Wrapper Interface**

HV\_STATUS

HvDeletePartition(

\_\_in HV\_PARTITION\_ID PartitionId

);

Native Interface

|  |  |
| --- | --- |
| HvDeletePartition [fast] | |
|  | Call Code = 0x0043 |
| 🡆 Input Parameters | |
| 0 | PartitionId (8 bytes) |

Description

The caller must be the parent of the specified partition. Partition deletion releases any memory allocated by HvCreatePartition back to the creating partition’s pool. The deleted partition’s resources are returned as described in section .

After a successful deletion, the partition ID is considered invalid.

Input Parameters

*PartitionId* specifies the partition to be deleted.

Output Parameters

None.

Restrictions

The caller must be the parent of the partition specified by *PartitionId.*

The partition specified by *PartitionId* must be in the “finalized” or “uninitialized” state.

The partition’s pool must be empty.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_OPERATION\_DENIED | The specified partition’s pool has not been drained of all pages. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “finalized” or “uninitialized” state. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | *PartitionId* is invalid. |

### HvGetPartitionProperty

The HvGetPartitionProperty hypercall allows an authorized guest to read a property of a partition.

**Wrapper Interface**

HV\_STATUS

HvGetPartitionProperty(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PARTITION\_PROPERTY\_CODE PropertyCode,

\_\_out PHV\_PARTITION\_PROPERTY PropertyValue

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvGetPartitionProperty | | |
|  | Call Code = 0x0044 | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | PropertyCode (4 bytes) | Padding (4 bytes) |
| 🡄 Output Parameters | | |
| 0 | PropertyValue (8 bytes) | |

Description

The caller must be the specified partition’s parent, or the root partition requesting its own properties.

Input Parameters

*PartitionId* specifies a partition.

*PropertyCode* specifies the property the caller is interested in retrieving.

Output Parameters

*PropertyValue* returns the property value.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must either be the parent of the partition specified by *PartitionId* or the root partition specifying its own partition ID.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified target partition (unless the caller and the target partition are both the root partition). |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_UNKNOWN\_PROPERTY | The specified property code is not a recognized property. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvSetPartitionProperty

The HvSetPartitionProperty hypercall allows a parent partition to modify a property of one of its children.

**Wrapper Interface**

HV\_STATUS

HvSetPartitionProperty(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PARTITION\_PROPERTY\_CODE PropertyCode,

\_\_in HV\_PARTITION\_PROPERTY PropertyValue

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvSetPartitionProperty | | |
|  | Call Code = 0x0045 | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | PropertyCode (4 bytes) | Padding (4 bytes) |
| 16 | PropertyValue (8 bytes) | |

Description

The caller must be the specified partition’s parent, or the root partition requesting its own properties.

Input Parameters

*PartitionId* specifies a partition.

*PropertyCode* specifies the property the caller is interested in modifying.

*PropertyValue* specifies the new property value.

Output Parameters

None.

Restrictions

The specified partition must be in the “active” state.

The caller must either be the parent of the partition specified by *PartitionId* or the root partition specifying its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified target partition (unless the caller and the target partition are both the root partition). |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_UNKNOWN\_PROPERTY | The specified property ID is not a recognized property. |
| HV\_STATUS\_PROPERTY\_VALUE\_OUT\_OF\_RANGE | The specified property value, while valid in some circumstances, violates an invariant (for example, a CPU reservation causes the total reservations to exceed 100%). |
| An attempt was made to modify a read-only partition privilege. |
| An attempt was made to set the *CpuPowerManagement* privilege on a hypervisor implementation that reserves this privilege to the root partition. |
| HV\_STATUS\_OBJECT\_IN\_USE | The specified property value, when non-zero, is required to be unique across all partitions but is currently in use. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified property value is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the specified partition is insufficient to perform the operation. |

### HvGetPartitionId

The HvGetPartitionId hypercall returns the partition ID of the caller.

**Wrapper Interface**

HV\_STATUS

HvGetPartitionId(

\_\_out PHV\_PARTITION\_ID PartitionId

);

Native Interface

|  |  |
| --- | --- |
| HvGetPartitionId | |
|  | Call Code = 0x0046 |
| 🡄 Output Parameters | |
| 0 | PartitionId (8 bytes) |

Description

The call returns the partition ID of the caller. Note that the partition ID can change if the partition is saved and subsequently restored or if the partition is migrated across systems. In such cases, the *AccessSelfPartitionId* privilege can be used to prevent a guest from obtaining its partition ID.

Input Parameters

None.

Output Parameters

*PartitionId* provides the partition ID of the caller.

Restrictions

The caller must possess the *AccessPartitionId* privilege.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition privilege flag *AccessSelfPartitionId* is cleared. |

### HvGetNextChildPartition

The HvGetNextChildPartition hypercall allows callers to enumerate the child partitions for a given parent partition.

**Wrapper Interface**

HV\_STATUS

HvGetNextChildPartition(

\_\_in HV\_PARTITION\_ID ParentId,

\_\_in HV\_PARTITION\_ID PreviousChildId,

\_\_out PHV\_PARTITION\_ID NextChildId

);

Native Interface

|  |  |
| --- | --- |
| HvGetNextChildPartition | |
|  | Call Code = 0x0047 |
| 🡆 Input Parameters | |
| 0 | ParentId (8 bytes) |
| 8 | PreviousChildId (8 bytes) |
| 🡄 Output Parameters | |
| 0 | NextChildId (8 bytes) |

Description

The hypercall returns the partition ID of the next child partition created after the one specified. To enumerate all children of a particular parent partition, the caller should specify HV\_PARTITION\_ID\_INVALID as the value for *PreviousChildId*. Subsequent calls should specify the ID of the previously-returned child ID. When the caller has enumerated all partition IDs, the value HV\_PARTITION\_ID\_INVALID will be returned in *NextChildId*. If a child partition is deleted during enumeration, the results are dependent on whether the caller has already enumerated past the deleted partition. Callers should be prepared to handle an HV\_STATUS\_INVALID\_PARTITION\_ID error condition which might indicate that the partition identified by *PreviousChildId* has been deleted.

Partitions are enumerated by the hypervisor in the order of their creation.

Input Parameters

*ParentId* specifies the ID of the parent partition whose children are being enumerated.

*PreviousChildId* specifies the ID of the previous child partition to be enumerated or HV\_PARTITION\_ID\_INVALID to specify that the first child partition should be returned.

Output Parameters

*NextChildId* indicates the next child partition in the list or HV\_PARTITION\_ID\_INVALID if the previous partition was the final child partition.

Restrictions

The partition specified by *ParentId* must be the parent of the partition specified by *PreviousChildId.*.

Return Values

| Status Code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified parent ID is invalid. |
| The specified *PreviousChildId* is not the value HV\_PARTITION\_ID\_INVALID and is not a valid child partition of the partition identified by *ParentId*. |

# Physical Hardware Management

## Overview

This section discusses the way in which the hypervisor manages aspects of the underlying physical hardware.

The hypervisor does not manage or control all hardware in the system. Rather, it manages a small subset of core hardware facilities—just enough to enforce isolation guarantees. In some cases, the hypervisor delegates hardware management responsibilities to the root partition.

The following table shows which aspects of hardware management are handled by the hypervisor, by the root partition, and (optionally) by other partitions.

|  |  |
| --- | --- |
| Software Component | Hardware Facility |
| Hypervisor | Logical processors (scheduling).  Local APICs (interrupt routing).  Constant-rate system counter (for example, PM timer).  System physical address space (access to RAM and device memory).  I/O space.  MSR space. |
| Root partition | Processor power management.  System power management.  Memory and processor hot add and removal.  Device hot add and removal.  PCI, PCIe and PCIx configuration space.  Devices connected directly to a PCIe bus or to a PCIe-attached peripheral bus. |

### System Physical Address Space

The *system physical address space* (or SPA space) of the physical machine is the range from 0 to some maximum address that depends on the underlying hardware architecture—typically 2n-1, where 32 ≤ n ≤ 64.

SPAs may be backed by RAM or memory-mapped device registers at the granularity of the architecturally-defined minimum page size HV\_PAGE\_SIZE. The layout of the SPA space is ultimately dictated by the physical machine.

SPA pages are specified in terms of *SPA page ranges,* each of which consists of one or more contiguous SPA pages. RAM SPA ranges are backed by normal memory.

It is a critical invariant for the correct functioning of the hypervisor that all RAM SPA pages are backed by RAM at all times; that is, read and write accesses to these pages should have the expected semantics. The hypervisor relies on the correct behavior of a privileged partition (typically the root partition) in ensuring that this invariant holds. Future versions of the hypervisor may also rely on hardware features to help enforce this invariant.

The following invariants are assumed (and enforced) by the hypervisor:

No two RAM page ranges can overlap.

Pages that are not within defined RAM page ranges are assumed to be “not backed” or backed by device memory and are not used by the hypervisor. They are inaccessible to all partitions except for the root partition.

The hypervisor uses only pages within RAM SPA page ranges to store its code and internal data structures.

### Logical Processors

The hypervisor maintains a list of *logical processors.* Conceptually, a logical processor is a thread implemented in hardware.

At hypervisor boot time, one virtual processor is created within the root partition for each present and potential logical processor. Unlike normal virtual processors associated with other partitions, the root partition’s virtual processors have hard affinities; that is, they are bound to a specific logical processor. Partitions possessing the *CpuPowerManagement* privilege (see section 5.2.3) also have hard affinities.

### Dynamic Addition of Logical Processors

The hypervisor does not support dynamic addition and removal of logical processors. All potential logical processors must be declared at boot time.

### Physical Nodes

A *physical node* represents a collection of logical processors and system physical pages. A physical node is sometimes referred to as a *proximity domain* because all logical processors and pages within the node are assumed to have similar proximity (that is, similar access times to intra-node hardware resources).

The hypervisor maintains a map between pairs of physical nodes that indicate the “distance” (that is, the cost of accessing resources) between the nodes.

### System Reset

The hypervisor provides an interface for “enlightened reboot”, that is, a mechanism through which a guest may request the hypervisor to reboot the hardware platform. The interface is provided though an MSR that is accessible on all virtual processors of a partition that possesses the *AccessResetMsr* privilege.

## Hardware Information

The hypervisor obtains information about the underlying physical hardware from three sources:

*Boot-time input parameters*: When the hypervisor is booted, input parameters provide information about some aspects of the physical hardware.

*Dynamic discovery*: At run-time, the hypervisor can discover information about the underlying physical hardware by using architecturally-defined mechanisms (for example, on x64 platforms, the CPUID instruction).

*Root partition input*: For more complex dynamic hardware changes (for example, power management and hot add / removal) the hypervisor must be told about hardware changes by code running within the root partition.

### Boot-Time Hardware Properties

Aspects of the hardware that are described to the hypervisor at boot time include:

Present and potential logical processors, including those that may be hot-plugged at run-time

Whether hyperthreading is enabled or disabled in the BIOS

Present RAM SPA ranges—system physical address ranges that are populated with RAM when the hypervisor is booted

Physical nodes (including those that have no associated resources at boot time but may be populated at run-time)

Memory access ratios between physical nodes

Addresses of certain hardware features that the hypervisor must access (for example, the power management timer)

For more information about the hypervisor boot process and boot-time input parameters, refer to Chapter .

### Discovered Hardware Properties

Hardware properties discovered or derived by the hypervisor at run-time include:

The relationship between logical processors (for example, multi-core, hyperthreading)

### Root Partition Hardware Properties

The root partition is responsible for notifying the hypervisor of the following hardware properties:

Power management state changes of logical processors

## Hardware Management Data Types

### Logical Processors

Logical processors are defined by a 32-bit index.

typedef UINT32 HV\_LOGICAL\_PROCESSOR\_INDEX;

### Power States

Each logical processor has an associated power state, which can be changed by the root partition as its policy dictates. The root partition uses power management configuration MSRs to communicate the recipe for power state transitions for each logical processor, and for informing the hypervisor when a power state change is to be triggered.

The hypervisor supports transitions to and from power states C1, C2, and C3.

#### Power State MSRs

Power state transitions are configured by using model-specific registers (MSRs) associated with each logical processor. Each of the power states has an associated pair of MSRs.

| MSR address | Register name | Function | |
| --- | --- | --- | --- |
| 0x400000D1 | HV\_X64\_MSR\_POWER\_STATE\_CONFIG\_C1 | | Configuration register for power state C1 |
| 0x400000C1 | HV\_X64\_MSR\_POWER\_STATE\_TRIGGER\_C1 | | Trigger register for power state C1 |
| 0x400000D2 | HV\_X64\_MSR\_POWER\_STATE\_CONFIG\_C2 | | Configuration register for power state C2 |
| 0x400000C2 | HV\_X64\_MSR\_POWER\_STATE\_TRIGGER\_C2 | | Trigger register for power state C2 |
| 0x400000D3 | HV\_X64\_MSR\_POWER\_STATE\_CONFIG\_C3 | | Configuration register for power state C3 |
| 0x400000C3 | HV\_X64\_MSR\_POWER\_STATE\_TRIGGER\_C3 | | Trigger register for power state C3 |

##### Power State Configuration Register

Power state configuration registers are read/write registers which are used by the power management partition to specify the “recipe” for transitioning to and from the corresponding power state.

| 63:60 | 59 | 58 | 57 | 56 | | 55:52 | | 51:0 |
| --- | --- | --- | --- | --- | --- | --- | --- | --- |
| RsvdZ | SetARB\_DIS | ClearBM\_RST | SetBM\_RST | | CheckBM\_STS | ChangeType | TypeSpecific | |

*SetARB\_DIS* defines the value for the ACPI power management bit ARB\_DIS, which is used to enable and disable the system arbiter.

*ClearBM\_RST* clears the ACPI power management bit BM\_RLD, which determines if the power state was exited as a result of bus master requests.

*SetBM\_RST* sets the ACPI power management bit BM\_RLD, which determines if the power state was exited as a result of bus master requests.

*CheckBM\_STS* determines the value of the ACPI power management bit BM\_STS, which determines the power state to enter when considering a transition to or from the C2/C3 power state..

*ChangeType* defines the power change method to use. Valid options are listed below, and should be specified using the values in HV\_X64\_POWER\_CHANGE\_METHOD.:

*Issue HLT* – execute a HLT instruction

*Read IO then Issue HLT* – read from an I/O port and then issue a HLT instruction

*Read IO* – read from an I/O port

*Issue MWAIT* – issue an MWAIT instruction.

typedef enum \_HV\_X64\_POWER\_CHANGE\_METHOD

{

    HvX64PowerChangeIssueHlt,

    HvX64PowerChangeReadIoThenIssueHlt,

    HvX64PowerChangeReadIo,

    HvX64PowerChangeIssueMwait

} HV\_X64\_POWER\_CHANGE\_METHOD, \*PHV\_X64\_POWER\_CHANGE\_METHOD;

*TypeSpecific* defines the input format specific to the particular type of power change method being requested.

Issue HLT

|  |  |
| --- | --- |
| 63:52 | 51:0 |
| Common input - see above | RsvdZ |

Read I/O

Read I/O then issue HLT

|  |  |  |
| --- | --- | --- |
| 63:52 | 51:16 | 0:15 |
| Common input - see above | RsvdZ | Port |

*Port* designates the I/O port to read from.

Issue MWAIT

|  |  |  |  |
| --- | --- | --- | --- |
| 63:52 | 51:33 | 32 | 31:0 |
| Common input - see above | RsvdZ | BreakOnMaskedInterrupt | Hints |

*BreakOnMaskedInterrupt* specifies that receipt of a masked interrupt should cause a break event.

*Hints* defines the hints to the MWAIT instruction specifying the lower power state to enter.

##### Power State Trigger Register

Power state trigger registers are read-only registers that cause the process to enter the corresponding power state.

| 63:56 | 55:1 | 0 |
| --- | --- | --- |
| IdleEntryCount | RsvdZ | ActiveBM\_STS |

IdleEntryCount contains the number of times that the logical processor went idle since the power management partition took its affinity bound virtual processor idle.

ActiveBM\_STS specifies the value of the ACPI power management bit BM\_STS, which determines the power state to enter when considering a transition to or from the C2/C3 power state.

### Logical Processor Run Time

Associated with each logical processor is a 64-bit quantity called a *run time*. The run time is defined as the number of 100ns time units that have elapsed since the logical processor was last reset (through the APIC) and during which it was actively executing code (either guest code or hypervisor code). When a logical processor has no code to run or enters a low-power or offline state, its run time counter effectively stops.

The following type is used for the run-time value.

typedef UINT64 HV\_NANO100\_TIME;

typedef HV\_NANO100\_TIME \*PHV\_NANO100\_TIME;

### Global Run Time

The hypervisor also maintains a global run time that starts when the hypervisor is booted. As with the logical processor run time, it is recorded in 100ns time units.

### System Reset MSR

A partition that possesses the *AccessSystemResetMsr* privilege has access to the system reset MSR, defined as follows:

#define HV\_X64\_MSR\_RESET 0x40000003

If the MSR is available to the guest, it is accessible from all of the partition’s virtual processors. The register has the following format:

|  |  |
| --- | --- |
| 63:1 | 0 |
| RsvdZ | Reset |

The MSR always reads as zero. When Reset is written as one, the hypervisor will perform the requested reboot operation.

## Hardware Management Interfaces

### HvGetLogicalProcessorRunTime

The HvGetLogicalProcessorRunTime hypercall allows an authorized partition to query the *run time* of the logical processor associated with the caller’s virtual processor.

**Wrapper Interface**

HV\_STATUS

HvGetLogicalProcessorRunTime(

\_\_out PHV\_NANO100\_TIME GlobalTime

\_\_out PHV\_NANO100\_TIME LocalRunTime,

\_\_out PHV\_NANO100\_TIME GroupRunTime

\_\_out PHV\_NANO100\_TIME HypervisorTime

);

Native Interface

|  |  |
| --- | --- |
| HvGetLogicalProcessorRunTime | |
|  | Call Code = 0x0004 |
| 🡄 Output Parameters | |
| 0 | GlobalTime (8 bytes) |
| 8 | LocalRunTime (8 bytes) |
| 16 | RsvdZ (8 bytes) |
| 24 | HypervisorTime (8 bytes) |

Description

The HvGetLogicalProcessorRunTime hypercall requires that the calling partition possess the *CpuPowerManagement* privilege. For more information on this flag and its application, see section 5.2.3.

For a definition of *run time*, see section . Utilization can be computed by call this function twice dividing (*LocalRunTime*n – *LocalRunTime*n-1) by (*GlobalTime*n – *GlobalTime*n-1).

This call can be used to determine whether the hypervisor is actively scheduling activity on the logical processor or whether it is idle and can be placed into a low-power state without negatively impacting the system’s performance.

Input Parameters

None.

Output Parameters

*GlobalTime* provides the current global reference time.

*LocalRunTime* provides the run time of the logical processor associated with the caller’s virtual processor.

*GroupRunTime* provides the run time of the logical processor run time group associated with the caller’s virtual processor. If no such run time group has been defined, the local run time is returned for this parameter.

*HypervisorTime* provides the amount of time the hypervisor has executed on the logical processor.

Restrictions

The partition must possess the *CpuPowerManagement* privilege.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition does not possess the *CpuPowerManagement* privilege. |

### HvParkLogicalProcessors

The HvParkLogicalProcessors hypercall allows an authorized partition to inform the Hypervisor of the set of processors that may be *parked*.

**Wrapper Interface**

HV\_STATUS

HvParkLogicalProcessors(

\_in HV\_INPUT\_PARK\_LOGICAL\_PROCESSORS ProcessorMask

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvParkLogicalProcessors [fast] | | |
|  | Call Code = 0x0009 | |
| 🡆 Input Parameters | | |
| 0 | ProcessorMask (8 bytes) |  |

Description

The HvParkLogicalProcessors hypercall allows an authorized partition to inform the Hypervisor of the set of processors that may be *parked*. A processor that is *parked* may be placed into a low power processor idle sleep state. This scheduling hint allows the hypervisor to attempt to not schedule threads on parked logical processors, increasing energy efficiency.

Input Parameters

ProcessorMask – Specifies set of processors the root partition has decided to park.

Output Parameters

None.

Restrictions

The caller must possess the CpuPowerManagement privilege.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller does not possess the CpuPowerManagement privilege. |

# Resource Management

## Overview

One of the design goals of the hypervisor is to provide availability guarantees to guests. For example, multiple servers that have been consolidated onto a single physical machine should not be able to prevent each other from making progress. It should be possible to run a partition that provides telephony or DVR support such that this partition continues to function irrespective of the potentially adversarial actions of other partitions. This section addresses only the memory management aspects of this problem and only as far as the hypervisor is involved.

In accordance with the overall microkernel architecture of the hypervisor, support for specific resource policy types, such as quotas or reserves, are moved from the hypervisor into guests. The hypervisor must provide minimal support for resource management in connection with several hypercalls that “consume” memory, where “consume” means that the calls may need to allocate memory for internal data structures and that this memory remains allocated beyond the duration of the call.

The resource management state affects, and is affected by, any hypercall that needs to allocate or free memory in the hypervisor. Whether and how a particular call interacts with the resource management state are described as part of the specification for each call. For example, the section on partition creation will specify from whose memory pool the initial resources needed to set up a partition will be deducted.

### Memory Pools

For each partition, the hypervisor maintains a *memory pool* of RAM SPA pages. This pool is managed like a bank account. The number of pages in the pool is referred to as the *balance*. Pages can be *deposited* into or *withdrawn* from the pool.

When a partition makes a hypercall that requires memory, the hypervisor draws the required memory from the pool. If the balance in the pool is insufficient, the call fails. If such a hypercall is made by one guest *on behalf of* another guest (in another partition), the hypervisor draws the required memory from the pool of the latter partition. The phrase “on behalf of” refers to any hypervisor call with a HV\_PARTITION\_ID parameter. Examples of such calls are HvCreateVp and HvMapGpaPages.

Pages within a partition’s memory pool are managed by the hypervisor. These pages cannot be accessed through any partition’s GPA space. That is, in all partitions’ GPA spaces, they must be inaccessible (mapped such that no read, write or execute access is allowed). In general, the only partition that can deposit into or withdraw from a partition is that partition’s parent.

When a partition is created, the memory pool is initially empty. Memory can be deposited by means of the hypercall HvDepositMemory.

The following invariants are maintained:

* All memory pool pages are RAM SPA pages that were previously mapped into the GPA space of the parent partition.
* All pages in a partition’s memory pool must have been explicitly deposited into that partition’s pool; that is, memory does not migrate between memory pools of different partitions.
* Pages in a partition’s memory pool cannot be accessed by any guest; that is, these pages are not mapped into any partition’s GPA space with read, write, or execute privileges.

### NUMA Proximity Domains

The ACPI tables include topology information for all processors and memory present in a system at system boot. They also contain information for memory that can be added while the system is running without requiring a reboot. The information identifies sets of logical processors and physical memory ranges, enabling a tight coupling between logical processors and memory ranges. Each coupling is referred to as a *Proximity Domain.* The latency of memory accesses by logical processors within the same proximity domain is typically shorter than the latency of accesses outside of the proximity domain. The root partition can relay this affinity information to the hypervisor by selecting memory resources or specifying the proximity domain to be used with certain hypercalls.

## Resource Management Data Types

### Proximity Domains

The following data types are used with proximity domains:

// Proximity domain ID is a 32-bit number

typedef UINT32 HV\_PROXIMITY\_DOMAIN\_ID;

// Identifier used when specifying proximity domain information

typedef struct

{

HV\_PROXIMITY\_DOMAIN\_ID Id;

struct

{

UINT32 ProximityPreferred:1;

UINT32 Reserved:30;

UINT32 ProximityInfoValid:1;

} Flags;

} HV\_PROXIMITY\_DOMAIN\_INFO;

*Flags* controls the proximity domain information:

*ProximityInfoValid* indicates the validity of the proximity domain information. If set, the *Id* value and the other flags are valid. Otherwise, a particular proximity domain is not specified and the hypervisor may use any domain.

*ProximityPreferred* indicates that the ID specifies the preferred proximity domain instead of a required domain. If the hypervisor is unable to complete the operation using that specific domain, it is free to use other domains.

*Id* specifies the proximity domain ID. All 32-bit values are supported.

## Resource Management Interfaces

### HvDepositMemory

The HvDepositMemory hypercall allows a partition to deposit memory into a child partition’s memory pool.

**Wrapper Interface**

HV\_STATUS

HvDepositMemory(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_inout PUINT32 PageCount,

\_\_in\_ecount(PageCount)

PCHV\_GPA\_PAGE\_NUMBER GpaPages

);

Native Interface

|  |  |
| --- | --- |
| HvDepositMemory [rep] | |
|  | Call Code = 0x0048 |
| 🡆 Input Parameter Header | |
| 0 | PartitionId (8 bytes) |
| 🡆 Input List Element | |
| 0 | GpaPage (8 bytes) |

Description

The specified pages become exclusively accessible by the hypervisor. The pages must be mapped within the caller’s GPA space with read, write and execute access. If mapped within the GPA space of any other partition, they must be inaccessible. On success, the pages will no longer be accessible to the caller. In addition, future mapping changes to the pages will be prohibited until they have been successfully withdrawn from the pool (using the HvWithdrawMemory hypercall).

The call will fail if a page has already been deposited into a partition’s memory pool. If the pages are being used for another purpose, then an error will be returned to indicate that the pages are currently in use. Such purposes include:

* Pages mapped as event log buffers
* Pages locked down for I/O operations

Input Parameters

*PartitionId* specifies the partition whose memory pool will be credited.

*GpaPage* specifies the GPA page that is to be deposited into the specified partition’s memory pool.

Output Parameters

None.

Restrictions

The caller must possess the *AccessMemoryPool* privilege.

The partition specified by *PartitionId* must be in the “active” or “uninitialized” state.

The caller must either be the parent of the partition specified by *PartitionId* or the root partition specifying its own partition ID.

The caller must possess read, write and execute access to the GPA specified by *GpaPage.*

No other partition may possess read, write or execute access to the GPA specified by *GpaPage.*

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified partition, and the caller’s partition privilege flag *AccessMemoryPool* is enabled.  2. The caller is the root partition, and the specified partition is the root partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specifiedpartition is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | A specified GPA page is not mapped within the caller’s GPA space. |
| A specified GPA page is not read, write and execute accessible within the GPA space of the caller. |
| A specified GPA page is readable, writable or executable within the GPA space of a partition that is not the caller. |
| A specified GPA page has already been deposited into the memory pool of a partition. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” or “uninitialized” state. |
| HV\_STATUS\_OBJECT\_IN\_USE | A specified GPA is being used for another purpose and therefore may not be deposited. |

### HvWithdrawMemory

The HvWithdrawMemory hypercall attempts to remove one or more pages from the memory pool of the specified child partition.

**Wrapper Interface**

HV\_STATUS

HvWithdrawMemory(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PROXIMITY\_DOMAIN\_INFO ProximityDomainInfo,

\_\_inout PUINT32 PageCount,

\_\_out\_ecount(\*PageCount)

PHV\_GPA\_PAGE\_NUMBER GpaPages

);

Native Interface

|  |  |
| --- | --- |
| HvWithdrawMemory [rep] | |
|  | Call Code = 0x0049 |
| 🡆 Input Parameters | |
| 0 | PartitionId (8 bytes) |
| 8 | ProximityDomainInfo (8 bytes) |
| 🡄 Output List Element | |
| 0 | GpaPage (8 bytes) |

Description

A page can be withdrawn only if it is not currently in use by the hypervisor. The hypervisor guarantees that the contents of the withdrawn page are zeroed. The GPA for the withdrawn page is the same as the GPA of a page that was previously deposited. On success, the page will be read, write and execute accessible.

If a partition is in the “uninitialized” or “finalized” state, this call is guaranteed to allow withdrawal of all pages within the partition’s pool. In other cases, success is not guaranteed – even if a recent call to HvGetMemoryBalance has indicated the presence of available pages – because the hypervisor may choose to use deposited pages at any time.

Input Parameters

*PartitionId* specifies the partition whose memory pool will be debited.

*ProximityDomainInfo* specifies the ACPI proximity domain information of the NUMA node from which the pages are to be withdrawn. The Proximity Domain specifier is described in section 7.2.1.

Output Parameters

*GpaPage* returns a page that is has been withdrawn from the memory pool.

Restrictions

The caller must possess the *AccessMemoryPool* privilege.

The caller must be the parent of the partition specified by *PartitionId* or the root partition specifying its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified partition, and the caller’s partition privilege flag *AccessMemoryPool* is enabled.  2. The caller is the root partition, and the specified partition is the root partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specifiedpartition is invalid. |
| HV\_STATUS\_INVALID\_PARAMETER | The *ProximityDomainInfo* parameter specifies an invalid flag bit or an invalid domain ID. |
| HV\_STATUS\_NO\_RESOURCES | No unused page was available. |

### HvGetMemoryBalance

The HvGetMemoryBalance hypercall allows a parent partition to query the state of the memory pool of a child partition.

**Wrapper Interface**

HV\_STATUS

HvGetMemoryBalance(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PROXIMITY\_DOMAIN\_INFO ProximityDomainInfo,

\_\_out PUINT64 PagesAvailable,

\_\_out PUINT64 PagesInUse

);

Native Interface

|  |  |
| --- | --- |
| HvGetMemoryBalance | |
|  | Call Code = 0x004A |
| 🡆 Input Parameters | |
| 0 | PartitionId (8 bytes) |
| 8 | ProximityDomainInfo (8 bytes) |
| 🡄 Output Parameters | |
| 0 | PagesAvailable (8 bytes) |
| 8 | PagesInUse (8 bytes) |

Description

The caller must either be the root partition or the parent of the partition specified as *PartitionId.*

Input Parameters

*PartitionId* specifies the partition.

*ProximityDomainInfo* specifies the ACPI proximity domain information of the NUMA node whose memory balance is being queried. The Proximity Domain specifier is described in section .

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified partition, and the caller’s partition privilege flag AccessMemoryPool is enabled.  2. The caller is the root partition, and the specified partition is the root partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specifiedpartition is invalid. |
| HV\_STATUS\_INVALID\_PARAMETER | The *ProximityDomainInfo* parameter specifies an invalid flag bit or an invalid domain ID. |

# Guest Physical Address Spaces

## Overview

### GPA Space

The size of the GPA space for a partition is the range from 0 to some maximum address that depends on architectural attributes of the virtual machine exposed by the partition.

Each page within a GPA space is in one of three states:

*Mapped*: A mapped GPA page is associated with a RAM SPA page.

*Inaccessible:* An inaccessible GPA page may not be read, written, or executed by the partition.

*Unmapped*: An unmapped GPA page is not associated with a RAM SPA page.

The way in which the GPA space is defined and the behavior associated with unmapped page accesses differs between the root partition and non-root partitions.

For the root partition:

Its GPA space is identity-mapped—that is, all mapped GPA pages map directly to the SPA page with the same address.

The GPA mappings are defined by the hypervisor at boot time or when SPA RAM ranges are added or removed. All valid SPA RAM pages are therefore always mapped within the root partition’s GPA space. The root partition is not allowed to unmap these. The root partition may, however, modify its access rights to its mapped pages.

Unmapped GPA pages within the root partition’s GPA space (that is, SPA pages that are not part of known SPA RAM ranges) can be accessed by the root partition. This allows the partition to access memory-mapped registers, video memory buffers and so on. Some unmapped pages are off limits even to the root partition. These include SPA pages that correspond to hardware resources that must be managed by the hypervisor for correct operation. For example, the hypervisor does not allow the root partition to directly access the local APIC’s memory-mapped registers.

For non-root partitions:

Its GPA mappings are not necessarily identity-mapped. That is, a GPA does not necessarily refer to the same SPA.

The GPA mappings are defined by the partition’s parent. At the time they are mapped (through a call to HvMapGpaPages), they are specified in terms of the parent’s GPA space. Therefore, these pages must be mapped into the parent’s GPA space; however, the parent is not required to have read, write or execute access to these mapped pages.

When a virtual processor accesses an unmapped GPA page, the hypervisor suspends the virtual processor and sends a message to the partition’s parent. Code within the parent will typically respond by creating a mapping or by emulating the instruction that generated the memory access. In either case, it is up to the software in the parent partition to “unsuspend” the child’s virtual processor. For a detailed description of the memory access messages, see section .

### Page Access Rights

Mapped GPA pages have the following attributes which define the access rights of the partition:

*Readable*: Data on the page can be read.

*Writeable*: Data to the page can be written.

*Executable*: Code on the page can be executed.

These access rights are enforced for explicit accesses performed by the child’s virtual processors. They are also enforced for implicit reads or writes performed by the hypervisor (for example, due to guest page table flag updates).

Access right combinations are limited by the underlying hardware. The following table shows the valid combinations for an x64 system.

| Access Type | | | Description |
| --- | --- | --- | --- |
| Read | Write | Exec |
| • | • | • | Instruction fetches, reads, and writes are allowed |
|  | • | • | Illegal combination |
| • |  | • | Instruction fetches and reads are allowed |
|  |  | • | Illegal combination |
| • | • |  | Reads and writes are allowed |
|  | • |  | Illegal combination |
| • |  |  | Reads are allowed |
|  |  |  | No access is allowed |

If an attempted memory access is not permitted according to the access rights, the virtual processor that performed the access is suspended (on an instruction boundary) and a message is sent to the parent partition. Code within the parent will typically respond by adjusting the access rights to allow the access or emulating the instruction that performed the memory access. In either case, it is up to the software in the parent partition to “unsuspend” the child’s virtual processor. For a detailed description of the memory access messages, see section .

Memory accesses that cross page boundaries are handled in a manner that is consistent with the underlying processor architecture. For x64, this means the entire access is validated before any data exchange occurs. For example, if a four-byte write is split across two pages and the first page is writable but the second is not, the first two bytes are not written.

### GPA Overlay Pages

The hypervisor defines several special pages that “overlay” the guest’s GPA space. The hypercall code page is an example of an overlay page. Overlays are addressed by guest physical addresses but are not included in the normal GPA map maintained internally by the hypervisor. Conceptually, they exist in a separate map that overlays the GPA map.

If a page within the GPA space is overlaid, any SPA page mapped to the GPA page is effectively “obscured” and generally unreachable by the virtual processor through processor memory accesses. Furthermore, access rights installed on the underlying GPA page are not honored when accessing an overlay page.

If an overlay page is disabled or is moved to a new location in the GPA space, the underlying GPA page is “uncovered”, and an existing mapping becomes accessible to the guest.

If multiple overlay pages are programmed to appear on top of each other (for example, the guest programs the APIC to appear on top of the hypercall page), the hypervisor will choose an ordering (which is undefined) and only one of these overlays will be visible to code running within the partition. In such cases, if the “top-most” overlay is disabled or moved, another overlay page will become visible.

Parent partitions that include instruction completion logic should use the hypercalls HvTranslateVirtualAddress, HvReadGpa, and HvWriteGpa to emulate virtual processor memory accesses correctly in the presence of overlays.

HvTranslateVirtualAddress returns a flag indicating whether the specified virtual address maps to an overlay page. HvReadGpa and HvWriteGpa perform the GPA access in the same way the specified virtual processor would have. If an overlay page is present at the specified address, the access is directed to that overlay page. Otherwise, the access is directed to the underlying GPA page.

When the hypervisor performs a guest page table walk either in response to a virtual processor memory access or a call to HvTranslateVirtualAddress, it might find that a page table is located on a GPA location associated with an overlay page. In this case, the hypervisor may choose to do any one of the following: generate a guest page fault, reference the contents of the overlay page, or reference the contents of the underlying GPA mapping. Because this behavior can vary from one hypervisor implementation to the next, it is strongly recommended that guests avoid this situation.

## GPA Data Types

### Map Page Flags

The following flags are used with the HvMapGpaPages hypercall.

typedef UINT32 HV\_MAP\_GPA\_FLAGS;

#define HV\_MAP\_GPA\_READABLE 0x00000001

#define HV\_MAP\_GPA\_WRITABLE 0x00000002

#define HV\_MAP\_GPA\_EXECUTABLE 0x00000004

## GPA Interfaces

### HvMapGpaPages

The HvMapGpaPages hypercall maps one or more GPA pages within the caller’s GPA space to pages within the target partition’s GPA space.

**Wrapper Interface**

HV\_STATUS

HvMapGpaPages(

\_\_in HV\_PARTITION\_ID TargetPartitionId,

\_\_in HV\_GPA\_PAGE\_NUMBER TargetGpaBase,

\_\_in HV\_MAP\_GPA\_FLAGS MapFlags,

\_\_inout PUINT32 PageCount,

\_\_in\_ecount(PageCount)

PCHV\_GPA\_PAGE\_NUMBER SourceGpaPageList

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvMapGpaPages [rep] | | |
|  | Call Code = 0x004B | |
| 🡆 Input Parameter Header | | |
| 0 | TargetPartitionId (8 bytes) | |
| 8 | TargetGpaBase (8 bytes) | |
| 16 | MapFlags (4 bytes) | Padding (4 bytes) |
| 🡆 Input List Element | | |
| 0 | SourceGpaPageList (8 bytes) | |

Description

HvMapGpaPages is a rep call. The *PageCount* parameter to the C wrapper is passed as the rep count to the hypercall. See section for details.

The mapping, if successful, supersedes any previous mapping to the affected GPA pages. Likewise, the specified access rights (readable, writeable and executable) supersede any previously-installed access rights on the GPA pages.

If an existing mapping is being superseded and the pages are being used for another purpose that cannot allow the mapping to be changed, then an error will be returned to indicate that the pages are currently in use. Such purposes include:

Pages mapped as event log buffers

Pages deposited into memory pools

Pages locked down for I/O operations

The GPA pages within the caller’s partition must be mapped and not deposited into any partition’s memory pool.

A parent partition is allowed to map its GPA pages into a child’s GPA space multiple times. It is also allowed to map its GPA pages into the GPA space of multiple children. This allows it to create shared areas between itself and one or more of its children

The root partition is special in that it can call HvMapGpaPages on itself. However, it can modify only the access rights to its GPA pages. It cannot create arbitrary mappings within its own GPA space. Its GPA map is defined by the hypervisor at boot time and at the time SPA RAM ranges are added or removed. Root GPA pages are identity-mapped to SPA pages. Although the root partition can modify the access rights to its own GPA mappings, it cannot establish non-identity mappings.

For best performance on x64 systems, callers should attempt to map at least 2 MB of consecutive GPA pages and make sure that these pages have the same alignment relative to a 2‑MB boundary. This will potentially enable the hypervisor to use “large page” mappings and in turn reduce the pressure on physical TLBs. Note that this mapping doesn’t need to be done in a single call, and in fact cannot be done in a single call due to the size constraints on the input parameter list.

Input Parameters

*TargetPartitionId* specifies the target partition whose GPA space is being modified.

*TargetGpaBase* specifies the base GPA page number in the GPA space of the target partition.

*MapFlags* specifies the access rights associated with the mapping. Valid access rights flags include HV\_MAP\_GPA\_READABLE, HV\_MAP\_GPA\_WRITABLE, and HV\_MAP\_GPA\_EXECUTABLE. For a description of the valid combinations, see section .

*SourceGpaPage* specifies a GPA page within the caller’s GPA space.

Output Parameters

None.

Restrictions

The partition specified by *TargetPartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId* or the root partition specifying its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified target partition (unless the caller and the target partition are both the root partition). |
|  | The target partition is the root partition and one of the following is true:  1. The target GPA base is not equal to the first source GPA in the list.  2. The source GPA list doesn’t consist of contiguous, incrementing pages.  3. The target GPA range references an inaccessible GPA page. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified target partition ID is invalid. |
| HV\_STATUS\_INVALID\_PARAMETER | A target GPA page lies outside the GPA address space of the target partition. |
|  | A reserved or invalid bit within the specified map flags is set. |
|  | An invalid combination of access rights is specified. |
|  | A source GPA page lies outside the GPA address space of the caller’s partition. |
| HV\_STATUS\_OPERATION\_DENIED | A source GPA page is unmapped. |
|  | A source GPA page has been deposited into a partition’s memory pool and is inaccessible. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the partition’s memory pool is insufficient to perform the operation. |
| HV\_STATUS\_OBJECT\_IN\_USE | An attempt was made to change the mapping of one or more pages that are being used for a purpose that is incompatible with the change is not permitted. |

### HvUnmapGpaPages

The HvUnmapGpaPages hypercall unmaps one or more GPA pages from a child partition.

**Wrapper Interface**

HV\_STATUS

HvUnmapGpaPages(

\_\_in HV\_PARTITION\_ID TargetPartitionId,

\_\_in HV\_GPA\_PAGE\_NUMBER TargetGpaBase,

\_\_inout PUINT32 PageCount

);

Native Interface

|  |  |
| --- | --- |
| HvUnmapGpaPages [rep] | |
|  | Call Code = 0x004C |
| 🡆 Input Parameter Header | |
| 0 | TargetPartitionId (8 bytes) |
| 8 | TargetGpaBase (8 bytes) |

Description

Note that this call is a “rep” hypercall despite the fact that its parameters do not include an input or output list. The *PageCount* parameter to the C wrapper is passed as the rep count to the hypercall. See section for details.

An attempt to unmap an already-unmapped page will be ignored, and a successful status will be returned. Pages deposited to pools are inaccessible and cannot be unmapped until they are first withdrawn (using the HvWithdrawMemory hypercall).

If any of the pages are being used for another purpose that cannot allow the unmapping to be performed, then an error will be returned to indicate that the pages are currently in use. Such purposes include:

* Pages mapped as event log buffers
* Pages deposited into memory pools
* Pages locked down for I/O operations

Unlike HvMapGpaPages, this hypercall cannot be called by the root partition to act upon itself.

Call Type

Rep hypercall

Input Parameters

*TargetPartitionId* specifies the partition whose GPA space is being modified.

*TargetGpaBase* specifies the base GPA page number of the GPA range being unmapped.

Output Parameters

None.

Restrictions

The partition specified by *TargetPartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *TargetPartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| The caller is the root partition attempting to unmap pages from its own GPA space. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified target partition ID is invalid. |
| HV\_STATUS\_INVALID\_PARAMETER | A specified GPA page lies outside the GPA address space of the specified target partition. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |
| HV\_STATUS\_OBJECT\_IN\_USE | An attempt was made to unmap one or more pages that are being used for a purpose that requires them is not permitted. |

### HvMapSparseGpaPages

The HvMapSparseGpaPages hypercall maps one or more GPA pages within the caller’s GPA space to pages within the target partition’s GPA space.

**Wrapper Interface**

typedef struct \_HV\_GPA\_MAPPING

{

HV\_GPA\_PAGE\_NUMBER TargetGpaPageNumber;

HV\_GPA\_PAGE\_NUMBER SourceGpaPageNumber;

} HV\_GPA\_MAPPING, \*PHV\_GPA\_MAPPING;

HV\_STATUS

HvMapSparseGpaPages(

\_\_in HV\_PARTITION\_ID TargetPartitionId,

\_\_in HV\_MAP\_GPA\_FLAGS MapFlags,

\_\_inout PUINT32 PageCount,

\_\_in\_ecount(PageCount)

PHV\_GPA\_MAPPING PageList

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvMapSparseGpaPages [rep] | | |
|  | Call Code = 0x006E | |
| 🡆 Input Parameter Header | | |
| 0 | TargetPartitionId (8 bytes) | |
| 8 | MapFlags (4 bytes) | Padding (4 bytes) |
| 🡆 Input List Element | | |
| 0 | PageList (16 bytes) | |

Description

HvMapSparseGpaPages is a rep call. The *PageCount* parameter to the C wrapper is passed as the rep count to the hypercall. See section for details.

The mapping, if successful, supersedes any previous mapping to the affected GPA pages. Likewise, the specified access rights (readable, writeable and executable) supersede any previously-installed access rights on the GPA pages.

If an existing mapping is being superseded and the pages are being used for another purpose that cannot allow the mapping to be changed, then an error will be returned to indicate that the pages are currently in use. Such purposes include:

* Pages mapped as event log buffers
* Pages deposited into memory pools
* Pages locked down for I/O operations

The GPA pages within the caller’s partition must be mapped and not deposited into any partition’s memory pool.

A parent partition is allowed to map its GPA pages into a child’s GPA space multiple times. It is also allowed to map its GPA pages into the GPA space of multiple children. This allows it to create shared areas between itself and one or more of its children

The root partition is special in that it can call HvMapSparseGpaPages on itself. However, it can modify only the access rights to its GPA pages. It cannot create arbitrary mappings within its own GPA space. Its GPA map is defined by the hypervisor at boot time and at the time SPA RAM ranges are added or removed. Root GPA pages are identity-mapped to SPA pages. Although the root partition can modify the access rights to its own GPA mappings, it cannot establish non-identity mappings.

For best performance on x64 systems, callers should attempt to map at least 2 MB of consecutive GPA pages and make sure that these pages have the same alignment relative to a 2‑MB boundary. This will potentially enable the hypervisor to use “large page” mappings and in turn reduce the pressure on physical TLBs. Note that this mapping doesn’t need to be done in a single call, and in fact cannot be done in a single call due to the size constraints on the input parameter list.

Input Parameters

*TargetPartitionId* specifies the target partition whose GPA space is being modified.

*MapFlags* specifies the access rights associated with the mapping. Valid access rights flags include HV\_MAP\_GPA\_READABLE, HV\_MAP\_GPA\_WRITABLE, and HV\_MAP\_GPA\_EXECUTABLE. For a description of the valid combinations, see section .

*PageList* specifies pair of GPA page within the target GPA space and source GPA page within the caller’s GPA space.

Output Parameters

None.

Restrictions

The partition specified by *TargetPartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId* or the root partition specifying its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified target partition (unless the caller and the target partition are both the root partition). |
| The target partition is the root partition and one of the following is true:  1. The target GPA base is not equal to the first source GPA in the list.  2. The source GPA list doesn’t consist of contiguous, incrementing pages.  3. The target GPA range references an inaccessible GPA page. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified target partition ID is invalid. |
| HV\_STATUS\_INVALID\_PARAMETER | A target GPA page lies outside the GPA address space of the target partition. |
| A reserved or invalid bit within the specified map flags is set. |
| An invalid combination of access rights is specified. |
| A source GPA page lies outside the GPA address space of the caller’s partition. |
| HV\_STATUS\_OPERATION\_DENIED | A source GPA page is unmapped. |
| A source GPA page has been deposited into a partition’s memory pool and is inaccessible. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the partition’s memory pool is insufficient to perform the operation. |
| HV\_STATUS\_OBJECT\_IN\_USE | An attempt was made to change the mapping of one or more pages that are being used for a purpose that is incompatible with the change is not permitted. |

# Intercepts

## Overview

### Programmable Intercept Types

This section describes the principal mechanism the hypervisor provides to facilitate the virtualization of certain guest events. These events occur when a virtual processor executes certain instructions or generates certain exceptions. An authorized guest (a parent partition) can install an *intercept* for certain events on another guest (a child partition). An intercept involves the detection of an event performed by a virtual processor (explicitly or implicitly). When an intercepted event occurs in the child partition, the virtual processor that triggered the event is suspended, and an *intercept message* is sent to the parent. The virtual processor remains suspended until the parent explicitly clears the virtual processor register HvRegisterInterceptSuspend by calling HvSetVpRegisters.

In general, the register state of the virtual processor when it is suspended corresponds to the state before the execution of the instruction that triggered the intercept. As such, the instruction can be restarted.

The purpose of this mechanism is to allow a virtualization-aware parent to create a virtual environment that allows an unmodified legacy guest—that was written to execute on the physical hardware—to execute in a hypervisor partition. Such legacy guests may attempt to access physical devices that do not exist in a hypervisor partition (for example, by accessing certain I/O ports). The mechanism described in this section makes it possible to intercept all such accesses and transfer control to the parent partition. The parent partition can alter the effect of the intercepted instruction such that, to the child, it mirrors the expected behavior in physical hardware.

An intercept only affects the state of a single virtual processor. Other virtual processors within the same partition continue to run. Therefore, it’s possible that multiple intercept messages can be “in progress” concurrently. Intercept messages are queued to the parent in the order in which they are detected.

The available processor intercept events depend on the (virtual) processor architecture and the capabilities of the physical hardware’s virtualization facilities.

The following types of processor events can be intercepted on x64 platforms:

Accesses to I/O Ports

Accesses to MSRs

Execution of the CPUID instruction

Exceptions

The following table describes the scope and intercept access flags that are allowed for each intercept type:

| Intercept Type | Intercept Applies To | Valid Access Flags |
| --- | --- | --- |
| I/O port access  (see section 11.9) | A specific I/O port. The I/O port is specified with each hypercall. | Read *and* Write access flags must be specified to install an intercept. |
| MSR access  (see section ) | All MSRs not being virtualized by the hypervisor. Note that certain privileges affect MSR virtualization. No MSR value is specified with the hypercall. | Read *and* Write access flags must be specified to install the intercept. |
| CPUID instruction execution  (see section 11.11) | A specific CPUID leaf. The CPUID leaf is specified with each hypercall. | Execute access flag must be specified to install an intercept. |
| Exceptions  (see section 11.12) | A specific exception vector. The exception vector is specified with each hypercall. | Execute access flag must be specified to install an intercept. |

### Unsolicited Intercept Types

Several conditions always cause a virtual processor to be suspended and an intercept message to be sent to the parent. If certain events occur within the root partition, which has no parent, the condition is considered fatal, and the system is restarted.

The following table describes the intercepts defined on x64 implementations of the hypervisor.

| Unsolicited Intercept Type | Root Partition Behavior |
| --- | --- |
| Invalid virtual processor state (see section 10.1.2) | Fatal error. The system is restarted. |
| Unrecoverable processor exception (for example, triple fault on X64) | Fatal error. The system is restarted. |
| Unsupported functionality error | Fatal error. The system is restarted. |
| FERR asserted (legacy floating point error) | Ignored. No message is generated. |
| APIC EOIs | Ignored. No message is generated. |

Unsupported functionality errors are delivered to the parent if the guest uses a feature of the underlying processor architecture that is not virtualized by the hypervisor and cannot otherwise be reported as “not implemented”. For example, on the x64 architecture, some features can be reported as “not implemented” by using CPUID feature bits or by generating a #GP fault when accessing an MSR. If there is no architectural way for a guest to determine whether a feature is supported, the hypervisor may detect the use of the unsupported feature and deliver an “unsupported functionality” error to the parent.

## Intercept Data Types

### Intercept Types

This data type enumerates the intercept classes the hypervisor recognizes.

typedef enum

{

// Platform-specific intercept types

HvInterceptTypeX64IoPort = 0x00000000,

HvInterceptTypeX64Msr = 0x00000001,

HvInterceptTypeX64Cpuid = 0x00000002,

HvInterceptTypeX64Exception = 0x00000003

} HV\_INTERCEPT\_TYPE;

### Intercept Parameters

The following data structures are used to describe information about installed intercepts. Reserved fields must be set to zero.

// Used with intercepts of type HV\_INTERCEPT\_TYPE\_X64\_IO\_PORT

typedef UINT16 HV\_X64\_IO\_PORT;

typedef union

{

// Ensure that the parameter is 8 bytes wide

UINT64 AsUINT64;

// HvInterceptTypeX64IoPort

HV\_X64\_IO\_PORT IoPort;

// HvInterceptTypeX64Msr has no parameters

// HvInterceptTypeX64Cpuid

UINT32 CpuidIndex;

// HvInterceptTypeX64Exception

UINT16 ExceptionVector;} HV\_INTERCEPT\_PARAMETERS;

typedef struct

{

HV\_INTERCEPT\_TYPE Type;

HV\_INTERCEPT\_PARAMETERS Parameters;

} HV\_INTERCEPT\_DESCRIPTOR;

### Intercept Access Types

This data type is used to specify different access types that can trigger intercepts. A corresponding mask can be used to specify combinations of accesses (for example, reads and writes).

typedef UINT8 HV\_INTERCEPT\_ACCESS\_TYPE\_MASK;

#define HV\_INTERCEPT\_ACCESS\_MASK\_NONE 0

#define HV\_INTERCEPT\_ACCESS\_MASK\_READ 1

#define HV\_INTERCEPT\_ACCESS\_MASK\_WRITE 2

#define HV\_INTERCEPT\_ACCESS\_MASK\_EXECUTE 4

### Unsupported Feature Codes

The hypervisor reports an attempt to access an “unsupported feature” using an intercept message. The message field *FeatureCode* is used to indicate the specific feature that the guest attempted to use. The codes are defined as follows:

typedef enum

{

HvUnsupportedFeatureIntercept = 1,

HvUnsupportedFeatureTaskSwitchTss = 2

} HV\_UNSUPPORTED\_FEATURE\_CODE;

## Intercept Interfaces

### HvInstallIntercept

The HvInstallIntercept hypercall enables or disables intercepts triggered by the specified child partition’s virtual processors.

**Wrapper Interface**

HV\_STATUS

HvInstallIntercept(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_INTERCEPT\_ACCESS\_TYPE\_MASK AccessType,

\_\_in PCHV\_INTERCEPT\_DESCRIPTOR Descriptor

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvInstallIntercept | | |
|  | Call Code = 0x004D | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | AccessType (4 bytes) | InterceptType (4 bytes) |
| 16 | InterceptParameter (8 bytes) | |

Description

Intercepts are installed by specifying the appropriate access type as described in section 9.1.1. To clear intercepts, the caller should specify an access type of HV\_INTERCEPT\_ACCESS\_MASK\_NONE.

In general, calls to HvInstallIntercept can fail if the hypervisor has insufficient memory required to allocate the requisite internal data structures. However, some intercept types do not require dynamic memory allocation and are guaranteed to never fail with an insufficient-memory error.

| Intercept Type | Can Fail Due to Insufficient Memory? |
| --- | --- |
| I/O port access | No |
| MSR access | No |
| CPUID instruction execution | Yes |
| Exceptions | No |

Input Parameters

*PartitionId* specifies a child partition.

*AccessType* specifies the access type(s) that should be intercepted. An access type of “none” indicates that the intercept should be removed. Other combinations of access type flags are permitted depending on the intercept type.

*InterceptType* specifies the type of intercept being installed.

*InterceptParameter* provides a parameter whose meaning is specific to the intercept type.

Output Parameters

None.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified intercept type is invalid. |
|  | The specified access type is not permitted for the given intercept type. (For valid access types, see the previous table.) |
|  | The specified intercept parameters are invalid or out of range for the given intercept type. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the partition’s memory pool is insufficient to perform the operation. |

## Intercept Messages and Message Formats

When an intercepted event occurs in a child partition, the hypervisor informs the parent partition by using a message. For a detailed description of the messages sent by the hypervisor when a processor intercept is triggered, see chapter .

# 

# Virtual Processor Management

## Overview

Each partition may have zero or more virtual processors. This section specifies how the state of a virtual processor can be programmatically accessed or modified by another sufficiently privileged partition by using hypercalls. This facility can be used in conjunction with intercepts to virtualize the processor.

### Virtual Processor Indices

A virtual processor is identified by a tuple composed of its partition ID and its processor index. The processor index is assigned to the virtual processor when it is created, and it is unchanged through the lifetime of the virtual processor.

### Virtual Processor Registers

Associated with each virtual processor is a variety of state modeled as processor registers. Most of this state is defined by the underlying processor architecture and consists of architected register values. The hypervisor provides a mechanism for reading and writing these registers through hypercalls HvGetVpRegisters and HvSetVpRegisters.

If a virtual processor register is modified and the newly-specified value is invalid in some way, the hypervisor may or may not immediately return an error. In some cases, a value is invalid only in certain contexts (for example, if a bit within another virtual processor register is set). Therefore, some invalid register values are not detected until the virtual processor resumes execution. In such a case, the virtual processor is suspended, and an intercept message (with a message type HvMessageTypeInvalidVpRegisterValue) is sent to its parent partition.

### Virtual Processor States

Conceptually, a virtual processor is in one of four states:

*Running*—actively consuming processor cycles of a logical processor

*Ready—*ready to run, but not actively running because other virtual processors are running

*Waiting—*in a state defined by the processor architecture that does not involve the active execution of instructions (for example, for the x64 architecture, at a HLT instruction, within “waiting for SIPI” state or if the scheduler has capped the virtual processor)

*Suspended—*stopped on a guest instruction boundary either explicitly suspended or implicitly suspended due to an intercept.

A suspended virtual processor consumes no processor cycles. It is stopped on a guest instruction boundary. A virtual processor is suspended *when any* of the following are true:

The virtual processor is explicitly suspended

The virtual processor is implicitly suspended due to an intercept

Both suspension reasons must be cleared before a virtual processor becomes active. They can be set or cleared by using HvSetVpRegisters.

### Virtual Processor Idle Sleep State

Virtual processors may be placed in a virtual idle processor power state, or processor sleep state. This enhanced virtual idle state allows a virtual processor that is placed in to a low power idle state to be woken with the arrival of an interrupt even when interrupt is masked on the virtual processor. The virtual idle state allows the operating system in the guest partition to take advantage of processor power saving techniques in the OS that would otherwise be unavailable when running in a guest partition.

A partition which possesses the AccessGuestIdleMsr privilege (refer to section 5.2.3) may trigger entry into the virtual processor idle sleep state through a read to the hypervisor-defined MSR HV\_X64\_MSR\_GUEST\_IDLE. The virtual processor will be woken when an interrupt arrives, regardless of if the interrupt is enabled on the virtual processor or not.

### Virtual Boot Processor

The virtual processor created with the index of zero is the *virtual* *boot processor* for the partition that it is related to*.* It will be the only virtual processor with the BSP flag set in the IA32\_APIC\_BASE\_MSR register. Virtual processors created with non-zero indices are *virtual application processors.* Both the virtual boot processor and virtual application processors may be created or deleted at any time.

### Virtual Processor APIC IDs

On a real x64 system, each processor starts with a hard-coded *initial APIC ID*. This value can be retrieved through the CPUID instruction. On some systems, the initial APIC ID is not necessarily unique across all processors, so the APIC ID accessible through the APIC’s MMIO space can be modified. This allows software to allocate unique APIC IDs for all processors within the system.

Virtual processors also have an initial APIC ID value that cannot be modified by software running within the partition. This value is initialized at virtual processor creation time to the virtual processor’s index. The value can be modified by the parent partition by calling HvSetVpRegisters and specifying the register HvX64RegisterInitialApicId.

Virtual processors also have a guest-programmable APIC ID that is accessible through the APIC’s MMIO space. This value is also initialized based on the virtual processor’s index.

## Virtual Processor Data Types

### Virtual Processor Index

Virtual processors are identified by using an index (VP index). The maximum number of virtual processors per partition supported by the current implementation of the hypervisor can be obtained through CPUID leaf 0x40000005. A virtual processor index must be less than the maximum number of virtual processors per partition.

A special value HV\_ANY\_VP can be used in certain situations to specify “any virtual processor”.

typedef UINT32 HV\_VP\_INDEX;

#define HV\_ANY\_VP 0xFFFFFFFF

A virtual processor’s ID can be retrieved by the guest through a hypervisor-defined MSR (model-specific register) HV\_X64\_MSR\_VP\_INDEX.

#define HV\_X64\_MSR\_VP\_INDEX 0x40000002

### Virtual Processor Register Names

Virtual processor state is referenced by *register names*, 32-bit numbers that uniquely identify a register.

typedef enum

{

// Suspend Registers

HvRegisterExplicitSuspend = 0x00000000,

HvRegisterInterceptSuspend = 0x00000001,

// Interrupt Registers

HvX64RegisterPendingInterruption = 0x00010002,

HvX64RegisterInterruptState = 0x00010003,

// User-Mode Registers

HvX64RegisterRax = 0x00020000,

HvX64RegisterRcx = 0x00020001,

HvX64RegisterRdx = 0x00020002,

HvX64RegisterRbx = 0x00020003,

HvX64RegisterRsp = 0x00020004,

HvX64RegisterRbp = 0x00020005,

HvX64RegisterRsi = 0x00020006,

HvX64RegisterRdi = 0x00020007,

HvX64RegisterR8 = 0x00020008,

HvX64RegisterR9 = 0x00020009,

HvX64RegisterR10 = 0x0002000A,

HvX64RegisterR11 = 0x0002000B,

HvX64RegisterR12 = 0x0002000C,

HvX64RegisterR13 = 0x0002000D,

HvX64RegisterR14 = 0x0002000E,

HvX64RegisterR15 = 0x0002000F,

HvX64RegisterRip = 0x00020010,

HvX64RegisterRflags = 0x00020011,

// Floating Point and Vector Registers

HvX64RegisterXmm0 = 0x00030000,

HvX64RegisterXmm1 = 0x00030001,

HvX64RegisterXmm2 = 0x00030002,

HvX64RegisterXmm3 = 0x00030003,

HvX64RegisterXmm4 = 0x00030004,

HvX64RegisterXmm5 = 0x00030005,

HvX64RegisterXmm6 = 0x00030006,

HvX64RegisterXmm7 = 0x00030007,

HvX64RegisterXmm8 = 0x00030008,

HvX64RegisterXmm9 = 0x00030009,

HvX64RegisterXmm10 = 0x0003000A,

HvX64RegisterXmm11 = 0x0003000B,

HvX64RegisterXmm12 = 0x0003000C,

HvX64RegisterXmm13 = 0x0003000D,

HvX64RegisterXmm14 = 0x0003000E,

HvX64RegisterXmm15 = 0x0003000F,

HvX64RegisterFpMmx0 = 0x00030010,

HvX64RegisterFpMmx1 = 0x00030011,

HvX64RegisterFpMmx2 = 0x00030012,

HvX64RegisterFpMmx3 = 0x00030013,

HvX64RegisterFpMmx4 = 0x00030014,

HvX64RegisterFpMmx5 = 0x00030015,

HvX64RegisterFpMmx6 = 0x00030016,

HvX64RegisterFpMmx7 = 0x00030017,

HvX64RegisterFpControlStatus = 0x00030018,

HvX64RegisterXmmControlStatus = 0x00030019,

// Control Registers

HvX64RegisterCr0 = 0x00040000,

HvX64RegisterCr2 = 0x00040001,

HvX64RegisterCr3 = 0x00040002,

HvX64RegisterCr4 = 0x00040003,

HvX64RegisterCr8 = 0x00040004,

HvX64RegisterXfem = 0x00040005,

// Debug Registers

HvX64RegisterDr0 = 0x00050000,

HvX64RegisterDr1 = 0x00050001,

HvX64RegisterDr2 = 0x00050002,

HvX64RegisterDr3 = 0x00050003,

HvX64RegisterDr6 = 0x00050004,

HvX64RegisterDr7 = 0x00050005,

// Segment Registers

HvX64RegisterEs = 0x00060000,

HvX64RegisterCs = 0x00060001,

HvX64RegisterSs = 0x00060002,

HvX64RegisterDs = 0x00060003,

HvX64RegisterFs = 0x00060004,

HvX64RegisterGs = 0x00060005,

HvX64RegisterLdtr = 0x00060006,

HvX64RegisterTr = 0x00060007,

// Table Registers

HvX64RegisterIdtr = 0x00070000,

HvX64RegisterGdtr = 0x00070001,

// Virtualized MSRs

HvX64RegisterTsc = 0x00080000,

HvX64RegisterEfer = 0x00080001,

HvX64RegisterKernelGsBase = 0x00080002,

HvX64RegisterApicBase = 0x00080003,

HvX64RegisterPat = 0x00080004,

HvX64RegisterSysenterCs = 0x00080005,

HvX64RegisterSysenterRip = 0x00080006,

HvX64RegisterSysenterRsp = 0x00080007,

HvX64RegisterStar = 0x00080008,

HvX64RegisterLstar = 0x00080009,

HvX64RegisterCstar = 0x0008000A,

HvX64RegisterSfmask = 0x0008000B,

HvX64RegisterInitialApicId = 0x0008000C,

// Cache control MSRs

HvX64RegisterMtrrCap = 0x0008000D,

HvX64RegisterMtrrDefType = 0x0008000E,

HvX64RegisterMtrrPhysBase0 = 0x00080010,

HvX64RegisterMtrrPhysBase1 = 0x00080011,

HvX64RegisterMtrrPhysBase2 = 0x00080012,

HvX64RegisterMtrrPhysBase3 = 0x00080013,

HvX64RegisterMtrrPhysBase4 = 0x00080014,

HvX64RegisterMtrrPhysBase5 = 0x00080015,

HvX64RegisterMtrrPhysBase6 = 0x00080016,

HvX64RegisterMtrrPhysBase7 = 0x00080017,

HvX64RegisterMtrrPhysMask0 = 0x00080040,

HvX64RegisterMtrrPhysMask1 = 0x00080041,

HvX64RegisterMtrrPhysMask2 = 0x00080042,

HvX64RegisterMtrrPhysMask3 = 0x00080043,

HvX64RegisterMtrrPhysMask4 = 0x00080044,

HvX64RegisterMtrrPhysMask5 = 0x00080045,

HvX64RegisterMtrrPhysMask6 = 0x00080046,

HvX64RegisterMtrrPhysMask7 = 0x00080047,

HvX64RegisterMtrrFix64k00000 = 0x00080070,

HvX64RegisterMtrrFix16k80000 = 0x00080071,

HvX64RegisterMtrrFix16kA0000 = 0x00080072,

HvX64RegisterMtrrFix4kC0000 = 0x00080073,

HvX64RegisterMtrrFix4kC8000 = 0x00080074,

HvX64RegisterMtrrFix4kD0000 = 0x00080075,

HvX64RegisterMtrrFix4kD8000 = 0x00080076,

HvX64RegisterMtrrFix4kE0000 = 0x00080077,

HvX64RegisterMtrrFix4kE8000 = 0x00080078,

HvX64RegisterMtrrFix4kF0000 = 0x00080079,

HvX64RegisterMtrrFix4kF8000 = 0x0008007A,

// Hypervisor-defined MSRs (Misc)

HvX64RegisterVpRuntime = 0x00090000,

HvX64RegisterHypercall = 0x00090001,

HvX64RegisterGuestOsId = 0x00090002,

HvX64RegisterVpIndex = 0x00090003,

HvX64RegisterTimeRefCount = 0x00090004,

// Virtual APIC registers MSRs

HvX64RegisterEoi = 0x00090010,

HvX64RegisterIcr = 0x00090011,

HvX64RegisterTpr = 0x00090012,

HvX64RegisterApicAssistPage = 0x00090013,

// Performance statistics MSRs

HvX64RegisterStatsPartitionRetail = 0x00090020,

HvX64RegisterStatsPartitionInternal = 0x00090021,

HvX64RegisterStatsVpRetail = 0x00090022,

HvX64RegisterStatsVpInternal = 0x00090023,

// Hypervisor-defined MSRs (Synic)

HvX64RegisterSint0 = 0x000A0000,

HvX64RegisterSint1 = 0x000A0001,

HvX64RegisterSint2 = 0x000A0002,

HvX64RegisterSint3 = 0x000A0003,

HvX64RegisterSint4 = 0x000A0004,

HvX64RegisterSint5 = 0x000A0005,

HvX64RegisterSint6 = 0x000A0006,

HvX64RegisterSint7 = 0x000A0007,

HvX64RegisterSint8 = 0x000A0008,

HvX64RegisterSint9 = 0x000A0009,

HvX64RegisterSint10 = 0x000A000A,

HvX64RegisterSint11 = 0x000A000B,

HvX64RegisterSint12 = 0x000A000C,

HvX64RegisterSint13 = 0x000A000D,

HvX64RegisterSint14 = 0x000A000E,

HvX64RegisterSint15 = 0x000A000F,

HvX64RegisterScontrol = 0x000A0010,

HvX64RegisterSversion = 0x000A0011,

HvX64RegisterSiefp = 0x000A0012,

HvX64RegisterSimp = 0x000A0013,

HvX64RegisterEom = 0x000A0014,

// Hypervisor-defined MSRs (Synthetic Timers)

HvX64RegisterStimer0Config = 0x000B0000,

HvX64RegisterStimer0Count = 0x000B0001,

HvX64RegisterStimer1Config = 0x000B0002,

HvX64RegisterStimer1Count = 0x000B0003,

HvX64RegisterStimer2Config = 0x000B0004,

HvX64RegisterStimer2Count = 0x000B0005,

HvX64RegisterStimer3Config = 0x000B0006,

HvX64RegisterStimer3Count = 0x000B0007,

//

// XSAVE/XRSTOR register names.

//

// XSAVE AFX extended state registers.

HvX64RegisterYmm0Low = 0x000C0000,

HvX64RegisterYmm1Low = 0x000C0001,

HvX64RegisterYmm2Low = 0x000C0002,

HvX64RegisterYmm3Low = 0x000C0003,

HvX64RegisterYmm4Low = 0x000C0004,

HvX64RegisterYmm5Low = 0x000C0005,

HvX64RegisterYmm6Low = 0x000C0006,

HvX64RegisterYmm7Low = 0x000C0007,

HvX64RegisterYmm8Low = 0x000C0008,

HvX64RegisterYmm9Low = 0x000C0009,

HvX64RegisterYmm10Low = 0x000C000A,

HvX64RegisterYmm11Low = 0x000C000B,

HvX64RegisterYmm12Low = 0x000C000C,

HvX64RegisterYmm13Low = 0x000C000D,

HvX64RegisterYmm14Low = 0x000C000E,

HvX64RegisterYmm15Low = 0x000C000F,

HvX64RegisterYmm0High = 0x000C0010,

HvX64RegisterYmm1High = 0x000C0011,

HvX64RegisterYmm2High = 0x000C0012,

HvX64RegisterYmm3High = 0x000C0013,

HvX64RegisterYmm4High = 0x000C0014,

HvX64RegisterYmm5High = 0x000C0015,

HvX64RegisterYmm6High = 0x000C0016,

HvX64RegisterYmm7High = 0x000C0017,

HvX64RegisterYmm8High = 0x000C0018,

HvX64RegisterYmm9High = 0x000C0019,

HvX64RegisterYmm10High = 0x000C001A,

HvX64RegisterYmm11High = 0x000C001B,

HvX64RegisterYmm12High = 0x000C001C,

HvX64RegisterYmm13High = 0x000C001D,

HvX64RegisterYmm14High = 0x000C001E,

HvX64RegisterYmm15High = 0x000C001F

} HV\_REGISTER\_NAME;

### Virtual Processor Register Values

Virtual processor register values are all 128 bits in size. Values that do not consume the full 128 bits are zero-extended to fill out the entire 128 bits.

typedef union

{

UINT128 Reg128;

UINT64 Reg64;

UINT32 Reg32;

UINT16 Reg16;

UINT8 Reg8;

HV\_X64\_FP\_REGISTER Fp;

HV\_X64\_FP\_CONTROL\_STATUS\_REGISTER FpControlStatus;

HV\_X64\_XMM\_CONTROL\_STATUS\_REGISTER XmmControlStatus;

HV\_X64\_SEGMENT\_REGISTER Segment;

HV\_X64\_TABLE\_REGISTER Table;

HV\_EXPLICIT\_SUSPEND\_REGISTER ExplicitSuspend;

HV\_INTERCEPT\_SUSPEND\_REGISTER InterceptSuspend;

HV\_X64\_INTERRUPT\_STATE\_REGISTER InterruptState;

HV\_X64\_PENDING\_INTERRUPTION\_REGISTER PendingInterruption;

} HV\_REGISTER\_VALUE;

typedef HV\_REGISTER\_VALUE \*PHV\_REGISTER\_VALUE;

## Virtual Processor Register Formats

### Virtual Processor Suspend Registers

The hypervisor recognizes two registers that allow the virtual processor to be suspended or unsuspended. Two registers are specified so callers can easily set or clear the two values independently. Unlike most registers, these are not defined by the underlying processor architecture and cannot be accessed by the virtual processor they are associated with.

The first register (HvRegisterExplicitSuspend) is used for cases where the virtual processor is to be explicitly suspended using the HvSetVpRegisters hypercall.

typedef struct

{

UINT64 Suspended:1;

UINT64 Reserved:63;

} HV\_EXPLICIT\_SUSPEND\_REGISTER;

The second register (HvRegisterInterceptSuspend) is set when a virtual processor is suspended by the hypervisor due to an intercept. Bits that are set in this register may be cleared together or independently using the HvSetVpRegisters hypercall. Note that bits that are clear may only be set by the hypervisor and any attempt by a guest to set them will result in an error. For more information about suspending a virtual processor, see section .

This register also contains a bit used to inhibit the virtual processor’s TLB from being flushed from another processor using the HvFlushVirtualAddressSpace or HvFlushVirtualAddressList hypercalls. If such an attempt is made while the flush inhibit bit is set, the caller’s virtual processor will be suspended. When the flush inhibit bit is cleared, all other virtual processors suspended by an attempt to flush that virtual processor’s TLB will be unsuspended. For more information about TLB flush restrictions, see section .

typedef struct

{

UINT64 Suspended:1;

UINT64 TlbFlushInhibit:1;

UINT64 Reserved:62;

} HV\_INTERCEPT\_SUSPEND\_REGISTER;

### Virtual Processor Run Time Register

The hypervisor’s scheduler internally tracks how much time each virtual processor consumes in executing code. The time tracked is a combination of the time the virtual processor consumes running guest code, and the time the associated logical processor spends running hypervisor code on behalf of that guest. This cumulative time is accessible through the 64-bit read-only HV\_X64\_MSR\_VP\_RUNTIME hypervisor MSR. The time quantity is measured in 100ns units.

|  |
| --- |
| 63:0 |
| VP Runtime |

### Virtual Processor Interrupt State Register

The *interrupt state* register provides information about the interrupt state of the virtual processor. It indicates whether the virtual processor is in an “interrupt shadow” and whether non-maskable interrupts are currently masked. Certain instructions inhibit the delivery of hardware interrupts and debug traps for one instruction. Furthermore, when a non-maskable interrupt is delivered to the virtual processor, subsequent non-maskable interrupts are masked until the virtual processor executes an IRET instruction.

The interrupt state register is encoded as follows:

typedef struct

{

UINT64 InterruptShadow:1;

UINT64 NmiMasked:1;

UINT64 Reserved:62;

} HV\_X64\_INTERRUPT\_STATE\_REGISTER;

### Virtual Processor Pending Interruption Register

The *pending interruption* register is used to indicate whether a pending interruption exists for the virtual processor. An interruption is defined as any event that is delivered through the virtual processor’s interrupt descriptor table (for example, exceptions, interrupts, or debug traps). If an interruption is pending, the hypervisor will generate the interruption when the virtual processor resumes execution. This allows code running within the parent partition, for example, to respond to an MSR intercept by generating a general protection fault.

If an intercept is generated during the delivery of an interruption, the interruption is held pending and an intercept message is sent to the parent partition. The parent partition can resolve the intercept and resume the virtual processor, in which case the interruption will be re-delivered.

The type of a pending interruption is encoded as follows:

typedef enum

{

HvX64PendingInterrupt = 0,

HvX64PendingNmi = 2,

HvX64PendingException = 3

} HV\_X64\_PENDING\_INTERRUPTION\_TYPE;

The format of the pending interruptionregister is as follows:

typedef struct

{

UINT32 InterruptionPending:1;

UINT32 InterruptionType:3

UINT32 DeliverErrorCode:1;

UINT32 Reserved1:11;

UINT32 InterruptionVector:16;

UINT32 ErrorCode;

} HV\_X64\_PENDING\_INTERRUPTION\_REGISTER;

If the *InterruptionPending* bit is cleared, no interruption is pending, and the values in the other fields are ignored.

*InterruptionType* indicates the type of the interruption and can be any of the following values:

*HvX64PendingInterrupt —* The interruption is due to an interrupt.

*HvX64PendingNmi —* The interruption is due to a non-maskable interrupt.

*HvX64PendingException —* The interruption is due to a hardware exception.

*DeliverErrorCode* indicates whether an error code should be pushed on the stack as part of the interruption.

*InterruptionVector* indicates the vector to use for the exception.

*ErrorCode* indicates the error code value that will be pushed as part of the interruption frame.

### Virtual Processor Floating-point and Vector Registers

Floating point registers are encoded as 80-bit values, as follows:

typedef struct

{

UINT64 Mantissa;

UINT64 BiasedExponent:15;

UINT64 Sign:1;

UINT64 Reserved:48;

} HV\_X64\_FP\_REGISTER;

Additional status and control information for the floating point and vector units are stored in the following formats:

typedef struct

{

UINT16 FpControl;

UINT16 FpStatus;

UINT8 FpTag;

UINT8 IgnNe:1;

UINT8 Reserved:7;

UINT16 LastFpOp;

union

{

UINT64 LastFpRip;

struct

{

UINT32 LastFpEip;

UINT16 LastFpCs;

};

};

} HV\_X64\_FP\_STATUS\_CONTROL\_REGISTER;

typedef struct

{

union

{

UINT64 LastFpRdp;

struct

{

UINT32 LastFpDp;

UINT16 LastFpDs;

};

};

UINT32 XmmStatusControl;

UINT32 XmmStatusControlMask;

} HV\_X64\_XMM\_STATUS\_CONTROL\_REGISTER;

### Virtual Processor Segment Registers

Segment register state is encoded as follows:

typedef struct

{

UINT64 Base;

UINT32 Limit;

union

{

struct

{

UINT16 SegmentType:4;

UINT16 NonSystemSegment:1;

UINT16 DescriptorPrivilegeLevel:2;

UINT16 Present:1;

UINT16 Reserved:4;

UINT16 Available:1;

UINT16 Long:1;

UINT16 Default:1;

UINT16 Granularity:1;

};

UINT16 Attributes;

};

UINT16 Selector;

} HV\_X64\_SEGMENT\_REGISTER;

The limit is encoded as a 32-bit value. For X64 long-mode segments, the limit is ignored. For legacy x86 segments, the limit must be expressible within the bounds of the x64 processor architecture. For example, if the “G” (granularity) bit is set within the attributes of a code or data segment, the low-order 12 bits of the limit must be 1s.

The “Present” bit controls whether the segment acts like a null segment (that is, whether a memory access performed through that segment generates a #GP fault).

The MSRs IA32\_FS\_BASE and IA32\_GS\_BASE are not defined in the register list, as they are aliases to the base element of the segment register structure, Use HvX64RegisterFs and HvX64RegisterGs and the structure above instead.

### Virtual Processor Table Registers

Table registers are similar to segment registers, but they have no selector or attributes, and the limit is restricted to 16 bits.

typedef struct

{

UINT16 Pad[3];

UINT16 Limit;

UINT64 Base;

} HV\_X64\_TABLE\_REGISTER;

## Virtual Processor Interfaces

### HvCreateVp

The HvCreateVp hypercall creates a new virtual processor in a partition.

**Wrapper Interface**

HV\_STATUS

HvCreateVp(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PROXIMITY\_DOMAIN\_INFO ProximityDomainInfo,

\_\_in HV\_VP\_INDEX VpIndex,

\_\_in UINT64 Flags

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvCreateVp | | |
|  | Call Code = 0x004E | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | Padding (4 bytes) |
| 16 | ProximityDomainInfo (8 bytes) | |
| 24 | Flags (8 bytes) | |

Description

The specified virtual processor index must be below the maximum virtual processor index supported by the hypervisor implementation.

When a virtual processor is successfully created, its architected state is initialized consistent with the reset state of a real processor. The initial APIC ID is set to the specified virtual processor index. The parent partition can modify this value by calling HvSetRegisterState and specifying the register HvX64RegisterInitialApicId. The guest-programmable APIC ID, which is accessible through the APIC’s MMIO space, is also initialized based on the specified virtual processor index.

Upon creation, a virtual processor is marked suspended (that is, HvRegisterExplicitSuspend contains a value of 1). This value must be cleared before the virtual processor will begin executing instructions.

A virtual processor created with an index of zero will be designated as the *boot processor* and placed in the ready state. The virtual processor will begin executing instructions once its HvExplicitSuspendRegister is cleared. Virtual processors created with a non-zero index will be *application processors* and placed in the waiting state. Such processors must receive a SIPI interrupt (*HvX64InterruptTypeSipi)* in orderto begin execution.

Input Parameters

*PartitionId* specifies the partition.

*VpIndex* specifies the index of the virtual processor.

*ProximityDomainInfo* specifies the ACPI proximity domain information of the NUMA node where the virtual processor’s initial data structures will reside. If there are no pages in the caller’s pool for the specified ID, then the call will fail. The Proximity Domain specifier is described in section 7.2.1.

*Flags* is an options mask that affects the creation of the virtual processor. No flags are currently defined and this parameter must be zero.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_OPERATION\_DENIED | The partition has per-VP reserve or capacity property value defined and creation of a virtual processor would cause the total reserve or capacity to exceed 100%. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index references an existing virtual processor within the specified partition. |
|  | The specified VP index exceeds the maximum index allowed by the hypervisor implementation. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified flags value is not zero. |
|  | The *ProximityDomainInfo* parameter specifies an invalid flag bit or an invalid domain ID. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The memory pool of the specified partition does not contain sufficient pages to perform the operation. |
| HV\_STATUS\_NO\_RESOURCES | A required system resource is unavailable or an implementation limit has been reached. |

### HvDeleteVp

The HvDeleteVp hypercall deletes an existing virtual processor and its related data structures.

**Wrapper Interface**

HV\_STATUS

HvDeleteVp(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_VP\_INDEX VpIndex

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvDeleteVp [fast] | | |
|  | Call Code = 0x004F | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | Padding (4 bytes) |

Description

It returns the memory consumed by its internal data structures back to the memory pool of its partition.

Input Parameters

*PartitionId* specifies the partition.

*VpIndex* specifies the index of the virtual processor.

Output Parameters

None.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |

### HvGetVpRegisters

The HvGetVpRegisters hypercall reads the architectural state of a virtual processor.

**Wrapper Interface**

HV\_STATUS

HvGetVpRegisters(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_VP\_INDEX VpIndex,

\_\_inout PUINT32 RegisterCount,

\_\_in\_ecount(RegisterCount)

PCHV\_REGISTER\_NAME RegisterNameList,

\_\_out\_ecount(RegisterCount)

PHV\_REGISTER\_VALUE RegisterValueList

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvGetVpRegisters [rep] | | |
|  | Call Code = 0x0050 | |
| 🡆 Input Parameter Header | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | Padding (4 bytes) |
| 🡆 Input List Element | | |
| 0 | RegisterName[0] (4 bytes) | RegisterName[1] (4 bytes) |
| 🡄 Output List Element | | |
| 0 | RegisterValue (low-order) (8 bytes) | |
| 8 | RegisterValue (high-order) (8 bytes) | |

Description

The state is returned as a series of register values, each corresponding to a register name provided as input.

Input Parameters

*PartitionId* specifies the partition.

*VpIndex* specifies the index of the virtual processor.

*RegisterName* specifies a list of names for the requested register state.

Output Parameters

*RegisterValue* returns a list of register values for the requested register state.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId* or the partition specifying its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is neither the partition itself nor the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified register name is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |

### HvSetVpRegisters

The HvSetVpRegisters hypercall writes the architectural state of a virtual processor.

**Wrapper Interface**

HV\_STATUS

HvSetVpRegisters(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_VP\_INDEX VpIndex,

\_\_inout PUINT32 RegisterCount,

\_\_in\_ecount(RegisterCount)

PCHV\_REGISTER\_NAME RegisterNameList,

\_\_in\_ecount(RegisterCount)

PCHV\_REGISTER\_VALUE RegisterValueList

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvSetVpRegisters [rep] | | |
|  | Call Code = 0x0051 | |
| 🡆 Input Parameter Header | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | RsvdZ (4 bytes) |
| 🡆 Input List Element | | |
| 0 | RegisterName (4 bytes) | Padding (4 bytes) |
|  |  | |
| 16 | RegisterValue (low-order) (8 bytes) | |
| 24 | RegisterValue (high-order) (8 bytes) | |

Description

The state is written as a series of register values, each corresponding to a register name provided as input.

Minimal error checking is performed when a register value is modified. In particular, the hypervisor will validate that reserved bits of a register are set to zero, bits that are architecturally defined as always containing a zero or a one are set appropriately, and specified bits beyond the architectural size of the register are zeroed.

This call cannot be used to modify the value of a read-only register.

Side-effects of modifying a register are not performed. This includes generation of exceptions, pipeline synchronizations, TLB flushes, and so on.

Input Parameters

*PartitionId* specifies the partition.

*VpIndex* specifies the index of the virtual processor.

*RegisterName* specifies the name of a register to be modified.

*RegisterValue* specifies the new value for the specified register.

Output Parameters

None.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified register name is invalid. |
| The specified register is read-only. |
| The specified register value is not valid (for example, a reserved bit is not zero). |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |

# 

# Virtual Processor Execution

The virtual machine interface exposed by each partition exposes virtual processors, and these are architecture specific. This section specifies the core CPU aspects of virtual processors. The following two sections specify the MMU (memory management unit) and interrupt controller aspects of virtual processors.

A complete definition of virtual processor behavior requires hundreds of pages of CPU manuals. This document specifies the behavior of virtual processors by referencing processor manuals for physical X64 processors and discussing only cases where a virtual processor’s behavior differs from that of a logical processor; that is, the baseline behavior of a virtual processor is defined by the Intel and AMD processor reference manuals.

Some differences exist between AMD’s and Intel’s implementations. These differences, which are visible to guests, may include the following. Note that this list is not comprehensive and is likely to change as newer generations of processors are released:

Intel processors support:

* SSE3 instruction
* Hyperthreading

AMD processors support:

* 3DNow! instructions
* Fast FXSAVE mechanism
* Additional MMX extensions
* PREFETCHW instruction
* SAHF and LAHF instructions in long mode

## Processor Features and CPUID

The processor intercept mechanism of the hypervisor allows a parent partition to intercept the execution of the CPUID instruction by virtual processors within its child partitions. The parent partition can set the return values of the CPUID instruction in arbitrary ways. Doing so does not automatically alter the set of processor features of a virtual processor; that is, if a parent partition chooses to alter the behavior of the CPUID instruction, it is responsible for ensuring that the set of virtual processor features matches what is indicated by the CPUID instruction.

## Family, Model and Stepping Reported by CPUID

The CPUID instruction can be used to obtain a logical processor’s family, model and stepping information. It is possible, but not guaranteed, that logical processors reporting differing information may coexist on a single system. To properly use this information, a partition must be able to execute with a hard affinity between virtual and logical processors. Only the root partition executes in this manner. As a result, the hypervisor will expose the true family, model and stepping only to the root partition and will report the minimum value detected in the logical processor configuration to all other partitions.

## Platform ID Reported by MSR

The value returned by the IA32\_PLATFORM\_ID MSR (0x17) can be used in conjunction with the family, model and stepping information as reported by the CPUID instruction (as described in the previous section). The hypervisor is consistent in its handling of this MSR and will expose the true content of the IA32\_PLATFORM\_ID MSR only to the root partition. It will present all other partitions with the value obtained from the logical processor possessing the minimum family, model and stepping value.

## Real Mode

The hypervisor attempts to support real mode in a fully transparent manner. There may be situations, however, where specific processor implementations may not make this entirely possible. As a result, the hypervisor may be required to emulate or manipulate the environment to some degree to provide real mode support. The following is a list of potential areas where real mode support may not be transparent.

Hypervisor overhead inconsistencies.  
As a consequence of an increase in the frequency of instruction emulation by the hypervisor, performance of both the guest and the system may be affected.

Visible processor state changes as a consequence of switching modes.  
The hypervisor may be required to make changes to the guest’s runtime environment when mode switches occur, such as between real and protected mode or vice versa. Such changes may be detected by the guest.

## MONITOR / MWAIT

The hypervisor does not support the use of the MONITOR instruction but does have limited support for MWAIT. Partitions possessing the *CpuPowerManagement* privilege may use MWAIT to set the logical processor’s C-state if support for the instruction is present in hardware. Availability is indicated by the presence of a flag returned by the CPUID instruction for a hypervisor leaf (see section 3.4). Any attempt to use these instructions when the hypervisor does not indicate availability will result in a #UD fault.

## System Management Mode

The hypervisor does not support or participate in the virtualization of system management mode within guest partitions. Physical system management interrupts are still handled normally by the system’s hardware and firmware and is opaque to the hypervisor.

## Time Stamp Counter

The time stamp counter (TSC) is virtualized for each virtual processor. Generally, the TSC value continues to run while a virtual processor is suspended.

Seamless TSC virtualization is not feasible on the x64 architecture. TSC virtualization is typically implemented through a simple TSC bias (an offset added to the logical processor’s TSC). Attempts will be made by the hypervisor to prevent the TSC from jumping forward or backward as a virtual processor is scheduled on different logical processors. However, it cannot compensate for the situation where the TSC for a logical processor is set to zero by an SMI handler. Furthermore, the TSC increment rate may slow down or speed up depending on thermal or performance throttling, over which the hypervisor has no control.

Guest software should only use the TSC for measuring short durations. Even when using the TSC in this simple way, algorithms should be resilient to sudden jumps forward or backward in the TSC value.

## Memory Accesses

The behavior of instructions that access memory may differ from the behavior of the same instruction on a logical processor. This is the result of the hypervisor’s physical memory virtualization mechanisms and of the existence of address ranges with special semantics (hypervisor call page or SynIC area). In a broad sense, this applies to all instructions because the processor fetches instructions from memory. However, it applies in particular to instructions with memory operands.

The following pseudo-code defines the different behaviors that can result from an access by a virtual processor to its partition’s GPA space. The pseudo-code assumes that a GPA memory access has been performed directly (that is, an explicit memory operand) or indirectly (an implicit access) by a virtual processor. The access is one of three types: Read, Write, and Execute (Instruction Fetch).

if the address is within an overlay page

{

if the access type is not allowed for the page

{

Generate #MC fault within guest

}

else

{

Perform access

}

}

else if the address is within an unmapped GPA page

{

if the partition is the root partition

{

Allow the access to proceed to identity-mapped SPA

}

else

{

Suspend VP and send message to parent (umapped GPA)

}

}

else if the address is within a mapped GPA and

the access type violates the mapping’s access rights

{

if the partition is the root partition

{

Generate #MC fault in root

}

else

{

Suspend VP and send message to parent (GPA access right)

}

}

else

{

Memory access proceeds normally

}

## I/O Port Accesses

The behavior of instructions that access I/O ports may differ from the behavior of the same instruction on a logical processor. This is the result of the hypervisor’s processor intercept mechanism.

The following pseudo-code defines the different behaviors that can result from an access by a virtual processor to I/O ports (through the instructions IN, OUT, INS, or OUTS). Note that each of these instructions has an operand size of 1, 2, or 4 bytes. As such, one or more I/O ports are effectively accessed.

if any of the accessed ports is virtualized  
 by the hypervisor for this access type

{

Access is emulated

}

else if the I/O Port intercept is installed

{

Suspend VP and send message to parent (I/O Port Intercept)

}

else if the partition is a non-root partition

{

Discard writes; return all bits set for reads

}

else

{

I/O port access proceeds normally

}

## MSR Accesses

The behavior of instructions that access MSRs may differ from the behavior of the same instruction on a logical processor. This is the result of the hypervisor’s processor intercept mechanism.

The following pseudo-code defines the different behaviors that can result from an access by a virtual processor to MSRs.

if the MSR is virtualized by the hypervisor AND

the partition possesses the privilege required by the MSR

{

Access is emulated

}

else if the MSR intercept is installed

{

Suspend VP and send message to parent (MSR intercept)

}

else

{

Generate #GP fault within the guest

}

The hypervisor may virtualize MSRs as part of its interface with the guest. A summary of these can be found in

For those MSRs that are not virtualized by the hypervisor, internal security policy may require that certain fields within certain MSRs remain unmodified, are explicitly set for or hidden from the guest. In these cases, the access will appear to succeed from the guest’s perspective, but the value actually written or read may not match the underlying physical MSR value. The tables in 0 outline the policy for those MSRs whose contents are modified by the hypervisor.

### Modified MSR Behavior

Certain MSR accesses are intercepted by the hypervisor. The behavior of MSR accesses by the root guest may differ from accesses by other guests. These are described in Appendix F: Architectural MSRs.

## CPUID Execution

The behavior of the CPUID instruction may differ from the behavior of the same instruction on a logical processor.

The following pseudo-code defines the different behaviors that can result from the execution of a CPUID instruction by a virtual processor.

if an intercept has been set for the CPUID instruction for

the index specified in EAX

{

Suspend VP and send message to parent (CPUID Intercept)

}

else

{

CPUID instruction returns information as dictated by the

logical processor and the hypervisor

}

The hypervisor may override the standard CPUID information returned by the logical processor. The table in Appendix D: Architectural CPUID details the ways in which CPUID information is modified by the hypervisor. In some cases, the CPUID values returned to the root partition differ from non-root partitions. In such cases, the differences are noted.

**NOTE**

The hypervisor does not attempt to dictate a processor selection or to standardize on a particular processor model. The manipulation of various CPUID output is used to accommodate processor specifics or to reflect limitations on the partition’s accessibility or privilege to use certain processor features.

## Exceptions

The hypervisor’s intercept redirection mechanism allows a parent partition to intercept processor-generated exceptions in the virtual processors of a child partition. When the intercept message is delivered, the virtual processor will be in a restartable state (that is, the instruction pointer will point to the instruction that generated the exception).

Exception intercepts are checked before multiple exceptions are combined into a double fault or a triple fault. For example, if an exception intercept is installed on the #NP exception and a #NP exception occurs during the delivery of a #GP exception, the #NP exception intercept is triggered. Conversely, if no intercept was installed on the #NP exception, the nested #NP exception is converted into a double fault, which will trigger an intercept on the #DF if such an intercept was installed.

Note that exception intercepts do not occur for software-generated interrupts (that is, through the instructions INT, INTO, INT 3, and ICEBKPT).

The order in which exceptions are detected and reported by the processor depends on the instruction. For example, many instructions can generate multiple exceptions and the order in which these exceptions are detected is well defined.

The way in which exception intercepts interact with other intercept types also depends on the instruction. For example, an IN instruction may generate a #GP exception intercept before an I/O port intercept, and a RDMSR instruction may generate an MSR intercept before a #GP exception intercept. For details on the order of intercept delivery, consult the documentation for Intel’s and AMD’s processor virtualization extensions.

The following pseudo-code shows what occurs when an exception occurs.

if an intercept has been set for the exception

{

Suspend VP and send message to parent (Exception Intercept)

}

else

{

Exception is generated normally

}

# 

# Virtual MMU and Caching

The virtual machine interface exposed by each partition includes a memory management unit (MMU). The virtual MMU exposed by hypervisor partitions is generally compatible with existing MMUs.

The hypervisor also supports guest-defined memory cacheability attributes for pages mapped into a partition’s GVA space.

## Virtual MMU Overview

Virtual processors expose virtual memory and a virtual TLB (translation look-aside buffer), which caches translations from virtual addresses to (guest) physical addresses. As with the TLB on a logical processor, the virtual TLB is a non-coherent cache, and this non-coherence is visible to guests. The hypervisor exposes operations to flush the TLB. Guests can use these operations to remove potentially inconsistent entries and make virtual address translations predictable.

### Compatibility

The virtual MMU exposed by the hypervisor is generally compatible with the physical MMU found within an x64 processor. The following guest-observable differences exist:

The CR3.PWT and CR3.PCD bits may not be supported in some hypervisor implementations. On such implementations, any attempt by the guest to set these flags through a MOV to CR3 instruction or a task gate switch will be ignored. Attempts to set these bits programmatically through HvSetVpRegisters or HvSwitchVirtualAddressSpace may result in an error.

The PWT and PCD bits within a leaf page table entry (for example, a PTE for 4‑K pages and a PDE for large pages) specify the cacheability of the page being mapped. The PAT, PWT, and PCD bits within non-leaf page table entries indicate the cacheability of the next page table in the hierarchy. Some hypervisor implementations may not support these flags. On such implementations, all page table accesses performed by the hypervisor are done by using write-back cache attributes. This affects, in particular, *accessed* and *dirty* bits written to the page table entries. If the guest sets the PAT, PWT, or PCD bits within non-leaf page table entries, an “unsupported feature” message may be generated when a virtual processor accesses a page that is mapped by that page table.

The CR0.CD (cache disable) bit may not be supported in some hypervisor implementations. On such implementations, the CR0.CD bit must be set to 0. Any attempt by the guest to set this flag through a MOV to CR0 instruction will be ignored. Attempts to set this bit programmatically through HvSetVpRegisters will result in an error.

The PAT (page address type) MSR is a per-VP register. However, when all the virtual processors in a partition set the PAT MSR to the same value, the new effect becomes a partition-wide effect.

For reasons of security and isolation, the INVD instruction will be virtualized to act like a WBINVD instruction.

Some hypervisor implementations may use internal write protection of guest page tables to lazily flush MMU mappings from internal data structures (for example, shadow page tables). This is architecturally invisible to the guest because writes to these tables will be handled transparently by the hypervisor. However, writes performed to the underlying SPA pages by other partitions or by devices (that is, through DMA) may not trigger the appropriate TLB flush.

Internally, the hypervisor may use *shadow page tables* that translate GVAs to SPAs. In such implementations, these shadow page tables appear to software as large TLBs. However, several differences may be observable. First, shadow page tables can be shared between two virtual processors, whereas traditional TLBs are per-processor structures and are independent. This sharing may be visible because a page access by one virtual processor can fill a shadow page table entry that is subsequently used by another virtual processor.

### Legacy TLB Management Operations

The x64 architecture provides several ways to manage the processor’s TLBs. The following mechanisms are virtualized by the hypervisor:

The INVLPG instruction invalidates the translation for a single page from the processor’s TLB. If the specified virtual address was originally mapped as a 4‑K page, the translation for this page is removed from the TLB. If the specified virtual address was originally mapped as a “large page” (either 2 MB or 4 MB, depending on the MMU mode), the translation for the entire large page is removed from the TLB. The INVLPG instruction flushes both global and non-global translations. Global translations are defined as those which have the “global” bit set within the page table entry.

The MOV to CR3 instruction and task switches that modify CR3 invalidate translations for all non-global pages within the processor’s TLB.

A MOV to CR4 instruction that modifies the CR4.PGE (global page enable) bit, the CR4.PSE (page size extensions) bit, or CR4.PAE (page address extensions) bit invalidates all translations (global and non-global) within the processor’s TLB.

Note that all of these invalidation operations affect only one processor. To invalidate translations on other processors, software must use a software-based “TLB shoot-down” mechanism (typically implemented by using inter-process interrupts).

### Virtual TLB Enhancements

In addition to supporting the legacy TLB management mechanisms described earlier, the hypervisor also supports a set of enhancements that enable a guest to manage the virtual TLB more efficiently.

These enhanced operations can be used interchangeably with legacy TLB management operations. On some systems (those with sufficient virtualization support in hardware), the legacy TLB management instructions may be faster for local or remote (cross-processor) TLB invalidation. Guests who are interested in optimal performance should use the CPUID leaf 0x40000004 to determine which behaviors to implement using hypercalls:

*UseHypercallForAddressSpaceSwitch*: If this flag is set, the caller should assume that it’s faster to use HvSwitchAddressSpace to switch between address spaces. If this flag is clear, a MOV to CR3 instruction is recommended.

*UseHypercallForLocalFlush*: If this flag is set, the caller should assume that it’s faster to use hypercalls (as opposed to INVLPG or MOV to CR3) to flush one or more pages from the virtual TLB.

*UseHypercallForRemoteFlush*: If this flag is set, the caller should assume that it’s faster to use hypercalls (as opposed to using guest-generated inter-processor interrupts) to flush one or more pages from the virtual TLB.

### Restrictions on TLB Flushes

When a virtual processor generates an intercept—especially those associated with memory accesses, software running within the parent may want to complete the intercepted instruction in software. This instruction completion logic will need to emulate the address translation normally performed by the processor’s MMU. If a TLB flush request is executed on another virtual processor during instruction completion, incorrect behavior can result. For example, the second virtual processor could clear the dirty bit within the guest’s page table and then request a TLB flush. If the instruction completion software modifies the contents of this page after the TLB flush request has been completed, the operating system running within the partition will not be notified of the page modification, and data corruption can occur.

To prevent this situation, the hypervisor provides a way to inhibit TLB flush hypercalls until intercept processing is complete. When a memory intercept message is generated by the hypervisor, the “TLB Flush Inhibit” bit (TlbFlushInhibit) will consequently be set. Any attempt to flush the TLB with a hypercall will place the caller’s virtual processor in a suspended state. The instruction pointer will not be incremented past the instruction that invoked the hypercall. After the memory intercept routine performs instruction completion, it should clear the TlbFlushInhibit bit of the HvRegisterInterceptSuspend register. This resumes virtual processors that were suspended when they attempted to flush the TLB while the bit was set. Since the instruction pointer has not been incremented, the flush hypercall will automatically be re-executed. If the TlbFlushInhibit bit is clear, the hypercall will complete the flush normally.

## Memory Cache Control Overview

### Cacheability Settings

The hypervisor supports guest-defined cacheability settings for pages mapped within the guest’s GVA space. For a detailed description of available cacheability settings and their meanings, refer to the Intel or AMD documentation.

When a virtual processor accesses a page through its GVA space, the hypervisor honors the cache attribute bits (PAT, PWT, and PCD) within the guest page table entry used to map the page. These three bits are used as an index into the partition’s PAT (page address type) register to look up the final cacheability setting for the page.

Pages accessed directly through the GPA space (for example, when paging is disabled because CR0.PG is cleared) use a cacheability defined by the MTRRs. If the hypervisor implementation doesn’t support virtual MTRRs, WB cacheability is assumed.

### Mixing Cache Types between a Partition and the Hypervisor

Guests should be aware that some pages within its GPA space may be accessed by the hypervisor. The following list, while not exhaustive, provides several examples:

A page that contains input or output parameters for a hypercall

All overlay pages including the hypercall page, SynIC SIEF and SIM pages, and stats pages

The hypervisor always performs accesses to hypercall parameters and overlay pages by using the WB cacheability setting.

## Virtual MMU Data Types

### Virtual Address Spaces

The hypervisor introduces the concept of a virtual address space. The guest uses virtual address spaces to define the mapping between guest virtual addresses (GVAs) and guest physical addresses (GPAs). The guest OS can decide how and where to use virtual address spaces. In most OSs (including Microsoft Windows®), a different virtual address space is used for each process.

Virtual address spaces are identified by a caller-defined 64-bit ID value. On x64 implementations of the hypervisor, this value is the same as the value within the virtual processor’s CR3 register, which points to the guest’s page table structures.

typedef UINT64 HV\_ADDRESS\_SPACE\_ID;

### Virtual Address Flush Flags

The hypervisor provides hypercalls that allow the guest to flush (that is, invalidate) entire virtual address spaces or portions of these address spaces. Behavior of the flush operation can be modified by using a set of flags, defined as follows:

typedef UINT32 HV\_FLUSH\_FLAGS;

#define HV\_FLUSH\_ALL\_PROCESSORS 0x00000001

#define HV\_FLUSH\_ALL\_VIRTUAL\_ADDRESS\_SPACES 0x00000002

#define HV\_FLUSH\_NON\_GLOBAL\_MAPPINGS\_ONLY 0x00000004

### Cache Types

Several structures include cache type fields. The following encodings are defined:

typedef enum

{

HvCacheTypeX64Uncached = 0,

HvCacheTypeX64WriteCombining = 1,

HvCacheTypeX64WriteThrough = 4,

HvCacheTypeX64WriteProtected = 5,

HvCacheTypeX64WriteBack = 6

} HV\_CACHE\_TYPE;

### Virtual Address Translation Types

The call HvTranslateVirtualAddress takes a collection of input control flags and returns a result code and a collection of output flags. The input control flags are defined as follows:

typedef UINT64 HV\_TRANSLATE\_GVA\_CONTROL\_FLAGS;

#define HV\_TRANSLATE\_GVA\_VALIDATE\_READ 0x0001

#define HV\_TRANSLATE\_GVA\_VALIDATE\_WRITE 0x0002

#define HV\_TRANSLATE\_GVA\_VALIDATE\_EXECUTE 0x0004

#define HV\_TRANSLATE\_GVA\_PRIVILEGE\_EXEMPT 0x0008

#define HV\_TRANSLATE\_GVA\_SET\_PAGE\_TABLE\_BITS 0x0010

#define HV\_TRANSLATE\_GVA\_TLB\_FLUSH\_INHIBIT 0x0020

The returned result code is defined as follows:

typedef enum

{

HvTranslateGvaSuccess = 0,

// Translation failures

HvTranslateGvaPageNotPresent = 1,

HvTranslateGvaPrivilegeViolation = 2,

HvTranslateGvaInvalidPageTableFlags = 3,

// GPA access failures

HvTranslateGvaGpaUnmapped = 4,

HvTranslateGvaGpaNoReadAccess = 5,

HvTranslateGvaGpaNoWriteAccess = 6,

HvTranslateGvaGpaIllegalOverlayAccess = 7

} HV\_TRANSLATE\_GVA\_RESULT\_CODE;

typedef enum HV\_TRANSLATE\_GVA\_RESULT\_CODE

\*PHV\_TRANSLATE\_GVA\_RESULT\_CODE;

typedef struct

{

HV\_TRANSLATE\_GVA\_RESULT\_CODE ResultCode;

UINT32 CacheType:8;

UINT32 OverlayPage:1;

UINT32 Reserved3:23;

} HV\_TRANSLATE\_GVA\_RESULT;

### Gpa Access Types

The calls HvReadGpa and HvWriteGpa take a collection of input control flags and return a result code. The input control flags are defined as follows:

typedef struct

{

UINT64 CacheType:8; // Cache type for access

UINT64 Reserved:56;

} HV\_ACCESS\_GPA\_CONTROL\_FLAGS;

The return result code is defined as follows:

typedef enum

{

HvAccessGpaSuccess = 0,

// GPA access failures

HvAccessGpaUnmapped = 1,

HvAccessGpaReadIntercept = 2,

HvAccessGpaWriteIntercept = 3,

HvAccessGpaIllegalOverlayAccess = 4

} HV\_ACCESS\_GPA\_RESULT\_CODE;

typedef struct

{

HV\_ACCESS\_GPA\_RESULT\_CODE ResultCode;

UINT32 Reserved;

} HV\_ACCESS\_GPA\_RESULT;

typedef HV\_ACCESS\_GPA\_RESULT \*PHV\_ACCESS\_GPA\_RESULT;

## Virtual MMU Interfaces

### HvSwitchVirtualAddressSpace

The HvSwitchVirtualAddressSpace hypercall switches the calling virtual processor’s virtual address space.

**Wrapper Interface**

HV\_STATUS

HvSwitchVirtualAddressSpace(

\_\_in HV\_ADDRESS\_SPACE\_ID AddressSpace

);

Native Interface

|  |  |
| --- | --- |
| HvSwitchVirtualAddressSpace [fast] | |
|  | Call Code = 0x0001 |
| 🡆 Input Parameters | |
| 0 | AddressSpace (8 bytes) |

Description

For x64 implementations of the hypervisor, this call also updates the CR3 register. However, unlike a MOV to CR3 instruction, this hypercall does not have the side-effect of flushing the virtual processor’s TLB.

This hypercall, unlike most, operates implicitly in the context of the calling partition and virtual processor.

Input Parameters

*AddressSpace* specifies a new address space ID (a new CR3 value).

Output Parameters

None.

Restrictions

None.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_INVALID\_PARAMETER | The specified address space ID is not a valid CR3 value (for restrictions, see earlier in this specification). |
| One or more reserved bits in the specified address space ID (as defined by the x64 architecture) were set. |

### HvFlushVirtualAddressSpace

The HvFlushVirtualAddressSpace hypercall invalidates all virtual TLB entries that belong to a specified address space.

**Wrapper Interface**

HV\_STATUS

HvFlushVirtualAddressSpace(

\_\_in HV\_ADDRESS\_SPACE\_ID AddressSpace,

\_\_in HV\_FLUSH\_FLAGS Flags,

\_\_in UINT64 ProcessorMask

);

Native Interface

|  |  |
| --- | --- |
| HvFlushVirtualAddressSpace | |
|  | Call Code = 0x0002 |
| 🡆 Input Parameters | |
| 0 | AddressSpace (8 bytes) |
| 8 | Flags (8 bytes) |
| 16 | ProcessorMask (8 bytes) |

Description

The virtual TLB invalidation operation acts on one or more processors.

If the guest has knowledge about which processors may need to be flushed, it can specify a processor mask. Each bit in the mask corresponds to a virtual processor index. For example, a mask of 0x0000000000000051 indicates that the hypervisor should flush only the TLB of virtual processors 0, 4, and 6. A virtual processor can determine its index by reading from MSR HV\_X64\_MSR\_VP\_INDEX.

NOTE

Future versions of the hypervisor may support more than 64 virtual processors per partition. In that case, a new field will be added to the flags value that allows the caller to define the “processor bank” to which the processor mask applies.

The following flags can be used to modify the behavior of the flush:

*HV\_FLUSH\_ALL\_PROCESSORS* indicates that the operation should apply to all virtual processors within the partition. If this flag is set, the *ProcessorMask* parameter is ignored.

*HV\_FLUSH\_ALL\_VIRTUAL\_ADDRESS\_SPACES* indicates that the operation should apply to all virtual address spaces. If this flag is set, the *AddressSpace* parameter is ignored.

*HV\_FLUSH\_NON\_GLOBAL\_MAPPINGS\_ONLY* indicates that the hypervisor is required only to flush page mappings that were not mapped as “global” (that is, the x64 “G” bit was set in the page table entry). Global entries may be (but are not required to be) left unflushed by the hypervisor.

All other flags are reserved and must be set to zero.

This call guarantees that by the time control returns back to the caller, the observable effects of all flushes on the specified virtual processors have occurred.

If a target virtual processor’s TLB requires flushing and that virtual processor’s TLB is currently “locked”, the caller’s virtual processor is suspended. When the caller’s virtual processor is “unsuspended”, the hypercall will be reissued. For more information on TLB locking, see section .

Input Parameters

*AddressSpace* specifies an address space ID (a CR3 value).

*Flags* specifies a set of flag bits that modify the operation of the flush.

*ProcessorMask* specifies a processor mask indicating which processors should be affected by the flush operation.

Output Parameters

None.

Restrictions

None.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_INVALID\_PARAMETER | The specified address space ID is not a valid CR3 value and the ”flush all virtual address spaces” flag was not specified. |
| One or more reserved bits in the specified address space ID (as defined by the x64 architecture) were set. |
| One or more reserved bits within the flags register are set. |
| All of the bits in the processor bit mask are set to zero, and the "flush all processors" flag was not specified. |

### HvFlushVirtualAddressList

The HvFlushVirtualAddressList hypercall invalidates portions of the virtual TLB that belong to a specified address space.

**Wrapper Interface**

HV\_STATUS

HvFlushVirtualAddressList(

\_\_in HV\_ADDRESS\_SPACE\_ID AddressSpace,

\_\_in HV\_FLUSH\_FLAGS Flags,

\_\_in UINT64 ProcessorMask,

\_\_inout PUINT32 GvaCount,

\_\_in\_ecount(GvaCount)

PCHV\_GVA GvaRangeList

);

Native Interface

|  |  |
| --- | --- |
| HvFlushVirtualAddressList [rep] | |
|  | Call Code = 0x0003 |
| 🡆 Input Parameter Header | |
| 0 | AddressSpace (8 bytes) |
| 8 | Flags (8 bytes) |
| 16 | ProcessorMask (8 bytes) |
| 🡆 Input List Element | |
| 0 | GvaRange (8 bytes) |

Description

The virtual TLB invalidation operation acts on one or more processors.

If the guest has knowledge about which processors may need to be flushed, it can specify a processor mask. Each bit in the mask corresponds to a virtual processor index. For example, a mask of 0x0000000000000051 indicates that the hypervisor should flush only the TLB of virtual processors 0, 4 and 6.

The following flags can be used to modify the behavior of the flush:

*HV\_FLUSH\_ALL\_PROCESSORS* indicates that the operation should apply to all virtual processors within the partition. If this flag is set, the *ProcessorMask* parameter is ignored.

*HV\_FLUSH\_ALL\_VIRTUAL\_ADDRESS\_SPACES* indicates that the operation should apply to all virtual address spaces. If this flag is set, the *AddressSpace* parameter is ignored.

*HV\_FLUSH\_NON\_GLOBAL\_MAPPINGS\_ONLY* does not make sense for this call and is treated as an invalid option.

All other flags are reserved and must be set to zero.

This call takes a list of GVA ranges. Each range has a base GVA. Because flushes are performed with page granularity, the bottom 12 bits of the GVA can be used to define a range length. These bits encode the number of additional pages (beyond the initial page) within the range. This allows each entry to encode a range of 1 to 4096 pages.

A GVA that falls within a “large page” mapping (2MB or 4MB) will cause the entire large page to be flushed from the virtual TLB.

This call guarantees that by the time control returns back to the caller, the observable effects of all flushes on the specified virtual processors have occurred.

Invalid GVAs (those that specify addresses beyond the end of the partition’s GVA space) are ignored.

If a target virtual processor’s TLB requires flushing and that virtual processor is inhibiting TLB flushes, the caller’s virtual processor is suspended. When TLB flushes are no longer inhibited, the virtual processor is “unsuspended” and the hypercall will be reissued. For more information on TLB flush inhibit, see section .

Input Parameters

*AddressSpace* specifies an address space ID (a CR3 value).

*Flags* specifies a set of flag bits that modify the operation of the flush.

*ProcessorMask* specifies a processor mask indicating which processors should be affected by the flush operation.

*GvaRange* specifies a guest virtual address range.

Output Parameters

None.

Restrictions

None.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_INVALID\_PARAMETER | The specified address space ID is not a valid CR3 value and the "flush all virtual address spaces” flag was not specified. |
| One or more reserved bits in the specified address space ID (as defined by the x64 architecture) were set. |
| One or more reserved bits within the flags register are set. |
| All of the bits in the processor bit mask are set to zero, and the "flush all processors" flag was not specified. |

### HvTranslateVirtualAddress

The HvTranslateVirtualAddress hypercall attempts to translate a specified GVA page number into a GPA page number.

**Wrapper Interface**

HV\_STATUS

HvTranslateVirtualAddress(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_VP\_INDEX VpIndex,

\_\_in HV\_TRANSLATE\_GVA\_CONTROL\_FLAGS ControlFlags,

\_\_in HV\_GVA\_PAGE\_NUMBER GvaPage,

\_\_out PHV\_TRANSLATE\_GVA\_RESULT TranslationResult,

\_\_out PHV\_GPA\_PAGE\_NUMBER GpaPage

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvTranslateVirtualAddress | | |
|  | Call Code = 0x0052 | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | Padding (4 bytes) |
| 16 | ControlFlags (8 bytes) | |
| 24 | GvaPage (8 bytes) | |
| 🡄 Output Parameters | | |
| 0 | TranslationResult (8 bytes) | |
| 8 | GpaPage (8 bytes) | |

Description

The translation considers the current modes and state of the specified virtual processor as well as the guest page tables.

The caller must specify whether the intended access is to read, write or execute by setting the appropriate control flags. Combinations of these access types are possible. Several other translation options are also available.

*HV\_TRANSLATE\_GVA\_PRIVILEGE\_EXEMPT*: Indicates that the access should be performed as though the processor was running at a privilege level zero rather than the current privilege level.

*HV\_TRANSLATE\_GVA\_SET\_PAGE\_TABLE\_BITS*: Indicates that the routine should set the dirty and accessed bits within the guest’s page tables if appropriate for the access type. The dirty bit will only be set if *HV\_TRANSLATE\_GVA\_VALIDATE\_WRITE* is also specified. If the caller has requested that accessed and dirty bits be set as part of the table walk, these bits are set as the walk occurs. If a walk is aborted, the accessed and dirty bits that were already set are not restored to their previous values.

*HV\_TRANSLATE\_GVA\_LOCK\_TLB*: Indicates that the *TlbFlushInhibit* flag in the virtual processor’s HvRegisterInterceptSuspend register should be set as a consequence of a successful return. This prevents other virtual processors associated with the target partition from flushing the TLB of the specified virtual processor until after the *TlbFlushInhibit* flag is cleared. See section for more information.

If paging is disabled in the virtual processor (that is, CR0.PG is clear), then no page tables are consulted, and translation success is guaranteed.

If paging is enabled in the virtual processor (that is, CR0.PG is set), then a page table walk is performed. The call uses the current state of the virtual processor to determine whether to perform a two-level, three-level, or four-level page table walk.

During the page table walk, a number of conditions can arise that cause the walk to be terminated.

A table entry is marked “not present” or the GVA is beyond the range permitted for the paging mode. In this case, *HvTranslateGvaPageNotPresent* is returned.

A privilege violation is detected based on the access type (read, write, execute) or on the current privilege level. In this case, *HvTranslateGvaPrivilegeViolation* is returned.

A reserved bit is set within a table entry. In this case, *HvTranslateGvaInvalidPageTableFlags* is returned.

A page table walk can also be terminated if one of the guest’s page table pages cannot be accessed. This can occur in one of the following situations:

The GPA is unmapped. In this case, *HvTranslateGvaGpaUnmapped* is returned.

The GPA mapping’s access rights indicate that the page is not readable. In this case, *HvTranslateGvaGpaNoReadAccess* or *HvTranslateGvaGpaNoWriteAccess* is returned.

The access targets an overlay page that doesn’t allow reads or writes. In this case, *HvTranslateGvaGpaIllegalOverlayAccess* is returned.

If any of these GPA access failures are reported, the *GpaPage* output parameter is used to indicate which GPA page could not be accessed.

If no translation error occurs, *HvTranslateGvaSuccess* is returned. In this case, the *GpaPage* output parameter is used to report the resulting translation, and the associated *CacheType* and *OverlayPage* fields are set appropriately. The *CacheType* field indicates the effective cache type used by the virtual processor to access the translated virtual address. The *OverlayPage* field indicates whether the translated GPA accesses an overlay page owned by the hypervisor. Callers can use this information to determine whether memory accesses performed by the virtual processor would have accessed a mapped GPA page or an overlay page.

If the caller has requested that accessed and dirty bits be set as part of the table walk, then these bits are set as the walk occurs. If a walk is aborted, then the accessed and dirty bits that were already set are not restored to their previous values.

The reported cache type considers all of the state of the virtual processor including the current virtual PAT register settings and (if supported by the hypervisor implementation) the value of the MTRR MSRs and CR0.CD.

If the call returns HV\_STATUS\_SUCCESS, the output parameter *TranslationResult* is valid. The caller must consult the result code and results flags to determine whether the *GpaPage* parameter is valid.

Input Parameters

*PartitionId* specifies a partition.

*VpIndex* specifies a virtual processor index.

*ControlFlags* specifies a set of flag bits that modify the behavior of the translation.

*GvaPage* specifies a guest virtual address page number.

Output Parameters

*TranslationResult* specifies information about the translation including the result code and flags.

GpaPage specifies the translated GPA (if the result code is HvTranslateGvaSuccess) or the address of a GPA access failure (if the result code is HvTranslateGvaGpaUnmapped, HvTranslateGvaGpaNoReadAccess, HvTranslateGvaGpaNoWriteAccess, or HvTranslateGvaGpaIllegalOverlayAccess). For other result codes, this return parameter is invalid.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition. |
| HV\_STATUS\_INVALID\_PARAMETER | All three of the control flags HV\_TRANSLATE\_GVA\_VALIDATE\_READ, HV\_TRANSLATE\_GVA\_VALIDATE\_WRITE, and HV\_TRANSLATE\_GVA\_VALIDATE\_EXECUTE are cleared. At least one of these must be set. |
| One or more reserved bits in the specified control flags are set. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvReadGpa

The HvReadGpa hypercall attempts to read from a range of bytes within the specified GPA page.

**Wrapper Interface**

HV\_STATUS

HvReadGpa(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_VP\_INDEX VpIndex,

\_\_in HV\_GPA BaseGpa,

\_\_in UINT32 ByteCount,

\_\_in HV\_ACCESS\_GPA\_CONTROL\_FLAGS ControlFlags,

\_\_out PHV\_ACCESS\_GPA\_RESULT AccessResult,

\_\_out\_ecount(ByteCount) PVOID DataBuffer

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvReadGpa | | |
|  | Call Code = 0x0053 | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | ByteCount (4 bytes) |
| 16 | BaseGpa (8 bytes) | |
| 24 | ControlFlags (8 bytes) | |
| 🡄 Output Parameters | | |
| 0 | AccessResult (8 bytes) | |
| 8 | DataLow (8 bytes) | |
| 16 | DataHigh (8 bytes) | |

Description

The range of bytes being read from is not allowed to cross a page boundary. Up to 16 bytes of data can be read. A byte count of zero is invalid.

The hypervisor performs the access according to the current GPA mapping. If the specified GPA references an overlay page, the access is directed to the overlay page and not the page that is potentially mapped to the specified GPA. This is also true of MMIO ranges managed by the hypervisor (namely, the APIC page). In other words, the access is performed as though the virtual processor itself performed the access.

Data will be accessed in an aligned manner using the appropriate combination of 32-bit, 16-bit and 8-bit reads and will be performed in ascending address order.

Accesses to overlay pages perform the side effects normally associated with the access.

Atomicity of the access is not guaranteed by this call. Another virtual processor may be attempting to modify the specified GPA range. Unless the other virtual processor uses atomic accesses, the call may return partially-modified data.

If the call returns HV\_STATUS\_SUCCESS, the *AccessResult* parameter is valid. Callers must check the *AccessResult* field to determine whether an error was detected when the specified GPA page was accessed. This can occur for one of three reasons:

The GPA is unmapped. In this case, *HvAccessGpaUnmapped* is returned.

The GPA mapping’s access rights indicate that the page is not readable. In this case, *HvAccessGpaReadIntercept* is returned.

The access targets an overlay page that doesn’t allow reads. In this case, *HvAccessGpaWriteIntercept* is returned.

Input Parameters

*PartitionId* specifies a partition.

*VpIndex* specifies a virtual processor index.

*BaseGpa* specifies the first guest physical address to be accessed.

*ByteCount* specifies the number of bytes to access (1 through 16, inclusive).

*ControlFlags* specifies a set of flag bits that modify the behavior of the access.

Output Parameters

*AccessResult* specifies information about the access including the result code.

*Data* returns the data that was read. *DataLow* contains bytes 1-8 and *DataHigh* contains bytes 9-16. *If* the caller specified a byte count smaller than 16 bytes, the remaining bytes in the output data will contain zeroed bytes.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified GPA is beyond the end of the partition’s GPA space. |
| The specified access crosses a page boundary. |
| The specified byte count is 0 or greater than 16. |
| A reserved bit within the control flags is set. |
| The specified cache type is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvWriteGpa

The HvWriteGpa hypercall attempts to write to a range of bytes within the specified GPA page.

**Wrapper Interface**

HV\_STATUS

HvWriteGpa(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_VP\_INDEX VpIndex,

\_\_in HV\_GPA BaseGpa,

\_\_in UINT32 ByteCount,

\_\_in HV\_ACCESS\_GPA\_CONTROL\_FLAGS ControlFlags,

\_\_in\_ecount(ByteCount) PCVOID DataBuffer,

\_\_out PHV\_ACCESS\_GPA\_RESULT AccessResult

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvWriteGpa | | |
|  | Call Code = 0x0054 | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | VpIndex (4 bytes) | ByteCount (4 bytes) |
| 16 | BaseGpa (8 bytes) | |
| 24 | ControlFlags (8 bytes) | |
| 32 | DataLow (8 bytes) | |
| 40 | DataHigh (8 bytes) | |
| 🡄 Output Parameters | | |
| 0 | AccessResult (8 bytes) | |

Description

The range of bytes being written to is not allowed to cross a page boundary. Up to 16 bytes of data can be written. A byte count of zero is invalid.

The hypervisor performs the access according to the current GPA mapping. If the specified GPA references an overlay page, the access is directed to the overlay page and not the page that is potentially mapped to the specified GPA. This is also true of MMIO ranges managed by the hypervisor (namely, the APIC page). In other words, the access is performed as though the virtual processor itself performed the access.

Data will be accessed in an aligned manner using the appropriate combination of 32-bit, 16-bit and 8-bit writes and will be performed in ascending address order.

Accesses to overlay pages perform the side effects normally associated with the access.

Atomicity of the access is not guaranteed by this call. Another virtual processor may be attempting to read or write specified GPA range. Unless the other virtual processor uses atomic accesses, the other virtual processor may observe partially-written data by this call.

If the call returns HV\_STATUS\_SUCCESS, the *AccessResult* parameter is valid. Callers must check the *AccessResult* field to determine whether an error was detected when the specified GPA page was accessed. This can occur for one of three reasons:

The GPA is unmapped. In this case, *HvAccessGpaUnmapped* is returned.

The GPA mapping’s access rights indicate that the page is not writable. In this case, *HvAccessGpaReadIntercept* is returned.

The access targets an overlay page that doesn’t allow writes. In this case, *HvAccessGpaWriteIntercept* is returned.

Input Parameters

*PartitionId* specifies a partition.

*VpIndex* specifies a virtual processor index.

*BaseGpa* specifies the first guest physical address to be accessed.

*ByteCount* specifies the number of bytes to access (1 through 16, inclusive).

*ControlFlags* specifies a set of flag bits that modify the behavior of the access.

*Data* specifies the data to be written. *DataLow* contains bytes 1-8 and *DataHigh* contains bytes 9-16. If the caller specified a byte count smaller than 16 bytes, the remaining bytes in the input data are ignored.

Output Parameters

*AccessResult* specifies information about the access including the result code.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified GPA is beyond the end of the partition’s GPA space. |
| The specified access crosses a page boundary. |
| The specified byte count is 0 or greater than 16. |
| A reserved bit within the control flags is set. |
| The specified cache type is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

# 

# Virtual Interrupt Control

## Overview

The hypervisor virtualizes interrupt delivery to virtual processors. This is done through the use of a synthetic interrupt controller (SynIC) which is an extension of a virtualized local APIC; that is, each virtual processor has a local APIC instance with the SynIC extensions. These extensions provide a simple inter-partition communication mechanism which is described in the following chapter.

Interrupts delivered to a partition fall into two categories: external and internal. External interrupts originate from other partitions or devices, and internal interrupts originate from within the partition itself.

External interrupts are generated in the following situations:

A physical hardware device generates a hardware interrupt (in which case the interrupt is reflected to the root partition).

A parent partition calls HvAssertVirtualInterrupt (typically in the process of emulating a hardware device).

The hypervisor delivers a message (for example, due to an intercept) to a partition.

Another partition calls HvPostMessage.

Another partition calls HvSignalEvent.

Internal interrupts are generated in the following situations:

A virtual processor accesses the APIC interrupt command register (ICR).

A synthetic timer expires.

## Local APIC

The SynIC is a superset of a local APIC. The interface to this APIC is given by a set of 32-bit memory mapped registers. This local APIC (including the behavior of the memory mapped registers) is generally compatible with the local APIC on P4/Xeon systems as described in Intel’s documentation.

### Local APIC Virtualization

The hypervisor’s local APIC virtualization may deviate from physical APIC operation in the following minor ways:

On physical systems, the IA32\_APIC\_BASE MSR can be different for each processor in the system. The hypervisor may require that this MSR contains the same value for all virtual processors within a partition. As such, this MSR may be treated as a partition-wide value. If a virtual processor modifies this register, the value may effectively propagate to all virtual processors within the partition.

The IA32\_APIC\_BASE MSR defines a “global enable” bit for enabling or disabling the APIC. The virtualized APIC may always be enabled. If so, this bit will always be set to 1.

The hypervisor’s local APIC may not be able to generate virtual SMIs (system management interrupts).

The hypervisor may allow accesses only to the APIC’s memory-mapped registers to be performed by one of the instructions in section . Furthermore, it may allow only accesses that are four bytes in size and aligned to four-byte boundaries. In such cases, if an unsupported access is attempted, the virtual processor will be suspended, and an unsupported feature error message will be delivered to the partition’s parent.

If multiple virtual processors within a partition are assigned identical APIC IDs, behavior of targeted interrupt delivery is boundedly undefined. That is, the hypervisor is free to deliver the interrupt to just one virtual processor, all virtual processors with the specified APIC ID, or no virtual processors. This situation is considered a guest programming error.

Some of the memory mapped APIC registers may be accessed by way of virtual MSRs.

The remaining parts of this section describe only those aspects of SynIC functionality that are extensions of the local APIC.

### Local APIC Memory-mapped Accesses

The hypervisor emulates accesses to memory-mapped registers within the virtualized local APIC. However, only certain instruction forms are supported, and use of other forms will result in #GP. Compatible guests should access only the local APIC registers by using the following instruction forms:

| Opcode | Instruction | Notes |
| --- | --- | --- |
| 89 /r | MOV m32,r32 | m32 must be 4-byte aligned. |
| 8B /r | MOV r32,m32 | m32 must be 4-byte aligned. |
| A1 | MOV EAX,moffs32 | moffs32 must be 4-byte aligned. |
| A3 | MOV moffs32,EAX | moffs32 must be 4-byte aligned. |
| C7 /0 | MOV m32,imm32 | m32 must be 4-byte aligned. |
| FF /6 | PUSH m32 | m32 must be 4-byte aligned. |

### Local APIC MSR Accesses

The hypervisor provides accelerated MSR access to high usage memory mapped APIC registers. These are the TPR, EOI, and the ICR registers. The ICR low and ICR high registers are combined into one MSR.

|  |  |  |
| --- | --- | --- |
| MSR Address | Register Name | Function |
| 0x40000070 | HV\_X64\_MSR\_EOI | Accesses the APIC EOI |
| 0x40000071 | HV\_X64\_MSR\_ICR | Accesses the APIC ICR-high and ICR-low |
| 0x40000072 | HV\_X64\_MSR\_TPR | Access the APIC TPR |

For performance reasons, the guest operating system should follow the hypervisor recommendation for the usage of the APIC MSRs (see section )

#### EOI Register

| 63:32 | 31:0 |
| --- | --- |
| Ignored | EOI value |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:32 | RsvdZ (reserved, should be zero) | Write |
| 31:0 | EOI value | Write |

This is a write-only register, and it sets a value into the APIC EOI register. Attempts to read from this register will result in a #GP fault.

#### ICR Register

| 63:32 | 31:0 |
| --- | --- |
| ICR high | ICR low |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:32 | ICR high value | Read/write |
| 31:0 | ICR low value | Read/write |

The values of ICR high and ICR low are read from or written into the corresponding APIC ICR high and low registers.

#### TPR Register

| 63:8 | 7:0 |
| --- | --- |
| RsvdZ | TPR value |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:8 | RsvdZ (reserved, should be zero) | Read/write |
| 7:0 | TPR value | Read/write |

The value of the APIC TPR register is read or written.

**NOTE**

This MSR is intended to accelerate access to the TPR in 32-bit mode guest partitions. 64-bit mode guest partitions should set the TPR by way of CR8.

## Virtual Interrupts

### Virtual Interrupt Overview

The hypervisor provides interfaces that allow a partition to send virtual interrupts to specified virtual processors. This is useful for emulating an IOAPIC or a legacy 8259 PIC (programmable interrupt controller).

### Virtual Interrupt Types

To send a virtual interrupt, software (typically running within the parent partition) must call HvAssertVirtualInterrupt and specify a virtual processor within the target partition. It must also specify the interrupt type that determines the behavior:

*HvX64InterruptTypeNmi* generates a non-maskable interrupt on the specified processor.

*HvX64InterruptTypeSmi* generates a system management interrupt on the specified processor.

*HvX64InterruptTypeInit* generates an INIT interrupt on the specified processor.

*HvX64InterruptTypeSipi* generates a start inter-processor interrupt. If the target processor is in wait-for-SIPI state, it causes the target processor to begin executing in real mode at an address determined by the SIPI vector as specified by the x64 architecture.

*HvX64InterruptTypeFixed* generates a fixed interrupt latched into the local APIC’s interrupt request register (IRR). A fixed interrupt can be edge-triggered or level-triggered. Withdrawing an edge-triggered interrupt does not clear the corresponding bit in the IRR. Withdrawing a level-triggered interrupt clears the corresponding bit in the IRR.

*HvX64InterruptTypeLowestPriority* is like a fixed interrupt except that it is delivered only to the lowest-priority destination virtual processor.

*HvX64InterruptTypeExtInt* generates a fixed level-triggered interrupt. The behavior is the same as with *HvX64InterruptTypeFixed,* with the following exceptions:

It is always directed at the boot processor, and

It can be used when the APIC is software disabled.

Regardless of whether the APIC is enabled or not, the PPR (process priority register) is not used in determining whether the interrupt will be serviced. This type is also special in that it is always directed at the boot processor. It also requires the use of a separate hypercall, HvClearVirtualInterrupt, to clear an acknowledged interrupt before subsequent interrupts of this type can be asserted.

### Trigger Types

Virtual interrupts are either edge-triggered or level-triggered. Edge-triggered interrupts are latched upon assertion and cannot be withdrawn. Level-triggered interrupts are not latched and can potentially be withdrawn by deasserting. The following table indicates, for each interrupt type, what the implicit interrupt trigger type is and whether a vector should be specified with the virtual interrupt.

| Interrupt type | Vector applicable? | Trigger type |
| --- | --- | --- |
| NMI | No | Edge |
| INIT | No | Edge |
| SIPI | Yes | Edge |
| Fixed | Yes | Edge or Level |
| Lowest Priority | Yes | Edge or Level |
| ExtINT | Yes | Level |
| SMI | Yes | ?? |

Sometime after a virtual interrupt is asserted, it may be acknowledged by the virtual processor. Until then, level-triggered virtual interrupts can be deasserted by calling HvAssertVirtualInterrupt with vector HV\_INTERRUPT\_VECTOR\_NONE or it can be re-asserted by calling HvAssertVirtualInterrupt. Deasserting an edge-triggered interrupt is unnecessary and has no effect.

### EOI Intercepts

An intercept is defined for processor events (specifically, memory accesses) that indicate the EOI (end of interrupt) for a level-triggered fixed interrupt. An EOI intercept is the expected (eventual) response by the child to a parent asserting a level triggered interrupt using the HvAssertVirtualInterrupt hypercall. The intercept is delivered at the instruction boundary following the instruction that issued the EOI.

For performance reasons, it is desirable to reduce the number of EOI intercepts. Most EOI intercepts can be eliminated and done lazily if the guest OS leaves a marker when it performs an EOI. However, there are two cases for which EOI intercepts are strictly necessary.

A level triggered interrupt is EOI’ed, since the hypervisor needs to either EOI the physical APIC (in case of the root partition) or send an EOI message (in case of a non-root partition) when the guest performs an EOI.

A lower priority interrupt is pending, since the hypervisor needs to re-evaluate interrupts when the guest performs an EOI.

For more information on intercepts, see section .

#### APIC Assist Page Register

The hypervisor provides a virtual APIC assist page per virtual processor which is overlaid on the guest GPA space. The OS has read/write access to the virtual APIC assist page. It specifies the location of the overlay page (in GPA space) by writing to the Virtual APIC Assist MSR (0x40000073). The format of the Virtual APIC Assist Page MSR is as follows:

| 63:12 | 11:1 | 0 |
| --- | --- | --- |
| Virtual APIC Assist Page Base Address | RsvdP | Enable |

Currently the only defined field in the virtual APIC assist page is the EOI Assist field. The EOI Assist field resides at offset 0 of the overlay page and is DWORD sized. The format of the EOI assist field is as follows:

| 31:1 | 0 |
| --- | --- |
| Reserved to Zero | No EOI Required |

The OS performs an EOI by atomically writing zero to the EOI Assist field of the virtual APIC assist page and checking whether the “No EOI required” field was previously zero. If it was, the OS must write to the HV\_X64\_APIC\_EOI MSR thereby triggering an intercept into the hypervisor. The following code is recommended to perform an EOI:

lea rcx, [VirtualApicAssistVa]

btr [rcx], 0

jc NoEoiRequired

mov ecx, HV\_X64\_APIC\_EOI

wrmsr

NoEoiRequired:

The hypervisor sets the “No EOI required” bit when it injects a virtual interrupt if the following conditions are satisfied:

* The virtual interrupt is edge-triggered, and
* There are no lower priority interrupts pending

If, at a later time, a lower priority interrupt is requested, the hypervisor clears the “No EOI required” such that a subsequent EOI causes an intercept.

In case of nested interrupts, the EOI intercept is avoided only for the highest priority interrupt. This is necessary since no count is maintained for the number of EOIs performed by the OS. Therefore only the first EOI can be avoided and since the first EOI clears the “No EOI Required” bit, the next EOI generates an intercept. However nested interrupts are rare, so this is not a problem in the common case.

Note that devices and/or the I/O APIC (physical or synthetic) need not be notified of an EOI for an edge-triggered interrupt – the hypervisor intercepts such EOIs only to update the virtual APIC state. In some cases, the virtual APIC state can be lazily updated – in such cases, the “NoEoiRequired” bit is set by the hypervisor indicating to the guest that an EOI intercept is not necessary. At a later instant, the hypervisor can derive the state of the local APIC depending on the current value of the “NoEoiRequired” bit.

Enabling and disabling this enlightenment can be done at any time independently of the interrupt activity and the APIC state at that moment. While the enlightenment is enabled, conventional EOIs can still be performed irrespective of the “No EOI required” value but they will not realize the performance benefit of the enlightenment.

## Virtual Interrupt Data Types

### Interrupt Types

Several virtual interrupt types are supported.

typedef enum

{

HvX64InterruptTypeFixed = 0x0000,

HvX64InterruptTypeLowestPriority = 0x0001,

HvX64InterruptTypeNmi = 0x0004,

HvX64InterruptTypeInit = 0x0005,

HvX64InterruptTypeSipi = 0x0006,

HvX64InterruptTypeExtInt = 0x0007

} HV\_INTERRUPT\_TYPE;

### Interrupt Control

The interrupt control specifies the type of the virtual interrupt, its destination mode and whether the virtual interrupt is edge or level triggered.

typedef struct

{

HV\_INTERRUPT\_TYPE InterruptType;

UINT32 LevelTriggered:1;

UINT32 LogicalDestinationMode:1;

UINT32 Reserved:30;

} HV\_INTERRUPT\_CONTROL;

### Interrupt Vectors

Interrupt vectors are represented by a 32-bit value. A special value is used to indicate “no interrupt vector” and is used by calls that indicate whether a previous interrupt was acknowledged.

typedef UINT32 HV\_INTERRUPT\_VECTOR;

typedef HV\_INTERRUPT\_VECTOR \*PHV\_INTERRUPT\_VECTOR;

#define HV\_INTERRUPT\_VECTOR\_NONE 0xFFFFFFFF

## Virtual Interrupt Interfaces

### HvAssertVirtualInterrupt

The HvAssertVirtualInterrupt hypercall requests a virtual interrupt to be presented to the specified virtual processor(s).

**Wrapper Interface**

HV\_STATUS

HvAssertVirtualInterrupt(

\_\_in HV\_PARTITION\_ID DestinationPartition,

\_\_in HV\_INTERRUPT\_CONTROL InterruptControl,

\_\_in UINT64 DestinationAddress,

\_\_in HV\_INTERRUPT\_VECTOR RequestedVector

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvAssertVirtualInterrupt | | |
|  | Call Code = 0x0055 | |
| 🡆 Input Parameters | | |
| 0 | DestinationPartition (8 bytes) | |
| 8 | InterruptControl (8 bytes) | |
| 16 | DestinationAddress (8 bytes) | |
| 24 | RequestedVector (4 bytes) | Padding (4 bytes) |

Description

For information on virtual interrupts, see section .

If the call is made twice in a row with the same interrupt type specified in the *InterruptControl* parameter, the behavior depends upon whether or not the first interrupt was acknowledged by the virtual processor before the second call is made.

If the first interrupt has already been acknowledged, then the second call is treated as a new assertion.

If the first interrupt has not yet been acknowledged, then the second call supersedes the previous assertion with the new vector. If the second call specifies the vector HV\_INTERRUPT\_VECTOR\_NONE, then the call acts as a deassertion.

The behavior of this call differs for interrupts of type *HvX64InterruptTypeExtInt* in the following ways:

This interrupt type is always targeted at the boot processor. The boot processor is identified by a virtual processor index of zero. The *DestinationAddress* parameter must, therefore, be zero.

Calls to HvAssertVirtualInterrupt will fail if the interrupt asserted by a previous call has already been acknowledged by the processor. This acknowledgement must first be cleared by calling HvClearVirtualInterrupt. This is especially useful when implementing an external interrupt controller, such as the 8259 PIC. It prevents HvAssertVirtualInterrupt from overwriting the previous acknowledgement, which may need to be reported through the external interrupt controller.

Input Parameters

*DestinationPartition* specifies the partition.

*InterruptControl* specifies the type of the virtual interrupt that should be asserted, its destination mode and whether the virtual interrupt is edge or level triggered.

*DestinationAddress* specifies the destination virtual processor(s). In case of physical destination mode, the destination address specifies the physical APIC ID of the target virtual processor. In case of logical destination mode, the destination address specifies the logical APIC ID of the set of target virtual processors. This value must be zero for external interrupt delivery mode where the interrupt request is always sent to the boot processor.

*RequestedVector* specifies the interrupt vector. This value is used only for fixed, lowest-priority, external, and SIPI interrupt types. In all other cases, a vector of zero must be specified.

Output Parameters

None.

Restrictions

The partition specified by *DestinationPartition* must be in the “active” state.

The caller must be the parent of the partition specified by *DestinationPartition*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The virtual processor selected by the *DestinationAddress* parameter is not valid. |
| For interrupts of type HvX64InterruptTypeExtInt, the DestinationAddress was non-zero. |
| HV\_STATUS\_INVALID\_PARAMETER | One or more fields of the specified interrupt control are invalid or reserved bits within the interrupt control are set. |
| The specified destination address is invalid or is non-zero for an external interrupt type. |
| The specified vector is not within a valid range (0 to 255 inclusive or HV\_INTERRUPT\_VECTOR\_NONE). |
| A non-zero vector is specified with an interrupt type that is not fixed, lowest-priority, external, or SIPI. |
| HV\_STATUS\_ACKNOWLEDGED | An external interrupt cannot be asserted because a previously-asserted external interrupt was acknowledged by the virtual processor and has not yet been cleared. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvClearVirtualInterrupt

The HvClearVirtualInterrupt hypercall determines whether a previously-requested virtual interrupt of type *HvX64InterruptTypeExtInt* has been acknowledged.

**Wrapper Interface**

HV\_STATUS

HvClearVirtualInterrupt(

\_\_in HV\_PARTITION\_ID DestinationPartition

);

Native Interface

|  |  |
| --- | --- |
| HvClearVirtualInterrupt [fast] | |
|  | Call Code = 0x0056 |
| 🡆 Input Parameters | |
| 0 | DestinationPartition (8 bytes) |

Description

An external interrupt is considered “acknowledged” when it has been delivered to the boot processor.

The call clears the acknowledged state which allows subsequent external interrupts to be asserted. If no external interrupt has been acknowledged, then the call fails with a status code of HV\_STATUS\_NOT\_ACKNOWLEDGED.

For more information on virtual interrupts, see section

Input Parameters

*DestinationPartition* specifies the partition.

Output Parameters

None.

Restrictions

The partition specified by *DestinationPartition* must be in the “active” state.

The caller must be the parent of the partition specified by *DestinationPartition*.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_NOT\_ACKNOWLEDGED | An external interrupt has not been asserted and acknowledged since the last call to HvClearVirtualInterrupt. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the "active" state. |

# 

# Inter-Partition Communication

## Overview

The hypervisor provides two simple mechanisms for one partition to communicate with another: messages and events. In both cases, notification is signaled by using the SynIC (synthetic interrupt controller).

## SynIC Messages

The hypervisor provides a simple inter-partition communication facility that allows one partition to send a parameterized message to another partition. (Because the message is sent asynchronously, it is said to be *posted*.) The destination partition may be notified of the arrival of this message through an interrupt.

### Message Buffers

A *message buffer* is used internally to the hypervisor to store a message until it is delivered to the recipient. The hypervisor maintains several sets of message buffers:

**Guest message buffers**: The hypervisor maintains a set of guest message buffers for each port. These buffers are used for messages sent explicitly from one partition to another by a guest. When a port is created, the hypervisor will allocate sixteen (16) message buffers from the port owner’s memory pool (see section ). These message buffers are returned to the memory pool when the port is deleted (see section ).

**Timer message buffers**: The hypervisor maintains four timer message buffers for each virtual processor (one per synthetic interrupt timer). They are allocated when a virtual processor is created. For more information about timers, see chapter , and for a detailed description of timer messages, see section .

**Intercept message buffers**: The hypervisor maintains one intercept message buffer for each virtual processor. It is used for intercepts. The intercept message buffer is allocated when the virtual processor is created.

**Event log message buffers**: The hypervisor maintains one event log message buffer for each event log group. It is used to notify the root partition when one or more event log buffers are full. For details about event logging, see chapter .

### Message Buffer Queues

For each partition and each virtual processor in the partition, the hypervisor maintains one queue of message buffers for each SINTx (synthetic interrupt source) in the virtual processor’s SynIC. All message queues of a virtual processor are empty upon creation or reset of the virtual processor.

### Reliability and Sequencing of Guest Message Buffers

Messages successfully posted by a guest have been queued for delivery by the hypervisor. Actual delivery and reception by the target partition is dependent upon its correct operation. Partitions may disable delivery of messages to particular virtual processors by either disabling its SynIC (see section ) or disabling the SIMP (see section 14.6.4).

Breaking a connection (see HvDisconnectPort, section ) will not affect undelivered (queued) messages. Deletion of the target port (see HvDeletePort, section ) will always free all of the port’s message buffers, whether they are available or contain undelivered (queued) messages.

Messages arrive in the order in which they have been successfully posted. If the receiving port is associated with a specific virtual processor, then messages will arrive in the same order in which they were posted. If the receiving port is associated with HV\_ANY\_VP, then messages are not guaranteed to arrive in any particular order.

### Messages

When a message is sent:

The hypervisor selects a free message buffer. The set of available message buffers depends on the event that triggered the sending of the message.

The hypervisor marks the message buffer “in use” and fills in the message header with the message type, payload size, and information about the sender. Finally, it fills in the message payload. The contents of the payload depend on the event that triggered the message. This document specifies the payloads of all messages generated by the hypervisor. The payload for messages sent by calling HvPostMessage must be defined by the caller.

The hypervisor then appends the message buffer to a receiving message queue. The receiving message queue depends on the event that triggered the sending of the message. For all message types, SINTx is either implicit (in the case of intercept messages), explicit (in the case of timer messages) or specified by a port ID (in the case of guest messages). The target virtual processor is either explicitly specified or chosen by the hypervisor when the message is enqueued. Virtual processors whose SynIC or SIM page (see section 14.7) is disabled will not be considered as potential targets. If no targets are available, the hypervisor terminates the operation and returns an error to the caller.

The hypervisor then determines whether the specified SINTx message slot within the SIM page for the target virtual processor is empty. (See section for a description of the SIM page.) If the message type in the message slot is equal to HvMessageTypeNone (that is, zero), the message slot is assumed to be empty. In this case, the hypervisor dequeues the message buffer and copies its contents to the message slot within the SIM page. The hypervisor may copy only the number of payload bytes associated with the message. The hypervisor also attempts to generate an edge-triggered interrupt for the specified SINTx. If the APIC is software disabled or the SINTx is masked, the interrupt is lost. The arrival of this interrupt notifies the guest that a new message has arrived. If the SIM page is disabled or the message slot within the SIM page is not empty, the message remains queued, and no interrupt is generated.

As with any fixed-priority interrupt, the interrupt is not acknowledged by the virtual processor until the PPR (process priority register) is less than the vector specified in the SINTx register and interrupts are not masked by the virtual processor (rFLAGS[IF] is set to 1).

Multiple message buffers with the same SINTx can be queued to a virtual processor. In this case, the hypervisor will deliver the first message (that is, write it to the SIM page) and leave the others queued until one of three events occur:

Another message buffer is queued.

The guest indicates the “end of interrupt” by writing to the APIC’s EOI register.

The guest indicates the “end of message” by writing to the SynIC’s EOM register.

In all three cases, the hypervisor will scan one or more message buffer queues and attempt to deliver additional messages. The hypervisor also attempts to generate an edge-triggered interrupt, indicating that a new message has arrived.

If a queued message cannot be delivered because the corresponding SIM entry is still in use, the hypervisor will attempt to deliver it again after an unspecified time (typically on the order of milliseconds). To avoid this potential latency, software should mark the SIM entry as unused before indicating an EOI or EOM.

### Recommended Message Handling

The SynIC message delivery mechanism is designed to accommodate efficient delivery and receipt of messages within a target partition. It is recommended that the message handling ISR (interrupt service routine) within the target partition perform the following steps:

Examine the message that was deposited into the SIM message slot.

Copy the contents of the message to another location and set the message type within the message slot to HvMessageTypeNone.

Indicate the end of interrupt for the vector by writing to the APIC’s EOI register.

Perform any actions implied by the message.

### Message Sources

The classes of events that can trigger the sending of a message are as follows:

**Intercepts:** Any intercept in a virtual processor will cause a message to be sent. The message buffer used is the intercept message buffer of the virtual processor that caused the intercept. The receiving message queue belongs to SINT0 of a virtual processor that the hypervisor selects non-deterministically from among the virtual processors of the parent partition. The message payload describes the event that caused the intercept. If the intercept message buffer is already queued when an intercept occurs, it is removed from the queue, overwritten, and placed back on the queue. This should occur only if the software running in the parent partition clears the “suspended for intercept” register before receiving the intercept message. This situation is considered a programming error.

**Timers:** The timer mechanism defined in chapter will cause messages to be sent. Associated with each virtual processor are four dedicated timer message buffers, one for each timer. The receiving message queue belongs to SINTx of the virtual processor whose timer triggered the sending of the message. Timer messages are defined in section .

**Guest messages:** The hypervisor supports message passing as an inter-partition communication mechanism between guests. The interfaces defined in this section allow one guest to send messages to another guest. The message buffers used for messages of this class are taken from the receiver’s per-port pool of guest message buffers.

**Event log buffers:** The hypervisor will send a message when an event log buffer has been filled.

## SynIC Event Flags

In addition to messages, the SynIC supports a second type of cross-partition notification mechanism called *event flags*. Event flags may be set explicitly using the HvSignalEvent hypercall or implicitly by the hypervisor as a consequence of the monitored notification facility.

### Event Flag Delivery

When a partition calls HvSignalEvent, it specifies an event flag number. The hypervisor responds by atomically setting a bit within the receiving virtual processor’s SIEF page. (See section for a detailed description of the SIEF page.) Virtual processors whose SynIC or SIEF page is disabled will not be considered as potential targets. If no targets are available, the hypervisor terminates the operation and returns an error to the caller.

If the event flag was previously cleared, the hypervisor attempts to notify the receiving partition that the flag is now set by generating an edge-triggered interrupt. The target virtual processor, along with the target SINTx, is specified as part of a port’s creation. (See the following for information about ports.) If the SINTx is masked, HvSignalEvent returns HV\_STATUS\_INVALID\_SYNIC\_STATE.

As with any fixed-priority external interrupt, the interrupt is not acknowledged by the virtual processor until the process priority register (PPR) is less than the vector specified in the SINTx register and interrupts are not masked by the virtual processor (rFLAGS[IF] is set to 1).

### Recommended Event Flag Handling

It is recommended that the event flag interrupt service routine (ISR) within the target partition perform the following steps:

Examine the event flags and determine which ones, if any, are set.

Clear one or more event flags by using a locked (atomic) operation such as LOCK AND or LOCK CMPXCHG.

Indicate the end of interrupt for the vector by writing to the APIC’s EOI register.

Perform any actions implied by the event flags that were set.

### Event Flags versus Messages

Event flags are lighter-weight than messages and are therefore lower overhead. Furthermore, event flags do not require any buffer allocation or queuing within the hypervisor, so HvSignalEvent will never fail due to insufficient resources.

## Ports and Connections

A message or event sent from one guest to another must be sent through a pre-allocated *connection.* A connection, in turn, must be associated with a destination *port*.

A port is allocated from the receiver’s memory pool and specifies which virtual processor and SINTx to target. Event ports have a “base flag number” and “flag count” that allow the caller to specify a range of valid event flags for that port.

Connections are allocated from the sender’s memory pool. When a connection is created, it must be associated with a valid port. This binding creates a simple, one-way communication channel. If a port is subsequently deleted, its connection, while it remains, becomes useless.

## Monitored Notifications

The monitored notification facility (MNF) introduces the concept of shared triggers between two communicating partitions. MNF uses a port (in the recipient partition) and a connection (in the originating partition) to establish a hypervisor-monitored, unidirectional notification channel. A monitor port-and-connection pair alone isn’t enough to form the said notification channel. It needs in addition to be associated with an event connection through the monitored notification parameters in the monitored notification page

When the channel is created, a monitored notification is established in an overlay page that includes the following:

A trigger,

A latency hint

A set of input parameters appropriate for the HvSignalEvent hypercall.

After the monitor page is established, the hypervisor periodically examines the trigger at a rate subject to the latency hint to determine if a notification is warranted. If so, the hypervisor invokes the HvSignalEvent hypercall internally on behalf of the originating guest. The behavior is the same as if the originating guest had invoked the HvSignalEvent directly.

### Monitored Notification Trigger

The trigger can be directly accessed by guests without hypervisor intervention. It is set or cleared by the inter-partition communication code running in the communicating guests. The trigger must be placed in memory that is shared by the two communicating partitions and the hypervisor.

### Monitored Notification Latency Hint

The latency hint specifies an approximate wait period between hypervisor examinations of the trigger. It is expressed in 100 nanosecond units. The hypervisor can override the specified latency value if making it somewhat smaller or larger is more efficient. The hypervisor can also override the specified latency value if it exceeds minimum or maximum values.

### Monitored Notification Parameters

Each MNF trigger is defined by a set of input parameters compatible with those accepted by an HvSignalEvent hypercall. These parameters include an event flag number and a connection ID. If the internal invocation of the HvSignalEvent hypercall fails, the error is discarded and the invocation is treated as a NOP.

### Monitored Notification Page

Monitored notifications are collected into monitor overlay pages that can be created or deleted only from a parent partition. The parent partition creates a monitor-page port in the recipient and specifies the GPA of the recipient’s associated monitor page. The parent subsequently creates a connection to that the monitor page port in the originator and specifies the GPA of the originator’s associated monitor page. While each of these two GPAs is partition-specific, the underlying physical page is a common page that is managed by the hypervisor. Changes to the page are visible from both partitions as well as the hypervisor.

## SynIC MSRs

In addition to the memory-mapped registers defined for a local APIC, the following model-specific registers (MSRs) are defined in the SynIC. Each virtual processor has its own copy of these registers, so they can be programmed independently.

| MSR Address | Register Name | Function |
| --- | --- | --- |
| 0x40000080 | SCONTROL | SynIC Control |
| 0x40000081 | SVERSION | SynIC Version |
| 0x40000082 | SIEFP | Interrupt Event Flags Page |
| 0x40000083 | SIMP | Interrupt Message Page |
| 0x40000084 | EOM | End of message |
| 0x40000090 | SINT0 | Interrupt source 0 (hypervisor) |
| 0x40000091 | SINT1 | Interrupt source 1 |
| 0x40000092 | SINT2 | Interrupt source 2 |
| 0x40000093 | SINT3 | Interrupt source 3 |
| 0x40000094 | SINT4 | Interrupt source 4 |
| 0x40000095 | SINT5 | Interrupt source 5 |
| 0x40000096 | SINT6 | Interrupt source 6 |
| 0x40000097 | SINT7 | Interrupt source 7 |
| 0x40000098 | SINT8 | Interrupt source 8 |
| 0x40000099 | SINT9 | Interrupt source 9 |
| 0x4000009A | SINT10 | Interrupt source 10 |
| 0x4000009B | SINT11 | Interrupt source 11 |
| 0x4000009C | SINT12 | Interrupt source 12 |
| 0x4000009D | SINT13 | Interrupt source 13 |
| 0x4000009E | SINT14 | Interrupt source 14 |
| 0x4000009F | SINT15 | Interrupt source 15 |

### SCONTROL Register

| 63:1 | 0 |
| --- | --- |
| RsvdP | Enable |

This register is used to control SynIC behavior of the virtual processor.

|  |  |  |
| --- | --- | --- |
| Bits | Description | Attributes |
| 63:1 | RsvdP (value must be preserved) | Read/write |
| 0 | Enable  When set, this virtual processor will allow message queuing and event flag notifications to be posted to its SynIC (see chapter for details). When clear, message queuing and event flag notifications cannot be directed to this virtual processor. | Read/write |

At virtual processor creation time and upon processor reset, the value of this SCONTROL (SynIC control register) is 0x0000000000000000. Thus, message queuing and event flag notifications will be disabled.

### SVERSION Register

| 63:32 | 31:0 |
| --- | --- |
| Rsvd | SynIC Version (0x00000001) |

This is a read-only register, and it returns the version number of the SynIC. For the first version of the hypervisor, the value is 0x00000001. Attempts to write to this register result in a #GP fault.

### SIEFP Register

| 63:12 | 11:1 | 0 |
| --- | --- | --- |
| SIEFP Base Address | RsvdP | Enable |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:12 | Base address (in GPA space) of SIEFP  (low 12 bits assumed to be zero) | Read/write |
| 11:1 | RsvdP (value should be preserved) | Read/write |
| 0 | SIEFP enable | Read/write |

At virtual processor creation time and upon processor reset, the value of this SIEFP (synthetic interrupt event flags page) register is 0x0000000000000000. Thus, the SIEFP is disabled by default. The guest must enable it by setting bit 0. If the specified base address is beyond the end of the partition’s GPA space, the SIEFP page will not be accessible to the guest. When modifying the register, guests should preserve the value of the reserved bits (1 through 11) for future compatibility.

### SIMP Register

| 63:12 | 11:1 | 0 |
| --- | --- | --- |
| SIMP Base Address | RsvdP | Enable |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:12 | Base address (in GPA space) of SIMP  (low 12 bits assumed to be zero) | Read/write |
| 11:1 | RsvdP (value should be preserved) | Read/write |
| 0 | SIMP enable | Read/write |

At virtual processor creation time and upon processor reset, the value of this SIMP (synthetic interrupt message page) register is 0x0000000000000000. Thus, the SIMP is disabled by default. The guest must enable it by setting bit 0. If the specified base address is beyond the end of the partition’s GPA space, the SIMP page will not be accessible to the guest. When modifying the register, guests should preserve the value of the reserved bits (1 through 11) for future compatibility.

### SINTx Registers

| 63:18 | 17 | 16 | 15:8 | 7:0 |
| --- | --- | --- | --- | --- |
| RsvdP | AutoEOI | Mask | RsvdP | Vector |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:18 | RsvdP (value should be preserved) | Read/write |
| 17 | AutoEOI  Set if an implicit EOI should be performed upon interrupt delivery | Read/write |
| 16 | Set if the SINT is masked | Read/write |
| 15:8 | RsvdP (value should be preserved) | Read/write |
| 7:0 | Vector | Read/write |

At virtual processor creation time, the default value of all SINTx (synthetic interrupt source) registers is 0x0000000000010000. Thus, all synthetic interrupt sources are masked by default. The guest must unmask them by programming an appropriate vector and clearing bit 16.

The AutoEOI flag indicates that an implicit EOI should be performed by the hypervisor when an interrupt is delivered to the virtual processor. In addition, the hypervisor will automatically clear the corresponding flag in the “in service register” (ISR) of the virtual APIC. If the guest enables this behavior, then it must not perform an EOI in its interrupt service routine.

The AutoEOI flag can be turned on at any time, though the guest must perform an explicit EOI on an in-flight interrupt The timing consideration makes it difficult to know whether a particular interrupt needs EOI or not, so it is recommended that once SINT is unmasked, its settings are not changed.

Likewise, the AutoEOI flag can be turned off at any time, though the same concerns about in-flight interrupts apply

Valid values for v*ector* are 16-255 inclusive. Specifying an invalid vector number results in #GP.

### EOM Register

| 63:0 |
| --- |
| RsvdZ |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:0 | RsvdZ (value should be set to zero) | Write-only trigger |

A write to the end of message (EOM) register by the guest causes the hypervisor to scan the internal message buffer queue(s) associated with the virtual processor. If a message buffer queue contains a queued message buffer, the hypervisor attempts to deliver the message. Message delivery succeeds if the SIM page is enabled and the message slot corresponding to the SINTx is empty (that is, the message type in the header is set to HvMessageTypeNone). If a message is successfully delivered, its corresponding internal message buffer is dequeued and marked free. If the corresponding SINTx is not masked, an edge-triggered interrupt is delivered (that is, the corresponding bit in the IRR is set).

This register can be used by guests to “poll” for messages. It can also be used as a way to drain the message queue for a SINTx that has been disabled (that is, masked).

If the message queues are all empty, a write to the EOM register is a no-op.

Reads from the EOM register always returns zeros.

## SIM and SIEF Pages

The SynIC defines two pages that extend the functionality of a traditional APIC. The pages for these two addresses are specified by the SIEFP register and the SIMP register (see earlier in this specification for these register formats).

The SIEF and SIM pages are implemented as GPA overlay pages. For a description of overlay pages, see section .

The addresses of the SIEF and SIM pages should be unique for each virtual processor. Programming these pages to overlap other instances of the SIEF or SIM pages or any other overlay page (for example, the hypercall page) will result in undefined behavior.

The hypervisor may implement the SIEF and SIM pages so that a SIEF or SIM instance associated with a virtual processor is not accessible to other virtual processors. In such implementations, an access by one virtual processor to another virtual processor’s SIEF or SIM page will result in a #MC fault. It is highly recommended that guests avoid performing such accesses.

Read and write accesses by a virtual processor to the SIEF and SIM pages behave like read and write accesses to RAM. However, the hypervisor’s SynIC implementation also writes to these pages in response to certain events.

Upon virtual processor creation and reset, the SIEF and SIM pages are cleared to zero.

The SIEF page consists of a 16-element array of 256-byte event flags (see the following for an explanation of event flags). Each array element corresponds to a single synthetic interrupt source (SINTx).

The SIM page consists of a 16-element array of 256-byte messages (see the following HV\_MESSAGE data structure). Each array element (also known as a *message slot*) corresponds to a single synthetic interrupt source (SINTx). A message slot is said to be “empty” if the message type of the message in the slot is equal to HvMessageTypeNone.

## Inter-Partition Communication Data Types

### Synthetic Interrupt Sources

The SynIC supports 16 synthetic interrupt sources.

#define HV\_SYNIC\_SINT\_COUNT 16

typedef UINT32 HV\_SYNIC\_SINT\_INDEX;

### SynIC Message Types

SynIC messages encode the message type as a 32-bit number.

typedef enum

{

HvMessageTypeNone = 0x00000000,

// Memory access messages

HvMessageTypeUnmappedGpa = 0x80000000,

HvMessageTypeGpaIntercept = 0x80000001,

// Timer notifications

HvMessageTimerExpired = 0x80000010,

// Error messages

HvMessageTypeInvalidVpRegisterValue = 0x80000020,

HvMessageTypeUnrecoverableException = 0x80000021,

HvMessageTypeUnsupportedFeature = 0x80000022,

// Default (non-programmable) intercept messages

HvMessageTypeApicEoi = 0x80000030,

HvMessageTypeFerrAsserted = 0x80000031,

// Trace buffer messages

HvMessageTypeEventLogBuffersComplete = 0x80000040,

// Platform-specific processor intercept messages

HvMessageTypeX64IoPortIntercept = 0x80010000,

HvMessageTypeX64MsrIntercept = 0x80010001,

HvMessageTypeX64CpuidIntercept = 0x80010002,

HvMessageTypeX64ExceptionIntercept = 0x80010003

} HV\_MESSAGE\_TYPE;

#define HV\_MESSAGE\_TYPE\_HYPERVISOR\_MASK 0x80000000

Any message type that has the high bit set is reserved for use by the hypervisor. Guest-initiated messages cannot send messages with a hypervisor message type.

For a complete list of messages sent by the hypervisor, see section .

### SynIC Message Flags

| 7:1 | 0 |
| --- | --- |
| RsvdZ | MessagePending |

| Bits | Description | Meaning |
| --- | --- | --- |
| 7:1 | RsvdP (value should be set to zero) | Reserved |
| 0 | MessagePending | One or more messages are pending in the message queue |

The *MessagePending* flag indicates whether or not there are any messages pending in the message queue of the synthetic interrupt source. If there are, then an “end of message” must be performed by the guest after emptying the message slot. This allows for opportunistic writes to the EOM MSR (only when required). Note that this flag may be set by the hypervisor upon message delivery or at any time afterwards. The flag should be tested after the message slot has been emptied and if set, then there are one or more pending messages and the “end of message” should be performed.

typedef struct

{

UINT8 MessagePending:1;

UINT8 Reserved:7;

} HV\_MESSAGE\_FLAGS;

### SynIC Message Format

SynIC messages are of fixed size composed of a message header (which includes the message type and information about where the message originated) followed by the payload. Messages that are sent in response to HvPostMessage contain the port ID. Intercept messages contain the partition ID of the partition whose virtual processor generated the intercept. Timer intercepts do not have an origination ID (that is, the specified ID is zero).

#define HV\_MESSAGE\_SIZE 256

#define HV\_MESSAGE\_MAX\_PAYLOAD\_BYTE\_COUNT 240

#define HV\_MESSAGE\_MAX\_PAYLOAD\_QWORD\_COUNT 30

typedef struct

{

HV\_MESSAGE\_TYPE MessageType;

UINT16 Reserved;

HV\_MESSAGE\_FLAGS MessageFlags;

UINT8 PayloadSize;

union

{

UINT64 OriginationId;

HV\_PARTITION\_ID Sender;

HV\_PORT\_ID Port;

};

} HV\_MESSAGE\_HEADER;

typedef struct

{

HV\_MESSAGE\_HEADER Header;

UINT64 Payload[HV\_MESSAGE\_MAX\_PAYLOAD\_QWORD\_COUNT];

} HV\_MESSAGE;

For a detailed description of the messages sent by the hypervisor, see chapter .

### SynIC Event Flags

SynIC event flags are fixed-size bitwise arrays. They are numbered such that the first byte of the array contains flags 0 through 7 (0 being the least significant bit) and the second byte of the array contains flags 8 through 15 (8 being the least significant bit), and so on.

#define HV\_EVENT\_FLAGS\_COUNT (256 \* 8)

#define HV\_EVENT\_FLAGS\_BYTE\_COUNT 256

typedef struct

{

UINT8 Flags[HV\_EVENT\_FLAGS\_BYTE\_COUNT];

} HV\_SYNIC\_EVENT\_FLAGS;

### Ports

Destination ports are identified by 32-bit IDs. The high 8 bits of the ID are reserved and must be zero. All port IDs are unique within a partition.

typedef union

{

UINT32 AsUint32;

struct

{

UINT32 Id:24;

UINT32 Reserved:8;

};

} HV\_PORT\_ID;

Three types of ports are supported: message ports, event ports, and monitor ports. Message ports are valid for use with the HvPostMessage hypercall. Event ports are valid for use with the HvSignalEvent hypercall. Monitor ports are valid for use with monitor pages that are monitored by the hypervisor and result in HvSignalEvent-based notifications when appropriate.

enum HV\_PORT\_TYPE

{

HvPortTypeMessage = 1,

HvPortTypeEvent = 2,

HvPortTypeMonitor = 3

};

When a port is created, the following information is specified.

typedef struct

{

HV\_PORT\_TYPE PortType;

UINT32 ReservedZ;

union

{

struct

{

HV\_SYNIC\_SINT\_INDEX TargetSint;

HV\_VP\_INDEX TargetVp;

UINT64 ReservedZ;

} MessagePortInfo;

struct

{

HV\_SYNIC\_SINT\_INDEX TargetSint;

HV\_VP\_INDEX TargetVp;

UINT16 BaseFlagNumber;

UINT16 FlagCount;

UINT32 ReservedZ;

} EventPortInfo;

struct

{

HV\_GPA MonitorAddress;

UINT64 ReservedZ;

} MonitorPortInfo;

};

} HV\_PORT\_INFO;

### Port Properties

Port properties provide a generic way to set or obtain certain characteristics of a port. Properties are identified by a 32-bit code. The associated property value is 64 bits in size.

typedef UINT64 HV\_PORT\_PROPERTY;

typedef HV\_PORT\_PROPERTY \*PHV\_PORT\_PROPERTY;

typedef enum

{

HvPortPropertyPostCount = 0x00000000,

HvPortPropertyPreferredTargetVp = 0x00000001

} HV\_PORT\_PROPERTY\_CODE;

The following table explains the meaning of each property.

| Port property | Meaning |
| --- | --- |
| HvPortPropertyPostCount | The number of messages that have been successfully posted to the port since its creation. |
| HvPortPropertyPreferredTargetVp | Specifies the preferred virtual processor that will be targeted by messages received through this port. For ports created with *TargetVp* specified as HV\_ANY\_VP, HvPortPropertyPreferredTargetVp allows the caller to supply the preferred virtual processor locality. |

### Connections

Connections are identified by 32-bit IDs. The high 8 bits are reserved and must be zero. All connection IDs are unique within a partition.

typedef union

{

UINT32 AsUint32;

struct

{

UINT32 Id:24;

UINT32 Reserved:8;

};

} HV\_CONNECTION\_ID;

The hypervisor does not ascribe special meaning to any connection IDs.

### Connection Information

The following structure contains the information that must be specified when creating a connection:

typedef struct

{

HV\_PORT\_TYPE PortType;

UINT32 ReservedZ;

union

{

struct

{

UINT64 ReservedZ;

} MessageConnectionInfo;

struct

{

UINT64 ReservedZ;

} EventConnectionInfo;

struct

{

HV\_GPA MonitorAddress;

} MonitorConnectionInfo;

};

} HV\_CONNECTION\_INFO, \*PHV\_CONNECTION\_INFO;

#### Monitored Notification Trigger Group

The monitored notification triggers group structure defines 32 triggers per group. The structure has the following format:

typedef struct

{

UINT64 AsUINT64;

struct

{

UINT32 Pending;

UINT32 Armed;

};

} HV\_MONITOR\_TRIGGER\_GROUP, \*PHV\_MONITOR\_TRIGGER\_GROUP;

The 32 triggers are represented by two related arrays of bits: *Pending* and *Armed*. Setting a trigger bit to 1in the *Pending* array indicates to the hypervisor that the related notification should eventually generate a signal event. The corresponding bit in the *Armed* array should be set to 0 whenever the matching *Pending* bit is modified. The *Armed* bit is used to ensure that a notification is deferred by at least the latency specified for the notification. *Both of these bits must be updated atomically.*

#### Monitored Notification Parameters

Each trigger has a set of associated notification parameters that are used by the hypervisor as inputs to the implicit HvSignalEvent hypercall that the hypervisor invokes when appropriate. The parameter structure has the following format:

typedef struct

{

HV\_CONNECTION\_ID ConnectionId;

UINT16 FlagNumber;

UINT16 ReservedZ;

} HV\_MONITOR\_PARAMETER, \*PHV\_MONITOR\_PARAMETER;

When the hypervisor detects that a monitored notification is pending, it signals the event by making an internal call to the HvSignalEvent hypercall and passing it the *ConnectionID* and *FlagNumber* members. If signaling an event causes an error, the error is discarded; that is, the internal HvSignalEvent call becomes a NOP.

### Monitored Notification Page

Monitored notifications are defined in a *MNF overlay page*, which supports four sets of monitored notification trigger groups. Each individual 32-bit group can be enabled independently using the following structure:

typedef union

{

UINT32 AsUINT32;

struct

{

UINT32 GroupEnable:4;

UINT32 MonitorDisabled:1;

UINT32 ReservedZ:27;

};

} HV\_MONITOR\_TRIGGER\_STATE, \*PHV\_MONITOR\_TRIGGER\_STATE;

GroupEnable and MonitorDisabled are described below.

The MNF overlay page has the following format:

typedef struct

{

HV\_MONITOR\_TRIGGER\_STATE TriggerState;

UINT32 Reserved1;

HV\_MONITOR\_TRIGGER\_GROUP TriggerGroup[4];

UINT8 Reserved2[536];

UINT16 Latency[4][32];

UINT8 Reserved3[256];

HV\_MONITOR\_PARAMETER Parameter[4][32];

UINT8 Reserved4[1984];

} HV\_MONITOR\_PAGE, \*PHV\_MONITOR\_PAGE;

typedef volatile HV\_MONITOR\_PAGE \*PVHV\_MONITOR\_PAGE;

*TriggerState* contains the *GroupEnable* and *MonitorDisabled* flags. The *GroupEnable* flags are an array of 4 bits, each associated with a trigger group. If *GroupEnable[n]* is set to one, the corresponding *TriggerGroup[n]* is enabled. Although the *GroupEnable* flags can be changed at any time, they are intended to be semi-static and are typically used to drain pending notifications during a save or restore process. The hypervisor inspects the group enable flags at varying rates. If they are all disabled (set to zero), the hypervisor might significantly reduce its inspection rate. The hypervisor inspects all of the enabled monitored notifications approximately at the lowest latency value specified for the monitors of each group.

The *MonitorDisabled* flag is set when the hypervisor is temporarily not monitoring the monitor page. If the caller observes this condition, it may trigger the read of the page by calling the HVSignalEvent hypercall.

*TriggerGroup* is an array of four trigger group structures. For details, see section 14.8.9.1.

*Latency* is a hint; suggesting how often the hypervisor should inspect the monitored notifications. The hypervisor might adjust it to be smaller or larger than this value if doing so is either more efficient or to conform to implementation-specific limitations. Latency is specified in 100-nanosecond units.

*Parameter* is an array of notification parameters, one per trigger. The hypervisor can monitor up to 128 notifications in groups of 32. For details, see section 14.8.9.2 .

The *Reservedn* bits are reserved for use by the hypervisor. Changing their value is boundedly undefined.

## Inter-Partition Communication Interfaces

### HvCreatePort

The HvCreatePort hypercall allocates a port object and a set of sixteen (16) message buffers to store queued pending messages.

**Wrapper Interface**

HV\_STATUS

HvCreatePort(

\_\_in HV\_PARTITION\_ID PortPartition,

\_\_in HV\_PORT\_ID PortId,

\_\_in HV\_PARTITION\_ID ConnectionPartition,

\_\_in PCHV\_PORT\_INFO PortInfo

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvCreatePort | | |
|  | Call Code = 0x0057 | |
| 🡆 Input Parameters | | |
| 0 | PortPartition (8 bytes) | |
| 8 | PortId (4 bytes) | Padding (4 bytes) |
| 16 | ConnectionPartition (8 bytes) | |
| 24 | PortTypeInfo (24 bytes) | |
| 32 |
| 40 |

Description

The memory for the port object and the message buffers is allocated from the memory pool of the partition identified by the *PortPartition* parameter.

The specified port ID must be unique among all ports associated with the receive partition.

The *ConnectionPartition* parameter must be specified to prevent other partitions from maliciously connecting to the newly-created port. Only connections from specified partition will be allowed.

Input Parameters

*PortPartition* specifies the partition that is to receive messages or events through this port.

*PortId* specifies an ID that must be unique among all port IDs for this partition. It is used to refer to the port. It is also used to identify the source of a message when it arrives.

*ConnectionPartition* specifies the partition that will be allowed to send messages or events through this port.

*PortTypeInfo* specifies information about the port:

For message ports (*PortType* is HvPortTypeMessage), *PortTypeInfo* is as follows:

|  |  |  |
| --- | --- | --- |
| 24 | PortType (4 bytes)  = HvPortTypeMessage | Padding (4 bytes) |
| 32 | TargetSint (4 bytes) | TargetVp (4 bytes) |
| 40 | RsvdZ (8 bytes) | |

*PortType* specifies HvPortTypeMessage, indicating that this is a message port.

*TargetSint* specifies the synthetic interrupt source to use for this port and may be in the range of 1 through 15. SINT0 is reserved for hypervisor use.

*TargetVp* specifies the virtual processor that will be targeted by messages received through this port. In most cases, a value of HV\_ANY\_VP is recommended. This instructs the hypervisor to route the message to any virtual processor in the target partition.

For event ports (*PortType* is HvPortTypeEvent), *PortTypeInfo* is as follows:

|  |  |  |  |
| --- | --- | --- | --- |
| 24 | PortType (4 bytes)  = HvPortTypeEvent | | Padding (4 bytes) |
| 32 | TargetSint (4 bytes) | | TargetVp (4 bytes) |
| 40 | BaseFlagNumber  (2 bytes) | FlagCount  (2 bytes) | RsvdZ (4 bytes) |

*PortType* specifies HvPortTypeEvent, indicating that this is an event signal port.

*TargetSint* specifies the synthetic interrupt source to use for this port and may be in the range of 1 through 15. SINT0 is reserved for hypervisor use.

*TargetVp* specifies the virtual processor that will be targeted by events received through this port. In most cases, a value of HV\_ANY\_VP is recommended. This instructs the hypervisor to route the event signal to any virtual processor in the target partition.

*BaseFlagNumber* specifies the starting event flag number to be assigned to the port and may be in the range of 0 through 2047.

*FlagCount* specifies the number of event flags, starting at *BaseFlagNumber*, which are to be assigned to the port and cannot be greater than 2048.

For monitor ports (*PortType* is HvPortTypeMonitor), *PortTypeInfo* is as follows:

|  |  |  |
| --- | --- | --- |
| 24 | PortType (4 bytes)  = HvPortTypeMonitor | Padding (4 bytes) |
| 32 | MonitorAddress (8 bytes) | |
| 40 | RsvdZ (8 bytes) | |

*PortType* specifies HvPortTypeMonitor, indicating that this is a monitor port.

*MonitorAddress* specifies the GPA where the monitor overlay page is to be mapped within the port partition.

Output Parameters

None.

Restrictions

The caller must possess the *CreatePort* privilege.

The partitions specified by *PortPartition* and *ConnectionPartition* must be in the “active” state.

The caller must be the parent of the partition specified by *PortPartition* or  
the caller must possess the *CreatePort* privilege and specify its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified *PortPartition.*  2. The caller is the specified *PortPartition,* and the partition has the CreatePort partition privilege. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified port partition ID is invalid. |
| The specified connection partition ID is invalid. |
| The connection and port partition IDs reference the same partition. |
| HV\_STATUS\_INVALID\_PORT\_ID | The specified port ID is already in use. |
| The reserved portion (high 8 bits) of the *PortId* are non-zero. |
| HV\_INVALID\_VP\_INDEX | The specified virtual processor index does not reference a valid VP and is not HV\_ANY\_VP. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified interrupt source index is not in the range 1 through 15 inclusive. |
| The specified port type is invalid. |
| For event ports, the specified base flag number plus the flag count is greater than or equal to HV\_EVENT\_FLAGS\_COUNT. |
| For event ports, the specified flag count is zero. |
| The reserved fields of the *PortTypeInfo* parameter are not set to zero. |
| HV\_STATUS\_INVALID\_ALIGNMENT | The specified monitor overlay page GPA is not page aligned. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified port partition is not in the "active" state. |
| The specified connection partition is not in the "active" state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the caller is insufficient to perform the operation. |
| HV\_STATUS\_NO\_RESOURCES | A required system resource is unavailable or an implementation limit has been reached. |

### HvDeletePort

The HvDeletePort hypercall deallocates a port object and the message buffers associated with it.

**Wrapper Interface**

HV\_STATUS

HvDeletePort(

\_\_in HV\_PARTITION\_ID PortPartition,

\_\_in HV\_PORT\_ID PortId

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvDeletePort | | |
|  | Call Code = 0x0058 | |
| 🡆 Input Parameters | | |
| 0 | PortPartition (8 bytes) | |
| 8 | PortId (4 bytes) | Padding (4 bytes) |

Description

At port deletion, any undelivered messages present in the queue will be lost. Subsequent references to the specified port ID will fail. Also, subsequent attempts to send messages or signal events through a connection to this port will fail.

Messages that arrived through the port may have already been copied to a virtual processor’s SIM page. It is therefore possible for messages with the deleted port’s ID to arrive after HvDeletePort has been called.

Input Parameters

*PortPartition* specifies the partition.

*PortId* specifies the ID of a port that was previously allocated by calling HvCreatePort.

Output Parameters

None.

Restrictions

The caller must possess the *CreatePort* privilege.

The partition specified by *PortPartition* must be in the “active” state.

The caller must be the parent of the partition specified by *PortPartition* or  
the caller must possess the CreatePort privilege and specify its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified partition.  2. The caller is the specified partition, and the partition has the CreatePort partition privilege. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_PORT\_ID | The specified port ID is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvConnectPort

The HvConnectPort hypercall creates a connection to a previously-created port in another partition.

**Wrapper Interface**

HV\_STATUS

HvConnectPort(

\_\_in HV\_PARTITION\_ID ConnectionPartition,

\_\_in HV\_CONNECTION\_ID ConnectionId,

\_\_in HV\_PARTITION\_ID PortPartition,

\_\_in HV\_PORT\_ID PortId,

\_\_in PCHV\_CONNECTION\_INFO ConnectionInfo

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvConnectPort | | |
|  | Call Code = 0x0059 | |
| 🡆 Input Parameters | | |
| 0 | ConnectionPartition (8 bytes) | |
| 8 | ConnectionId (4 bytes) | Padding (4 bytes) |
| 16 | PortPartition (8 bytes) | |
| 24 | PortId (4 bytes) | Padding (4 bytes) |
| 32 | ConnectionInfo (8 bytes) | |

Description

The memory for the connection object is allocated from the connection partition’s memory pool.

The specified connection ID must be unique among all connections associated with the connection partition.

Input Parameters

*ConnectionPartition* specifies the partition that is to send messages or signal events using the connection.

*ConnectionId* specifies an ID that is unique among all connection IDs for this partition. It is used to refer to the connection.

*PortPartition* specifies the partition whose port is being connected to.

*PortId* specifies the ID of a previously-created port within the *PortPartition*.

*ConnectionInfo* specifies information about the connection based upon the type of the target port (the port that the connection is to be associated with):

For message ports (*PortType* is HvPortTypeMessage), *ConnectionInfo* is as follows:

|  |  |
| --- | --- |
| 30 | RsvdZ (8 bytes) |

For event ports (*PortType* is HvPortTypeEvent), *ConnectionInfo* is as follows:

|  |  |
| --- | --- |
| 30 | RsvdZ (8 bytes) |

For monitor ports (*PortType* is HvPortTypeMonitor), *ConnectionInfo* is as follows:

|  |  |
| --- | --- |
| 30 | MonitorAddress (8 bytes) |

*MonitorAddress* specifies the monitor overlay page GPA within the connection partition.

Output Parameters

None.

Restrictions

The caller must possess the *ConnectPort* privilege.

The partitions specified by *ConnectionPartition* and *PortPartition* must be in the “active” state.

The caller must be the parent of the partition specified by *ConnectionPartition* or  
the caller must specify its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified *ConnectionPartition.*  2. The caller is the specified *ConnectionPartition,* and the partition has the ConnectPort partition property. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified connection partition ID is invalid. |
| The specified port partition ID is invalid. |
| The *PortPartition* does not match the *ConnectionPartition* that the port was created with (see section ). |
| HV\_STATUS\_INVALID\_CONNECTION\_ID | The specified connection ID is already in use. |
| The reserved portion (high 8 bits) of the *ConnectionId* are non-zero. |
| HV\_STATUS\_INVALID\_PORT\_ID | The specified port ID does not exist. |
| The *PortPartition* does not have a defined port with the specified port ID. |
| The port specified by the port ID is already connected. |
| The *PortPartition* does not match the *ConnectionPartition* that the port was created with (see section ). |
| The reserved portion (high 8 bits) of the *PortId* are non-zero. |
| HV\_STATUS\_INVALID\_ALIGNMENT | The specified monitor overlay page GPA is not page aligned. |
| HV\_STATUS\_INVALID\_PARAMETER | The *ConnectionInfo* parameter is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified port partition is not in the “active” state. |
| The specified connection partition is not in the “active” state. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the caller is insufficient to perform the operation. |
| HV\_STATUS\_NO\_RESOURCES | A required system resource is unavailable or an implementation limit has been reached. |

### HvGetPortProperty

The HvGetPortProperty hypercall allows an authorized guest to read a property of a port.

Wrapper Interface

HV\_STATUS

HvGetPortProperty(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PORT\_ID PortId,

\_\_in HV\_PORT\_PROPERTY\_CODE PropertyCode,

\_\_out PHV\_PORT\_PROPERTY PropertyValue

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvGetPortProperty | | |
|  | Call Code = 0x005A | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | PortId (4 bytes) | Padding (4 bytes) |
| 16 | PropertyCode (4 bytes) | Padding (4 bytes) |
| 🡄 Output Parameters | | |
| 0 | PropertyValue (8 bytes) | |

Description

Port properties are described in section .

Input Parameters

*PartitionId* specifies the partition that owns the port.

*PortId* specifies a port.

*PropertyCode* specifies the property whose value the caller is interested in retrieving.

Output Parameters

*PropertyValue* returns the property value.

Restrictions

The caller must possess the *CreatePort* privilege.

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId* or  
the caller must specify its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified target partition or the partition itself. |
| The caller does not possess the CreatePort partition privilege. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_PORT\_ID | The specified port ID is invalid. |
| HV\_STATUS\_UNKNOWN\_PROPERTY | The specified property code is not a recognized property. |

### HvSetPortProperty

The HvSetPortProperty hypercall allows an authorized guest to set a property of a port.

Wrapper Interface

HV\_STATUS

HvSetPortProperty(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_PORT\_ID PortId,

\_\_in HV\_PORT\_PROPERTY\_CODE PropertyCode,

\_\_in HV\_PORT\_PROPERTY PropertyValue,

\_\_out PHV\_PORT\_PROPERTY PropertyValue

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvSetPortProperty | | |
|  | Call Code = 0x0070 | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | PortId (4 bytes) | Padding (4 bytes) |
| 16 | PropertyCode (4 bytes) | Padding (4 bytes) |
| 24 | PropertyValue (8 bytes) |  |
| 🡄 Output Parameters | | |
| 0 | PropertyValue (8 bytes) | |

Description

Port properties are described in section .

Input Parameters

*PartitionId* specifies the partition that owns the port.

*PortId* specifies a port.

*PropertyCode* specifies the property whose value the caller is interested in setting.

*PropertyValue* specifies the value to which to set the property.

Output Parameters

*PropertyValue* returns the property value.

Restrictions

The caller must possess the *CreatePort* privilege.

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId* or  
the caller must specify its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified target partition or the partition itself. |
| The caller does not possess the CreatePort partition privilege. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_PORT\_ID | The specified port ID is invalid. |
| HV\_STATUS\_UNKNOWN\_PROPERTY | The specified property code is not a recognized property. |

### HvDisconnectPort

The HvDisconnectPort hypercall severs a connection created by HvConnectPort.

**Wrapper Interface**

HV\_STATUS

HvDisconnectPort(

\_\_in HV\_PARTITION\_ID ConnectionPartition,

\_\_in HV\_CONNECTION\_ID ConnectionId

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvDisconnectPort [fast] | | |
|  | Call Code = 0x005B | |
| 🡆 Input Parameters | | |
| 0 | ConnectionPartition (8 bytes) | |
| 8 | ConnectionId (4 bytes) | Padding (4 bytes) |

Description

When severing a connection, any messages present in the port’s message queue will be unaffected.

Input Parameters

*ConnectionPartition* specifies the partition.

*ConnectionId* specifies the connection ID.

Output Parameters

None.

Restrictions

The caller must possess the *ConnectPort* privilege.

The partition specified by *ConnectionPartition* must be in the “active” state.

The caller must be the parent of the partition specified by *ConnectionPartition* or  
the caller must possess the *ConnectPort* privilege and specify its own partition ID.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | Neither of the following is true:  1. The caller is the parent of the specified partition.  2. The caller is the specified partition, and the partition has the ConnectPort partition property. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_INVALID\_CONNECTION\_ID | The specified connection ID is invalid. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvPostMessage

The HvPostMessage hypercall attempts to post (that is, send asynchronously) a message to the specified connection, which has an associated destination port.

**Wrapper Interface**

HV\_STATUS

HvPostMessage(

\_\_in HV\_CONNECTION\_ID ConnectionId,

\_\_in HV\_MESSAGE\_TYPE MessageType,

\_\_in UINT32 PayloadSize,

\_\_in\_ecount(PayloadSize)

PCVOID Message

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvPostMessage | | |
|  | Call Code = 0x005C | |
| 🡆 Input Parameters | | |
| 0 | ConnectionId (4 bytes) | Padding (4 bytes) |
| 8 | MessageType (4 bytes) | PayloadSize (4 bytes) |
| 16 | Message[0] (8 bytes) | |
| … | ... | |
| 248 | Message[29] (8 bytes) | |

Description

If the message is successfully posted, then it will be queued for delivery to a virtual processor within the partition associated with the port.

For details about message delivery, see section .

Input Parameters

*ConnectionId* specifies the ID of the connection created by calling HvConnectPort.

*MessageType* specifies the message type that will appear within the message header. The caller can specify any 32-bit message type whose most significant bit is cleared, with the exception of zero. Message types with the high bit set are reserved for use by the hypervisor.

*PayloadSize* specifies the number of bytes that are included in the message.

*Message* specifies the payload of the message—up to 240 bytes total. Only the first *n* bytes are actually sent to the destination partition, where *n* is provided in the *PayloadSize* parameter.

Output Parameters

None.

Restrictions

The partition that is the target of the connection must be in the “active” state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition does not possess the *PostMessages* privilege. |
| HV\_STATUS\_INVALID\_CONNECTION\_ID | The specified connection ID is invalid. |
| HV\_STATUS\_INVALID\_PORT\_ID | The port associated with the specified connection has been deleted. |
| The port associated with the specified connection belongs to a partition that is not in the “active” state. |
| The port associated with the specified connection is not a "message" type port. |
| HV\_STATUS\_INVALID\_PARAMETER | The most significant bit of the specified message type is set. |
| The *MessageType* parameter specifies a value of zero. |
| The specified payload size exceeds 240 bytes. |
| HV\_STATUS\_INSUFFICIENT\_BUFFERS | The port has no available guest message buffers. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The target VP no longer exists or there are no available VPs to which the message can be posted. |
| HV\_STATUS\_INVALID\_SYNIC\_STATE | The target VP’s SynIC is disabled and cannot accept posted messages. For ports targeted at HV\_ANY\_VP, this indicates that the SynIC of all of the partition’s VPs are disabled. |
| The target VP’s SIM page is disabled. For ports targeted at HV\_ANY\_VP, this indicates that the SIM page of all of the partition’s VPs are disabled. |

### HvSignalEvent

The HvSignalEvent hypercall signals an event in a partition that owns the port associated with the specified connection.

Wrapper Interface

HV\_STATUS

HvSignalEvent(

\_\_in HV\_CONNECTION\_ID ConnectionId,

\_\_in UINT16 FlagNumber

);

Native Interface

|  |  |  |  |
| --- | --- | --- | --- |
| HvSignalEvent | | | |
|  | Call Code = 0x005D | | |
| 🡆 Input Parameter Header | | | |
| 0 | ConnectionId  (4 bytes) | FlagNumber  (2 bytes) | RsvdZ  (2 bytes) |

Description

The event is signaled by setting a bit within the SIEF page of one of the receive partition’s virtual processors.

The caller specifies a relative flag number. The actual SIEF bit number is calculated by the hypervisor by adding the specified flag number to the base flag number associated with the port.

Input Parameters

*ConnectionId* specifies the ID of the connection.

*FlagNumber* specifies the relative index of the event flag that the caller wants to set within the target SIEF area. This number is relative to the base flag number associated with the port.

Output Parameters

None.

Restrictions

The partition that is the target of the connection must be in the “active” state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition does not possess the *SignalEvents* privilege. |
| HV\_STATUS\_INVALID\_CONNECTION\_ID | The specified connection ID is invalid. |
| HV\_STATUS\_INVALID\_PORT\_ID | The port associated with the specified connection has been deleted. |
| The port associated with the specified connection belongs to a partition that is not in the “active” state. |
| The port associated with the specified connection is not an "event" type port. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified flag number is greater than or equal to the port’s flag count. |
| HV\_STATUS\_INVALID\_VP\_INDEX | The target VP no longer exists or there are no available VPs to which the message can be posted. |
| HV\_STATUS\_INVALID\_SYNIC\_STATE | The target VP’s SynIC is disabled and cannot accept signaled events. For ports targeted at HV\_ANY\_VP, this indicates that the SynIC of all of the partition’s VPs are disabled. |
| The target VP’s SIEF page is disabled. For ports targeted at HV\_ANY\_VP, this indicates that the SIEF page of all of the partition’s VPs are disabled. |
| The target SINTx is masked. |

# 

# Timers

## Overview

### Timer Services

The hypervisor provides simple timing services. These are based on a constant-rate reference time source (typically the ACPI timer on x64 systems).

The following timer services are provided:

* A per-partition reference time counter.
* Four synthetic timers per virtual processor. Each synthetic timer is a single-shot or periodic timer that delivers a message when it expires.
* One virtual APIC timer per virtual processor.
* Two timer assists: an emulated periodic timer and a PM Timer assist.
* A partition reference time enlightenment, based on the host platform’s support for an Invariant Time Stamp Counter (iTSC).

### Reference Counter

The hypervisor maintains a per-partition reference time counter. It has the characteristic that successive accesses to it return strictly monotonically increasing (time) values as seen by any and all virtual processors of a partition. Furthermore, the reference counter is rate constant and unaffected by processor or bus speed transitions or deep processor power savings states. A partition’s reference time counter is initialized to zero when the partition is created. The reference counter for all partitions count at the same rate, but at any time, their absolute values will typically differ because partitions will have different creation times.

The reference counter continues to count up as long as at least one virtual processor is not explicitly suspended.

### Synthetic Timers

Synthetic timers provide a mechanism for generating an interrupt after some specified time in the future. Both one-shot and periodic timers are supported. A synthetic timer sends a message to a specified SynIC SINTx (synthetic interrupt source) upon expiration.

The hypervisor guarantees that a timer expiration signal will never be delivered before the expiration time. The signal may arrive any time after the expiration time.

### Periodic Timers

The hypervisor attempts to signal periodic timers on a regular basis.

For example, if a timer has a requested period of 1ms, here is the idealized schedule for timer expiration notifications:

Time (1ms increments)

Timer Expiration Signals

However, if the virtual processor used to signal the expiration is not available, some of the timer expirations may be delayed. A virtual processor may be unavailable because it is suspended (for example, during intercept handling) or because the hypervisor’s scheduler decided that the virtual processor should not be scheduled on a logical processor (for example, because another virtual processor is using the logical processor or the virtual processor has exceeded its quota).

The shaded portions of the following diagram show periods of inactivity during which a periodic timer expiration signal could not be delivered. Consequently, the signal is deferred until the virtual processor becomes available.

If a virtual processor is unavailable for a sufficiently long period of time, a full timer period may be missed. In this case, the hypervisor uses one of two techniques. The first technique involves timer period modulation, in effect shortening the period until the timer “catches up”.

The following diagram shows the period modulation technique.

If a significant number of timer signals have been missed, the hypervisor may be unable to compensate by using period modulation. In this case, some timer expiration signals may be skipped completely.

For timers that are marked as *lazy*, the hypervisor uses a second technique for dealing with the situation in which a virtual processor is unavailable for a long period of time. In this case, the timer signal is deferred until this virtual processor is available. If it doesn’t become available until shortly before the next timer is due to expire, it is skipped entirely.

The following diagram shows the lazy timer technique.

### Periodic Timer Assist

The parent partition may enable a periodic timer that is to be applied to a child partition. A period defines the rate of the interrupts, and a local APIC destination specifies which virtual processors are to be interrupted. The local APIC destination is usually defined by the I/O APIC redirection entry associated with the assisted timer source. The period is established by the parent partition by programming the periodic timer assist. Using this timer assist will reduce the cost of interrupt delivery and an improve interrupt accuracy.

The periodic timer assist is managed using two partition properties, HvPartitionPropertyEmulatedTimerPeriod and HvPartitionPropertyEmulatedTimerControl. Both properties default to all zeroes.

The timer period property specifies the time in 100 nanosecond units and is defined as follows:

// Timer Period

typedef union

{

UINT64 AsUINT64;

HV\_NANO100\_DURATION Period;

} HV\_EMULATED\_TIMER\_PERIOD, \*PHV\_EMULATED\_TIMER\_PERIOD;

The local APIC destination is encoded in the timer control and is defined as follows:

// Periodic Timer Control

typedef union

{

UINT64 AsUINT64;

struct

{

UINT32 Vector:8;

UINT32 DeliveryMode:3;

UINT32 LogicalDestinationMode:1;

UINT32 Enabled:1;

UINT32 Reserved1:19;

UINT32 Reserved2:24;

UINT32 Mda[[3]](#footnote-3):8;

};

} HV\_EMULATED\_TIMER\_CONTROL, \*PHV\_EMULATED\_TIMER\_CONTROL;

The defined (named) fields of the Periodic Timer Control conform to those used by the local APIC’s Interrupt Command Register (ICR). When the timer period property is set to zero, the timer control *Enabled* flag will be cleared, disabling the timer assist.

Normally, HvPartitionPropertyEmulatedTimerPeriod is set, and then HvPartitionPropertyEmulatedTimerControl updated to enable the assist. If those operations are done in the reverse order, the pending timer with the old timer period will be removed and a new timer with the new period inserted. Likewise, if the values in HvPartitionPropertyEmulatedTimerControl are changed without disabling the timer, the new values will take effect immediately.

The periodic timer assist acts in the same way as the non-lazy periodic timer described in the previous section, i.e. it will try to “catch up” if timer ticks are missed.

### PM Timer Assist

The parent partition may request the hypervisor to assist the child’s use of the ACPI PM timer. The parent specifies an I/O port and whether the child’s ACPI PM Timer is 24- or 32-bits wide. When the child partition performs an IN from the specified port, the hypervisor will directly provide the value of the PM timer based upon the partition’s reference time. The returned ACPI PM Timer value is computed as:

ACPI\_PM\_Timer = Modulus(Partition\_Reference\_Time \* 0.3579545, 2\*\*[24|32]);

This assist supports only IN operations on the I/O port (note that INS is not supported). Attempts to OUT to the port, to IN from any of the three subsequent I/O ports (that is, ports n+1, n+2 or n+3) or to IN from any of the three preceding I/O ports (that is, ports n-1, n-2 or n-3) will follow the rules for I/O port accesses as described in section .

If both an intercept and the PM timer assist are installed for the same I/O port, the assist will take precedence for the IN operation. The hypervisor will generate an intercept message for all OUT operations.  
  
The caller specifies the PM Timer property’s port and whether the value returned is 24 or 32 bits wide as follows:

// ACPI PM Timer

typedef union

{

UINT64 AsUINT64;

struct

{

UINT64 Port:16;

UINT64 Width24:1;

UINT64 Enabled:1;

UINT64 ReservedZ1:14;

UINT64 ReservedZ2:32;

};

} HV\_PM\_TIMER\_INFO, \*PHV\_PM\_TIMER\_INFO;

*Port* is the I/O port that the reference timer assist is to use.

*Width24* indicates the width of the timer data. When set, the data is 24 bits wide. When clear, it is 32 bits wide.

*Enabled* is used to enable or disable the assist.

### Ordering of Timer Expirations

Synthetic and virtualized timers generate interrupts at or near their designated expiration time. Due to hardware and other scheduling interactions, interrupts could potentially be delayed. No ordering may be assumed between any set of timers.

### Timer Expiration Messages

For details about timer expiration messages, see section .

### Partition Reference Time Enlightenment

The partition reference time enlightenment presents a reference time source to which does not require an intercept into the hypervisor. This enlightenment is available only when the underlying platform provides support of an invariant processor Time Stamp Counter (TSC), or iTSC. In such platforms, the processor TSC frequency remains constant irrespective of changes in the processor’s clock frequency due to the use of power management states such as ACPI processor performance states, processor idle sleep states (ACPI C-states), etc.

The partition reference time enlightenment uses a virtual TSC value, an offset and a multiplier to enable a guest partition to compute the normalized reference time since partition creation, in 100nS units. The mechanism also allows a guest partition to atomically compute the reference time when the guest partition is migrated to a platform with a different TSC rate, and provides a fall-back mechanism to support migration to platforms without the constant rate TSC feature.

This facility is not intended to be used a source of wall clock time, since the reference time computed using this facility will appear to stop during the time that a guest partition is saved until the subsequent restore.

## Reference Counter MSR

A partition’s reference counter is accessed through a partition-wide MSR.

|  |  |  |
| --- | --- | --- |
| MSR Address | Register Name | Function |
| 0x40000020 | HV\_X64\_MSR\_TIME\_REF\_COUNT | Time reference count (partition-wide) |

### Reference Counter MSR

| 63:0 |
| --- |
| Count |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:0 | Count—Partition’s reference counter value in 100 nanosecond units | Read-only |

When a partition is created, the value of the TIME\_REF\_COUNT MSR is set to 0x0000000000000000. This value cannot be modified by a virtual processor. Any attempt to write to it results in a #GP fault.

## Synthetic Timer MSRs

Synthetic timers are configured by using model-specific registers (MSRs) associated with each virtual processor. Each of the four synthetic timers has an associated pair of MSRs.

| MSR address | Register name | Function | |
| --- | --- | --- | --- |
| 0x400000B0 | HV\_X64\_MSR\_STIMER0\_CONFIG | | Configuration register for synthetic timer 0 |
| 0x400000B1 | HV\_X64\_MSR\_STIMER0\_COUNT | | Expiration time or period for synthetic timer 0 |
| 0x400000B2 | HV\_X64\_MSR\_STIMER1\_CONFIG | | Configuration register for synthetic timer 1 |
| 0x400000B3 | HV\_X64\_MSR\_STIMER1\_COUNT | | Expiration time or period for synthetic timer 1 |
| 0x400000B4 | HV\_X64\_MSR\_STIMER2\_CONFIG | | Configuration register for synthetic timer 2 |
| 0x400000B5 | HX\_X64\_MSR\_STIMER2\_COUNT | | Expiration time or period for synthetic timer 2 |
| 0x400000B6 | HV\_X64\_MSR\_STIMER3\_CONFIG | | Configuration register for synthetic timer 3 |
| 0x400000B7 | HV\_X64\_MSR\_STIMER3\_COUNT | | Expiration time or period for synthetic timer 3 |

### Synthetic Timer Configuration Register

| 63:20 | 19:16 | 15:4 | 3 | 2 | 1 | 0 |
| --- | --- | --- | --- | --- | --- | --- |
| RsvdZ | SINTx | RsvdZ | Auto  Enable | Lazy | Periodic | Enable |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:20 | RsvdZ (value should be set to zero) | Read/write |
| 19:16 | SINTx—synthetic interrupt source | Read/write |
| 15:3 | RsvdZ (value should be set to zero) | Read/write |
| 3 | AutoEnable—set if writing the corresponding counter implicitly causes the counter to be enabled, cleared if not | Read/write |
| 2 | Lazy—set if timer is lazy, cleared if not | Read/write |
| 1 | Periodic—set if timer is periodic, cleared if one-shot | Read/write |
| 0 | Enable—set if timer is enabled | Read/write |

When a virtual processor is created and reset, the value of all HV\_X64\_MSR\_STIMERx\_CONFIG (synthetic timer configuration) registers is set to 0x0000000000000000. Thus, all synthetic timers are disabled by default.

If *AutoEnable* is set, then writing a non-zero value to the corresponding count register will cause Enable to be set and activate the counter. Otherwise, Enable should be set after writing the corresponding count register in order to activate the counter. For information about the Count register, see the following section.

When a one-shot timer expires, it is automatically marked as disabled. Periodic timers remain enabled until explicitly disabled.

If a one-shot is enabled and the specified count is in the past, it will expire immediately.

It is not permitted to set the SINTx field to zero for an enabled timer. If attempted, the timer will be marked disabled (that is, bit 0 cleared) immediately.

Writing the configuration register of a timer that is already enabled may result in undefined behavior. For example, merely changing a timer from one-shot to periodic may not produce what is intended. Timers should always be disabled prior to changing any other properties.

### Synthetic Timer Count Register

| 63:0 |
| --- |
| Count |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:0 | Count—expiration time for one-shot timers, duration for periodic timers | Read/write |

The value programmed into the Count register is a time value measured in 100 nanosecond units. Writing the value zero to the Count register will stop the counter, thereby disabling the timer, independent of the setting of *AutoEnable* in the configuration register.

Note that the Count register is permitted to wrap. Wrapping will have no effect on the behavior of the timer, regardless of any timer property.

For one-shot timers, it represents the absolute timer expiration time. The timer expires when the reference counter for the partition is equal to or greater than the specified count value.

For periodic timers, the count represents the period of the timer. The first period begins when the synthetic timer is enabled.

## Partition Reference Time Enlightenment

The hypervisor provides a partition-wide virtual reference TSC page which is overlaid on the partition’s GPA space. A partition’s reference time stamp counter page is accessed through the Reference TSC MSR. A partition which possesses the AccessPartitionReferenceTsc privilege may access the reference TSC MSR.

### Reference Time Stamp Counter (TSC) Page MSR

A guest wishing to access its reference TSC page must use the following model-specific register (MSR).

|  |  |  |
| --- | --- | --- |
| MSR Address | Register Name | Function |
| 0x40000021 | HV\_X64\_MSR\_REFERENCE\_TSC | Time reference count (partition-wide) |

The format of the Reference TSC MSR is as follows:

| 63:12 | 11:1 | 0 |
| --- | --- | --- |
| GPA Page Number | RsvdP | Enable |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:12 | GPA Page Number | Read/write |
| 11:1 | RsvdP (value should be preserved) | Read/write |
| 0 | Enable—set if reference TSC is enabled | Read/write |

At the guest partition creation time, the value of the reference TSC MSR is 0x0000000000000000. Thus, the reference TSC page is disabled by default. The guest must enable the reference TSC page by setting bit 0. If the specified base address is beyond the end of the partition’s GPA space, the reference TSC page will not be accessible to the guest. When modifying the register, guests should preserve the value of the reserved bits (1 through 11) for future compatibility.

Restrictions

The caller must possess the AccessPartitionReferenceTsc privilege.

### Format of the Reference TSC Page

The reference TSC page is defined using the following structure:

typedef struct \_HV\_REFERENCE\_TSC\_PAGE

{

volatile UINT32 TscSequence;

UINT32 Reserved1;

volatile UINT64 TscScale;

volatile INT64 TscOffset;

UINT64 Reserved2[509];

} HV\_REFERENCE\_TSC\_PAGE, \*PHV\_REFERENCE\_TSC\_PAGE;

### Partition Reference TSC Mechanism

The partition reference time is computed by the following formula:

ReferenceTime = ((VirtualTsc \* TscScale) >> 64) + TscOffset

The multiplication is a 64 bit multiplication, which results in a 128 bit number which is then shifted 64 times to the right to obtain the high 64 bits.

#### TscScale

The TscScale value is used to adjust the Virtual TSC value across migration events to mitigate TSC frequency changes from one platform to another.

#### TscSequence

The TscSequence value is used to synchronize access to the enlightened reference time if the scale and/or the offset fields are changed during save/restore or live migration. This field serves as a sequence number which is incremented whenever the scale and/or the offset fields are modified. The valid values of this field range from 0 to 0xFFFFFFFE.

A special value of 0xFFFFFFFF is used to indicate that this facility is no longer a reliable source of reference time and the VM must fall back to a different source (e.g. virtual PM timer).

#### Reference TSC during Save/Restore and Migration

To address migration scenarios to physical platforms which do not support iTSC, the TscSequence field is used. In the event a guest partition is migrated from an iTSC capable host to a non-iTSC capable host, the hypervisor sets TscSequence to the special value of 0xFFFFFFFF, which directs the guest operating system to fall back to a different clock source (the virtual PM timer).The recommended code for computing the partition reference time using this enlightenment is shown below:

Repeat:

Sequence = Stats->ReferenceTscSequence;

Tsc = RDTSC();

Scale = Stats->ReferenceTscScale;

Offset = Stats->ReferenceTscOffset;

ReferenceTime = (Tsc \* Scale) >> 64 + Offset;

If (Stats->Sequence == Sequence)

{

return ReferenceTime;

}

If (Stats->Sequence != 0xFFFFFFFF)

{

goto Repeat; // Fall back to a different method to get

reference time

}

# Message Formats

## Overview

The hypervisor supports a simple message-based inter-partition communication mechanism. Messages can be sent by the hypervisor to a partition or can be sent from one partition to another. This section describes all of the messages sent by the hypervisor.

Each message has a message type, a source partition, and a message payload. For a complete list of message types, see chapter 14The format of the message payload depends on the message type.

The messages sent by the hypervisor fall into the following categories:

Memory access messages (unmapped GPA, GPA access violations, and so on.)

Processor intercepts

Error messages

Timer notifications

Event log events

## Message Data Types

Intercept messages are delivered by the SynIC. For a description of SynIC messages, including the message header layout, see chapter 14**.**

### Message Header

Each message begins with a common message header. The significant fields are the *MessageType*, *PayloadSize* and the *OriginationId* (the source of the message). It is important to note that the payload size reflects only the size of the data and does not include the message header. The message header is described in section .

### Intercept Message Header

All x64 memory access messages and processor intercept messages contain a common payload header. This header contains information about the state of the virtual processor at the time of the intercept, making it easier for the recipient of the message to complete the intercepted instruction in software.

typedef struct

{

HV\_VP\_INDEX VpIndex;

UINT8 InstructionLength;

HV\_INTERCEPT\_ACCESS\_TYPE\_MASK InterceptAccessType;

HV\_X64\_VP\_EXECUTION\_STATE ExecutionState;

HV\_X64\_SEGMENT\_REGISTER CsSegment;

UINT64 Rip;

UINT64 Rflags;

} HV\_X64\_INTERCEPT\_MESSAGE\_HEADER;

*VpIndex* indicates the index of the virtual processor that generated the intercept.

*InstructionLength* indicates the byte length of the instruction that generated the intercept. If the instruction length is unknown, a length of zero is reported, and the recipient of the message must fetch and decode the instruction to determine its length. The hypervisor guarantees that it will fill in the correct instruction length for CPUID, I/O port, and MSR intercepts.

*InterceptAccessType* indicates the access type (read, write, or execute) of the event that triggered the intercept.

*ExecutionState* provides miscellaneous information about the virtual processor’s state at the time the intercept was triggered.

*CsSegment* provides information about the code segment at the time the intercept was triggered.

*Rip* provides the instruction pointer at the time the intercept was triggered.

*Rflags* provides the flags register at the time the intercept was triggered.

### VP Execution State

The execution state is a collection of flags that specify miscellaneous states of the virtual processor.

typedef struct

{

UINT16 Cpl:2;

UINT16 Cr0Pe:1;

UINT16 Cr0Am:1;

UINT16 EferLma:1;

UINT16 DebugActive:1;

UINT16 InterruptionPending:1;

UINT16 Reserved:9;

} HV\_X64\_VP\_EXECUTION\_STATE;

*Cpl* indicates the current privilege level at the time of the intercept. Real mode has an implied CPL of 0, and v86 has an implied CPL of 3. In other modes, the CPL is defined by the low-order two bits of the code segment (CS).

*Cr0Pe* indicates whether the processor is executing within protected mode.

*Cr0Am* indicates whether alignment must be checked for non-privileged accesses.

*EferLma* indicates whether the processor is executing within long mode (64-bit mode).

*DebugActive* indicates that one or more debug registers are marked as active, so the recipient of the message may need to perform additional work to correctly emulate the behavior of the debug breakpoint facilities.

*InterruptionPending* indicates that the intercept was generated while delivering an interruption. The interruption is held pending and, unless removed, will be re-delivered when the virtual processor is resumed. For a description of the *Pending Interruption* register, see section .

### I/O Port Access Information

On x64 platforms, I/O port access messages include a collection of flags that provide information about the memory access.

typedef struct

{

UINT8 AccessSize:3;

UINT8 StringOp:1;

UINT8 RepPrefix:1;

UINT8 Reserved:3;

} HV\_X64\_IO\_PORT\_ACCESS\_INFO;

*AccessSize* indicates the size of the access. The following encodings are used: 001b = 8 bits; 010b = 16 bits; 100b = 32 bits. All other combinations are reserved.

*StringOp* indicates that the instruction is a string form (INS or OUTS).

*RepPrefix* indicates that the instruction has a “rep” prefix. This flag is used only for string operations.

### Exception Information

On x64 platforms, exception intercept messages include a collection of flags that provide information about the exception.

typedef struct

{

UINT8 ErrorCodeValid:1;

UINT8 Reserved:7;

} HV\_X64\_EXCEPTION\_INFO;

*ErrorCodeValid* indicates that the error code field in the exception message is valid.

### Memory Access Flags

Memory intercept messages include a collection of flags that provide information about the intercept.

typedef struct

{

UINT8 GvaValid:1;

UINT8 Reserved:7;

} HV\_X64\_MEMORY\_ACCESS\_INFO;

*GvaValid* indicates that the *Gva* field of the memory access message contains a valid guest virtual address.

## Memory Access Messages

Memory access messages are sent in response to certain memory accesses performed by a virtual processor or performed on its behalf by the hypervisor. Both memory access messages (unmapped GPA and GPA access violations) share the same format.

|  |  |  |  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | | | Rsvd (3 bytes) | | | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | | | | | |
| Memory Access Payload | 16 | VpIndex (4 bytes) | | | InstLen  (1 byte) | Access  Type  (1 byte) | ExecutionState  (2 bytes) | |
| 24 | CsSegment[0] (8 bytes) | | | | | | |
| 32 | CsSegment[1] (8 bytes) | | | | | | |
| 40 | Rip (8 bytes) | | | | | | |
| 48 | Rflags (8 bytes) | | | | | | |
| 56 | Rsvd  (2 bytes) | Flags  (1 byte) | InstByteCount  (1 byte) | CacheType (4 bytes) | | | |
| 64 | Gva (8 bytes) | | | | | | |
| 72 | Gpa (8 bytes) | | | | | | |
| 80 | InstructionBytes[0] | | | | | | |
| 88 | InstructionBytes[1] | | | | | | |
| 96 | DsSegment[0] (8 bytes) | | | | | | |
| 104 | DsSegment[1] (8 bytes) | | | | | | |
| 112 | SsSegment[0] (8 bytes) | | | | | | |
| 120 | SsSegment[1] (8 bytes) | | | | | | |
| 128 | Rax (8 bytes) | | | | | | |
| … | ... | | | | | | |
| 248 | R15 (8 bytes) | | | | | | |

*Flags* is of type HV\_X64\_MEMORY\_ACCESS\_FLAGS and provides information about the intercept.

*InstByteCount* indicates how many instruction bytes have been provided in the *InstructionBytes* fields. Valid values are in the range 0 through 16.

*CacheType* is of type HV\_CACHE\_TYPE (see section ). When *GvaValid* is set, this field is the cache type for the page causing the access intercept obtained from the guest’s page table. When *GvaValid* is clear, the intercept was caused by an access to guest pages table pages and this field will be HvCacheTypeX64WriteBack.

*Gva* indicates the guest virtual address of the access that was intercepted. This field is valid if the bit *GvaValid* is set in *Flags*. GvaValid will not be set if the memory intercept was caused by an access to guest page table pages.

*Gpa* indicates the guest physical address of the access that was intercepted. If the intercepted instruction was attempting to perform a multi-byte memory access that crossed page boundaries, this value will indicate the lowest address that resulted in an intercept.

*InstructionBytes* includes up to 16 bytes from the instruction stream starting at the current CS:RIP. The number of valid bytes is specified by *InstByteCount*. If the access was due to the delivery of an interruption (as indicated by the *InterruptionPending* bit in the *ExecutionState* field) then the message will not include any instruction bytes. In other cases, the hypervisor may fetch only up to the next page boundary or up to the end of the code segment resulting in an instruction byte count that is less than 16.

*DsSegment* provides the selector, base, limit, and flags for the current data segment (DS).

*SsSegment* provides the selector, base, limit, and flags for the current stack segment (SS).

*Rax...R15* provides the current values of the virtual processor’s sixteen 64-bit general purpose registers.

### Unmapped GPA Message

An “unmapped GPA” message is delivered by the hypervisor when a virtual processor (or the hypervisor on its behalf) accesses a GPA page for which no mapping has been indicated.

### GPA Access Violation Message

A “GPA access violation” message is delivered by the hypervisor when a virtual processor (or the hypervisor on its behalf) accesses a GPA page and the access type is not allowed by the current mapping.

## Timer Messages

### Timer Expiration Message

Timer expiration messages are sent when a timer event fires.

|  |  |  |  |  |
| --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | PayloadSize  (1 byte) |
| 8 | Rsvd (8 bytes) | | |
| Timer Expiration Payload | 16 | TimerIndex (4 bytes) | Rsvd (4 bytes) | |
| 24 | ExpirationTime (8 bytes) | | |
| 32 | DeliveryTime (8 bytes) | | |

*TimerIndex* is the index of the synthetic timer (0 through 3) that generated the message. This allows a client to configure multiple timers to use the same interrupt vector and differentiate between their messages.

*ExpirationTime* is the expected expiration time of the timer measured in 100-nanosecond units by using the time base of the partition’s reference time counter. Note that the expiration message may arrive after the expiration time.

*DeliveryTime* is the time when the message is placed into the respective message slot of the SIM page. The time is measured in 100-nanosecond units based on the partition’s reference time counter.

## Processor Event Messages

Processor event messages are sent in response to certain actions performed by a virtual processor.

### CPUID Intercept Message

A “CPUID intercept” message is delivered by the hypervisor when a virtual processor executes a CPUID instruction and the parent has installed an intercept on such instructions.

| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | | | PayloadSize  (1 byte) |
| --- | --- | --- | --- | --- | --- | --- |
| 8 | SourcePartition (8 bytes) | | | | |
| CPUID Intercept Payload | 16 | VpIndex (4 bytes) | InstLen  (1 byte) | Access Type  (1 byte) | ExecutionState  (2 bytes) | |
| 24 | CsSegment[0] (8 bytes) | | | | |
| 32 | CsSegment[1] (8 bytes) | | | | |
| 40 | Rip (8 bytes) | | | | |
| 48 | Rflags (8 bytes) | | | | |
| 56 | Rax (8 bytes) | | | | |
| 64 | Rcx (8 bytes) | | | | |
| 72 | Rdx (8 bytes) | | | | |
| 80 | Rbx (8 bytes) | | | | |
| 88 | DefaultResultRax (8 bytes) | | | | |
| 96 | DefaultResultRcx (8 bytes) | | | | |
| 104 | DefaultResultRdx (8 bytes) | | | | |
| 112 | DefaultResultRbx (8 bytes) | | | | |

*Rax-Rbx* provides the values in the corresponding registers when the CPUID instruction is executed.

*DefaultResultRax*-*DefaultResultRbx* provides the default return values that the hypervisor would have returned in response to the CPUID instruction if the intercept had not been requested. The intercept handler is free to override these default values.

### MSR Intercept Message

A “MSR intercept” message is delivered by the hypervisor when a virtual processor executes a RDMSR or WRMSR instruction and the parent has installed an intercept on the specified MSR.

|  |  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | | | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | | | |
| MSR Intercept Payload | 16 | VpIndex (4 bytes) | InstLen  (1 byte) | AccessType  (1 byte) | ExecutionState  (2 bytes) | |
| 24 | CsSegment[0] (8 bytes) | | | | |
| 32 | CsSegment[1] (8 bytes) | | | | |
| 40 | Rip (8 bytes) | | | | |
| 48 | Rflags (8 bytes) | | | | |
| 56 | MsrNumber (4 bytes) | Rsvd (4 bytes) | | | |
| 64 | Rdx (8 bytes) | | | | |
| 72 | Rax (8 bytes) | | | | |

*MsrNumber* provides the value in the RCX register that indicates the index of the MSR being accessed.

*Rdx* provides the value in the RDX register. For writes, it represents the top half of the value being written to the MSR.

*Rax* provides the value in the RAX register. For writes, it represents the bottom half of the value being written to the MSR.

### I/O Port Intercept Message

An “I/O Port Intercept” message is delivered by the hypervisor when a virtual processor executes an IN, OUT, INS, or OUTS instruction and the parent has installed an intercept on one of the accessed I/O ports. A multi-byte I/O port access is treated as though it accesses multiple consecutive ports. If intercepts have been installed for any of the ports within the range, an intercept is generated.

|  |  |  |  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | | | Rsvd (3 bytes) | | | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | | | | | |
| IO Port Intercept Payload | 16 | VpIndex (4 bytes) | | | InstLen  (1 byte) | Access Type  (1 byte) | ExecutionState  (2 bytes) | |
| 24 | CsSegment[0] (8 bytes) | | | | | | |
| 32 | CsSegment[1] (8 bytes) | | | | | | |
| 40 | Rip (8 bytes) | | | | | | |
| 48 | Rflags (8 bytes) | | | | | | |
| 56 | PortNum  (2 bytes) | Access Info  (1 byte) | InstByteCount  (1 byte) | Rsvd (4 bytes) | | | |
| 64 | Rax (8 bytes) | | | | | | |
| 72 | InstructionBytes[0] (8 bytes) | | | | | | |
| 80 | InstructionBytes[1] (8 bytes) | | | | | | |
| 88 | DsSegment[0] (8 bytes) | | | | | | |
| 96 | DsSegment[1] (8 bytes) | | | | | | |
| 104 | EsSegment[0] (8 bytes) | | | | | | |
| 112 | EsSegment[1] (8 bytes) | | | | | | |
| 120 | Rcx (8 bytes) | | | | | | |
| 128 | Rsi (8 bytes) | | | | | | |
| 136 | Rdi (8 bytes) | | | | | | |

*PortNum* provides the number of the I/O port being accessed.

*AccessInfo* is of type HV\_X64\_IO\_PORT\_ACCESS\_INFO and indicates the size of the access, whether the intercepted instruction was a string operation, and whether a “rep” prefix was specified.

*InstByteCount* indicates how many instruction bytes have been provided in the InstructionBytes fields. Valid values are in the range 0 through 16. This field is valid only for string operations. It is set to zero and should be ignored for simple (non-string) I/O port accesses.

*Rax* provides the value of the Rax register. For OUT instructions, it represents the value written to the I/O port. For IN instructions, it can be used by an intercept handler to insert the input value into AL, AX, or EAX and compute the final value of RAX.

The remaining fields (offsets 72 and beyond) are applicable only for string operations. They are set to zero and should be ignored for simple (non-string) I/O port accesses.

*InstructionBytes* includes up to 16 bytes from the instruction stream starting at the current CS:RIP. The number of valid bytes is specified by *InstByteCount*. The hypervisor may only fetch up to the next page boundary resulting in an instruction byte count that is less than 16.

*DsSegment* provides the selector, base, limit, and flags for the current data segment (DS).

*EsSegment* provides the selector, base, limit, and flags for the current extended segment (ES).

*Rcx, Rsi,* and *Rdi* provide the values of these three registers.

### Exception Intercept Message

An “exception intercept” message is delivered by the hypervisor when a virtual processor generates an exception.

|  |  |  |  |  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | | | | Rsvd (3 bytes) | | | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | | | | | | |
| Exception Intercept Payload | 16 | VpIndex (4 bytes) | | | | InstLen = 0  (1 byte) | AccessType  (1 byte) | ExecutionState  (2 bytes) | |
| 24 | CsSegment[0] (8 bytes) | | | | | | | |
| 32 | CsSegment[1] (8 bytes) | | | | | | | |
| 40 | Rip (8 bytes) | | | | | | | |
| 48 | Rflags (8 bytes) | | | | | | | |
| 56 | Exception Vector  (2 bytes) | Exception Flags (1 byte) | InstrByte Count  (1 byte) | ErrorCode (4 bytes) | | | | |
| 64 | ExceptionParameter (8 bytes) | | | | | | | |
| 72 | Rsvd (8 bytes) | | | | | | | |
| 80 | InstructionBytes[0] (8 bytes) | | | | | | | |
| 88 | InstructionBytes[1] (8 bytes) | | | | | | | |
| 96 | DsSegment[0] (8 bytes) | | | | | | | |
| 104 | DsSegment[1] (8 bytes) | | | | | | | |
| 112 | SsSegment[0] (8 bytes) | | | | | | | |
| 120 | SsSegment[1] (8 bytes) | | | | | | | |
| 128 | Rax (8 bytes) | | | | | | | |
| … | ... | | | | | | | |
| 248 | R15 (8 bytes) | | | | | | | |

*ExceptionVector* provides the vector number of the exception that was generated.

*ExceptionFlags* is of type HV\_X64\_EXCEPTION\_INFO andprovides information about the exception.

*InstByteCount* indicates how many instruction bytes have been provided in the InstructionBytes fields. Valid values are in the range 0 through 16.

*ErrorCode* provides the error code that would have been pushed as part of the exception frame. This field is valid only if the exception flags indicate that an error code is present.

*ExceptionParameter* provides additional information whose meaning is specific to the exception type. It is only defined for page faults (vector 0x0E) and breakpoint exceptions (vector 0x01). In the case of page faults, the virtual address that caused the fault is provided (that is, the value that would normally be placed in CR2 by the processor). In the case of breakpoint exceptions, the value that would have been placed in DR6 is provided. This field is reserved for all other exception type.

*InstructionBytes* includes up to 16 bytes from the instruction stream starting at the current CS:RIP. The number of valid bytes is specified by *InstByteCount*. If the access was due to the delivery of an interruption (as indicated by the *InterruptionPending* bit in the *ExecutionState* field), the message will not include any instruction bytes. In other cases, the hypervisor may only fetch up to the next page boundary or up to the end of the code segment, resulting in an instruction byte count that is less than 16.

*DsSegment* provides the selector, base, limit, and flags for the current data segment (DS).

*SsSegment* provides the selector, base, limit, and flags for the current stack segment (SS).

*Rax...R15* provides the current values of the virtual processor’s sixteen 64-bit general purpose registers.

### APIC EOI Message

An APIC EOI message is delivered by the hypervisor after a virtual processor executes an instruction that writes to the APIC’s memory-mapped EOI register or APIC EOI MSR, indicating the “end of interrupt”. This message is generated only for level-triggered fixed or external interrupts.

|  |  |  |  |  |
| --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | |
| APIC EOI Payload | 16 | VpIndex (4 bytes) | InterruptVector (4 bytes) | |
|  | | | |

*VpIndex* provides the index of the virtual processor.

*InterruptVector* is the vector of the interrupt that was just EOI’ed.

### FERR Asserted Message

A FERR Asserted (legacy floating point support) message is delivered by the hypervisor after a virtual processor executes an instruction that generates an unmasked floating point error when CR0.NE is zero and the IGNNE# bit (in HvX64RegisterFpControlStatus) is zero.

|  |  |  |  |  |
| --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | |
| Legacy FP Payload | 16 | VpIndex (4 bytes) | Rsvd (4 bytes) | |
|  | | | |

*VpIndex* provides the index of the virtual processor.

### Invalid VP Register Value Message

When a virtual processor’s registers are modified, it is possible to specify an illegal register value or combination of register values. In this case, a “bad VP register value” message is generated when the virtual processor resumes execution.

|  |  |  |  |  |
| --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | |
| Invalid Register Payload | 16 | VpIndex (4 bytes) | Rsvd (4 bytes) | |
|  | | | |

*VpIndex* provides the index of the virtual processor.

### Unrecoverable Exception Message

An “unrecoverable exception” message is delivered by the hypervisor when a virtual processor generates an exception that cannot be delivered (for example, a triple fault on x64).

|  |  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | | | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | | | |
| Unrecoverable Exception Payload | 16 | VpIndex (4 bytes) | InstLen = 0  (1 byte) | AccessType  (1 byte) | ExecutionState  (2 bytes) | |
| 24 | CsSegment[0] (8 bytes) | | | | |
| 32 | CsSegment[1] (8 bytes) | | | | |
| 40 | Rip (8 bytes) | | | | |
| 48 | Rflags (8 bytes) | | | | |

### Unsupported Feature Message

This message is generated when a virtual processor accesses a feature of the architecture that is not properly virtualized by the hypervisor.

|  |  |  |  |  |  |
| --- | --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | | Rsvd (3 bytes) | PayloadSize  (1 byte) |
| 8 | SourcePartition (8 bytes) | | | |
| Unsupported Feature Payload | 16 | VpIndex (4 bytes) | | FeatureCode (4 bytes) | |
| 24 | | FeatureParameter (8 bytes) | | |

*VpIndex* provides the index of the virtual processor.

*FeatureCode* provides a code that indicates which feature was accessed. Codes are defined in section 9.2.4.

*FeatureParameter* provides additional information whose meaning is specific to the *FeatureCode*. In the case of an unsupported intercept, the intercept number is supplied. In the case of an unsupported TSS in a task switch request, the TSS type is supplied.

### Event Log Buffers Ready Message

This message is sent to the root partition when either:

the number of pending ready event log buffers reaches the threshold established by the guest using the HvInitializeEventLogBufferGroup hypercall, or

the guest has requested that all pending complete and in-use event log buffers be flushed.

Event Logging support is described in detail in chapter .

|  |  |  |  |  |
| --- | --- | --- | --- | --- |
| Message Header | 0 | MessageType (4 bytes) | Rsvd (3 bytes) | PayloadSize  (1 byte) |
| 8 | Rsvd (8 bytes) | | |
| Buffer Complete Payload | 16 | EventLogType (4 bytes) | BufferIndex (4 bytes) | |
|  | | | |

*EventLogType* provides the type of the buffer that has been placed into a “complete” state.

*BufferIndex* is the index of the first buffer in a ready buffer list. The *NextBufferIndex* field of the event log buffer header (in the message payload) will contain the index of the next buffer in the list. The end of the list is indicated by a buffer index matching the value HV\_EVENTLOG\_BUFFER\_INDEX\_NONE. Buffers appear in the list in the order in which they were placed into the “complete” state. If the event log buffer type uses buffers of the local buffer class, then the list contains only buffers that were completed by this logical processor.

# Partition Save and Restore

The hypervisor provides a means to store certain state of a partition and later restore this state. This support allows a caller to resume execution of the partition at a later time—potentially on a different piece of hardware.

## Overview

### Saved State

The hypervisor allows a parent partition to save the state of a child partition. The *saved state* may be migrated (restored) on another system or check pointed for a subsequent restart.

All virtual processors must be explicitly suspended before a partition can be saved.

### Summary State

In addition to *saved state*, the hypervisor allows a caller to retrieve *summary state* data for the partition or system. Using this state, the caller can determine whether a partition’s potential complete saved state is specifically compatible with respect to the underlying target hardware and hypervisor implementation.

Summary state consists of a minimal amount of information relating specifically to hardware and hypervisor compatibility and does not include most partition-specific context information. Unlike saved state, summary state can be retrieved without the necessity of suspending all of the partition’s virtual processors in order to obtain it.

The HvSavePartitionState and HvRestorePartitionState hypercalls are used to retrieve and subsequently test compatibility using the summary state. The summary state can be related to the system and, optionally, to a particular partition on the system. Absence of a partition ID indicates that system information alone is to be evaluated. Presence of a partition ID indicates that both system and partition-related information (for that particular partition) is to be evaluated.

### Saved State Compatibility and Versioning

The partition and virtual processor state saved from one version of the hypervisor is not guaranteed to be compatible with future versions of the hypervisor. Attempts will be made to minimize version incompatibilities.

Some differences between hardware platforms also introduce saved state incompatibilities, so it may not be possible to save a partition’s state on one physical system and restore it on a second physical system with sufficiently different hardware.

The list of potential incompatibilities is not architecturally defined, but the following list provides some potential examples of hardware incompatibilities that may result in an inability to save and restore:

Virtual processors that are executing real-mode code on CPUs developed by Intel may not be restorable on CPUs developed by AMD.

Partitions that are executing on logical processors that support various guest-visible features (for example, 3DNow!, SSE3, and so on.) may not be restorable on a system whose logical processors do not support these features.

### State That Is Not Saved by the Hypervisor

In general, state that is explicitly managed and set by the parent partition is not saved by the hypervisor. It is expected that code running in the parent partition will save and restore this information.

Information that is *not* saved by the hypervisor includes:

* Partition creation flags
* Partition properties
* Scheduling policy settings and affinities
* Partition ID or parent ID
* GPA mappings and access rights
* Contents of pages
* Intercepts
* Memory pool balance
* Ports and connections
* Partition state
* Default CPUID return values
* Partition statistics
* Monitored notification context

### State That Is Saved by the Hypervisor

The specific state saved may differ from one hypervisor implementation to the next. The format of the saved state is purposely not defined.

In general, state that is not explicitly managed and set by the parent partition and is potentially changed by a guest running within the child partition is saved.

Partition state that is saved includes:

* Partition version information
* Logical processor information (sufficient to determine compatibility for restoration)
* Partition-wide MSR values (for example, hypercall base address)
* Locations of overlay pages
* Reference counter

Virtual processor state that is saved includes:

* All virtual processor registers
* Undelivered (queued) messages
* APIC and SynIC state
* SIEF and SIM page contents
* Synthetic and periodic timers
* Periodic and PM timer assists

### Partition State Streams

A partition’s state is saved as a stream of data. The format of the stream is not defined and may change between different versions of the hypervisor or different platforms supported by a single hypervisor.

Failure to save and restore the complete save state (see section ) of a partition may result in a partition that functions incorrectly or not at all. In particular, an “invalid state” error may (but is not guaranteed to) be generated when its virtual processors are unsuspended.

Changes to the partition or its virtual processors will necessarily change the state. Care should be taken to avoid modifying a partition or its virtual processors in the middle of the save or restore process. Failure to follow this rule may result in a partition that functions incorrectly after restoration.

Summary state (see section ) is intended solely for testing a partition’s save-restore compatibility and cannot be used to affect any partition-specific context.

### Recommended Save Process

It is recommended that a parent partition use the following procedure to save a partition.

* Suspend all of the virtual processors within a partition by calling HvSetVpRegisters and modifying the “explicit suspend” register (see chapter ). Note that it is possible for virtual processors to already be suspended due to intercepts. It is recommended that all outstanding intercepts be addressed prior to proceeding.
* Drain and process all outstanding messages on ports with a connection from the child partition being saved (see chapter ). If the parent generates messages to be sent back to the child, it should be prepared to save any messages that it was unable to successfully post. Suspension of the child’s virtual processors precludes the child’s ability to receive messages and could lead to a lack of message buffer availability on any of the child’s ports. The parent may elect to include the drained messages with its saved context for execution after the child partition’s restoration (and subsequent resumption) with its new parent.
* Call HvSavePartitionState (see section ) repeatedly until the entire stream of state data has been retrieved.

### Recommended Restore Process

It is recommended that a parent partition use the following procedure to restore a partition.

* Call HvRestorePartitionState supplying the saved or summary data and appropriate flags to determine whether the partition state can be restored (see section ).
* Create a new partition (see section ) and populate it with resources, such as GPA mappings (see chapters and ) and so on.
* Call HvInitializePartition to initialize the new partition (see section ).
* Create new virtual processors. The count and IDs of the virtual processors should match those of the partition that was previously saved. (see chapter ).
* Recreate all child ports and reestablish all connections with the new parent (see chapter ).
* Call HvRestorePartitionState repeatedly, supplying the saved state stream until it indicates that the partition’s state has been completely restored (see section ).
* Resume execution by clearing the “explicit suspend” state of each of the virtual processors (see chapter ).

## Partition Save and Restore Data Types

### Partition Save and Restore State

The calls HvSavePartitionState and HvRestorePartitionState are expected to be called several times in succession until the complete state has either been returned (save) or restored (restore). When the hypercall return value indicates successful completion, a separate status representing the *overall state* of the save and restore is returned with each of the calls. The *overall state* provides updated feedback concerning the save or restore process and is used to indicate completion, interim success (incomplete) or an error. The errors indicate that continuation is not possible and why. Note that while a hypercall may have succeeded (as indicated by HV\_STATUS\_SUCCESS in the hypercall return value), the *overall state* may return an error. The *overall state* should not be examined in the event of a hypercall failure, in accordance with the hypercall status rules described in section 4.11.

typedef enum

{

HvStateComplete = 0,

HvStateIncomplete = 1,

HvStateRestorable = 2,

HvStateCorruptData = 3,

HvStateUnsupportedVersion = 4,

HvStateProcessorFeatureMismatch = 5,

HvStateHardwareFeatureMismatch = 6,

HvStateProcessorCountMismatch = 7,

HvStateProcessorFlagsMismatch = 8,

HvStateProcessorIndexMismatch = 9,

HvStateProcessorInsufficientMemory = 10,

HvStateIncompatibleProcessor = 11,

HvStateProcessorFeatureSse3Mismatch = 12,

HvStateProcessorFeatureLahfSahfMismatch = 13,

HvStateProcessorFeaturSsse3eMismatch = 14,

HvStateProcessorFeatureSse41Mismatch = 15,

HvStateProcessorFeatureSse42Mismatch = 16,

HvStateProcessorFeatureSse4aMismatch = 17,

HvStateProcessorFeatureSse5Mismatch = 18,

HvStateProcessorFeaturePopcntMismatch = 19,

HvStateProcessorFeatureCmpxchg16bMismatch = 20,

HvStateProcessorFeatureAltmovcr8Mismatch = 21,

HvStateProcessorFeatureLzcntMismatch = 22,

HvStateProcessorFeatureMisalignedSseMismatch = 23,

HvStateProcessorFeatureMmxExtMismatch = 24,

HvStateProcessorFeature3DNowMismatch = 25,

HvStateProcessorFeatureExtended3DNowMismatch = 26,

HvStateProcessorFeaturePage1GBMismatch = 27,

HvStateProcessorCacheLineFlushSizeMismatch = 28,

HvStateProcessorFeatureXsaveMismatch = 29,

HvStateProcessorFeatureXsaveoptMismatch = 30,

HvStateProcessorFeatureXsaveLegacySseMismatch = 31,

HvStateProcessorFeatureXsaveAvxMismatch = 32,

HvStateProcessorFeatureXsaveUnknownFeatureMismatch = 33,

HvStateProcessorXsaveSaveAreaMismatch = 34

} HV\_SAVE\_RESTORE\_STATE\_RESULT;

typedef HV\_SAVE\_RESTORE\_STATE\_RESULT \*PHV\_SAVE\_RESTORE\_STATE\_RESULT;

The following table describes the meaning of these values. Applicability to save and restore is shown.

| Value | Save | Restore | Meaning |
| --- | --- | --- | --- |
| HvStateComplete | ✔ | ✔ | For HvSavePartitionState, the complete state of the partition has been returned and no more state data is available.  For HvRestorePartitionState, the complete state of the partition has been restored and no more state data is expected. |
| HvStateIncomplete | ✔ | ✔ | The partition save or restore process has not completed and additional calls (HvSavePartitionState or HvRestorePartitionState) are required. At this point in the save or restore process no error condition has thus far been detected. |
| HvStateRestorable |  | ✔ | The partition’s potential complete saved state is specifically compatible with respect to the underlying target hardware and hypervisor implementation. |
| HvStateCorruptData |  | ✔ | The partition state appears to be corrupted. |
| HvStateUnsupportedVersion |  | ✔ | The hypervisor implementation is not compatible with the version of the saved state. |
| HvStateProcessorFeatureMismatch |  | ✔ | The features supported by the logical processor(s) are incompatible with the features of the logical processor(s) in use when the state was saved. |
| HvStateHardwareFeatureMismatch |  | ✔ | The features supported by the physical hardware are incompatible with the features of the physical hardware in use when the state was saved. |
| HvStateProcessorCountMismatch |  | ✔ | The number of virtual processors associated with the restore partition is different from the number of virtual processors present when the partition’s state was saved. |
| HvStateProcessorFlagsMismatch |  | ✔ | The option flags associated with a virtual processor in the restore partition mismatches those associated with the equivalent virtual processor when the state was saved. |
| HvStateProcessorIndexMismatch |  | ✔ | The indexes of the virtual processors in the restore partition differ from those associated with the virtual processors when the state was saved. |

### Save and Restore Partition State Flags

The following flags are used with the HvSavePartitionState and HvRestorePartitionState hypercalls.

typedef UINT32 HV\_SAVE\_RESTORE\_STATE\_FLAGS;

#define HV\_SAVE\_RESTORE\_STATE\_START 0x00000001

#define HV\_SAVE\_RESTORE\_STATE\_SUMMARY 0x00000002

The flag HV\_SAVE\_RESTORE\_STATE\_START is required to begin a save or restore process and should be specified with the first invocation of HvSavePartitionState or HvRestorePartitionState. Subsequent invocations should not specify this flag unless it is prematurely terminating a save or restore process. In this case it is used to ensure that all intervening save or restore context maintained by the hypervisor is released.

The flag HV\_SAVE\_RESTORE\_STATE\_SUMMARY indicates that the save data is summary data (see section ) and not complete partition save data.

The HV\_SAVE\_RESTORE\_STATE\_START and HV\_SAVE\_RESTORE\_STATE\_SUMMARY flags may both be present.

## Partition Save and Restore Interfaces

### HvSavePartitionState

The HvSavePartitionState hypercall saves part of the state of a partition. It may also be used to obtain summary state data for the system or a partition.

**Wrapper Interface**

HV\_STATUS

HvSavePartitionState(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_SAVE\_RESTORE\_STATE\_FLAGS Flags,

\_\_out PHV\_SAVE\_RESTORE\_STATE\_RESULT SaveState,

\_\_out PUINT32 SaveDataCount,

\_\_out bcount\_part(4080, \*SaveData),

PVOID SaveData

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvSavePartitionState | | |
|  | Call Code = 0x005E | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | Flags (4 bytes) | Padding (4 bytes) |
| 🡄 Output Parameters | | |
| 0 | SaveState (4 bytes) | SaveDataCount (4 bytes) |
| 8 | SaveData (up to 4080 bytes) | |

Description

For information regarding summary state data for the system or a partition, see section . For a description of the recommended save process, see section .

Save data is returned in a sequence-sensitive stream. The caller begins the save process by specifying the HV\_SAVE\_RESTORE\_STATE\_START flag. Subsequent calls should not specify this flag. The process should progress as long as the call to HvSavePartitionState returns success and *SaveState* indicates HvStateIncomplete. The process ends either when an error status is returned or success is returned and *SaveState* indicates HvStateComplete.

Input Parameters

*PartitionId* specifies the partition ID of the partition that is being saved, or zero if system summary state data is being retrieved.

*Flags* specifies what data are to be saved.

Output Parameters

*SaveDataCount* indicates how many bytes of state were saved (retrieved).

*SaveData* contains some or all of the requested save state, the size of which is returned in *SaveDataCount.* At most 4080 bytes can be returned and, as with all hypercalls, the output parameters can never cross a page boundary. A size of zero is legal and does not have any significance.

*SaveState* contains the status feedback of the overall save procedure.

Restrictions

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId.*

All virtual processors must be in an explicit suspended state except when saving summary information.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | The specified partition has one or more virtual processors that are not explicitly suspended. (This does not apply when saving summary information.) |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |
| HV\_STATUS\_INVALID\_PARAMETER | The flags parameter specifies an unsupported option. |
| HV\_STATUS\_INVALID\_SAVE\_RESTORE\_STATE | HvSavePartitionState has not been called with HV\_SAVE\_RESTORE\_STATE\_START to initialize the save process. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the caller is insufficient to perform the operation. |

### HvRestorePartitionState

The HvRestorePartitionState hypercall restores partition state that was previously saved. It may also be used to submit summary state data for a system or partition for compatibility assessment purposes.

**Wrapper Interface**

HV\_STATUS

HvRestorePartitionState(

\_\_in HV\_PARTITION\_ID PartitionId,

\_\_in HV\_SAVE\_RESTORE\_STATE\_FLAGS Flags,

\_\_inout PUINT32 RestoreDataCount,

\_\_in bcount(RestoreDataCount),

PCVOID RestoreData,

\_\_out HV\_SAVE\_RESTORE\_STATE\_RESULT RestoreState

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvRestorePartitionState | | |
|  | Call Code = 0x005F | |
| 🡆 Input Parameters | | |
| 0 | PartitionId (8 bytes) | |
| 8 | Flags (4 bytes) | RestoreDataCount (4 bytes) |
| 16 | RestoreData (up to 4080 bytes) | |
| 🡄 Output Parameters | | |
| 0 | RestoreState (4 bytes) | RestoreDataConsumed (4 bytes) |

Description

For information regarding summary state data for a system or partition, see section .

The caller should make this call repeatedly until the entire save stream has been presented to the hypervisor. For the recommended restore process, see section .

Input Parameters

*PartitionId* specifies the partition ID of the partition that is being restored, or zero if system summary state data is being presented for compatibility assessment purposes.

*Flags* specifies which data are to be restored.

*RestoreDataCount* supplies the number of valid bytes provided in *RestoreData.*

*RestoreData* provides some or all of the (previously saved) state that is to be restored. At most 4080 bytes can be provided and, as with all hypercalls, the input parameters can never cross a page boundary.

Output Parameters

*RestoreState* receives the status feedback of the overall restore procedure.

*RestoreDataConsumed* returns the number of bytes of *RestoreData* that has been processed by the hypervisor. This value may range from zero to the value provided in *RestoreDataCount.*

Restrictions

The caller must possess the *CreatePartitions* privilege.

The partition specified by *PartitionId* must be in the “active” state.

The caller must be the parent of the partition specified by *PartitionId.*

All virtual processors must be in an explicit suspend state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the parent of the specified partition.  The caller’s partition privilege flag *CreatePartitions* is cleared. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | The specified partition has one or more virtual processors that are not explicitly suspended. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |
|  |  |
|  |
|  |
|  |  |
| HV\_STATUS\_INVALID\_PARAMETER | The flags parameter specifies an unsupported option. |
| HV\_STATUS\_INVALID\_SAVE\_RESTORE\_STATE | HvRestorePartitionState has not been called with HV\_SAVE\_RESTORE\_STATE\_START to initialize the restoration process. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the caller is insufficient to perform the operation. |

# Scheduler

## Scheduling Concepts

The hypervisor schedules virtual processors to run on logical processors. The interfaces in this section allow a parent partition to set the scheduling policy for its child partitions.

The hypervisor scheduler makes scheduling decisions based on these policy settings. Because these decisions are made at discreet times, not continuously, the policies are guaranteed only over a sufficiently long period of time. This time is purposely undefined but will typically be on the order of hundredths of a second.

## Scheduling Policy Settings

Various schedule policy settings can be obtained using HvGetPartitionProperty (see section ) and adjusted using the HvSetPartitionProperty (see section ) hypercall interfaces. These include:

* CPU Reserve
* CPU Cap
* CPU Weight

Each of these is described in more detail in the following sections.

### CPU Reserve

A CPU reserve can be supplied for each partition. The hypervisor guarantees that this fraction of CPU time is available to each virtual processor within the partition as needed. It does not necessarily mean that the virtual processors *will* consume the entire reserve. If they are idle or waiting on hypervisor work, other virtual processors may consume the available processor cycles.

The CPU reserve is specified as a fraction of a logical processor’s capacity. A reserve value of 0.5 indicates that 50% of a logical processor is reserved for each virtual processor in the partition. Valid reserve values range from 0 (no reserve) to 1 (in which case each virtual processor is guaranteed to get 100% of a logical processor if required). The reserve value may not be greater than the CPU Cap (see section ).

The reserve value is expressed as an integer ranging from 0x00000000 to 0x00010000. For example, the value 0x0000C000 indicates 0.75, or 75% of a logical processor. By default, a partition’s reserve is set to 0x00000000, indicating that there is no reserve.

The total reserves for all virtual processors cannot exceed the number of logical processors in the system. Reserves are not guaranteed if the number of virtual processors in a partition exceeds the number of logical processors that are available for scheduling.

If the number of virtual processors in a partition is greater than the number of logical processors, then reserves are not guaranteed, and the partition’s reserve may be reset to zero.

### CPU Cap

A CPU cap can be specified for each partition. The hypervisor guarantees that the fraction of CPU time consumed by each virtual processor within the partition will not exceed this cap.

The CPU cap is specified as a fraction of a logical processor’s capacity. A cap value of 0.5 indicates that each virtual processor will be restricted to using 50% of a logical processor. Valid cap values range from 0 to 1 (in which case the cap has no effect).

The CPU cap value may not be less than the CPU reserve (see section 18.2.1).

The cap value is expressed as an integer ranging from 0x00000000 to 0x00010000. For example, the value 0x0000C000 indicates 0.75, or 75% of a logical processor.

By default, a partition’s cap is set to 0x00010000, indicating that there is no cap.

### CPU Weight

The CPU weight is a relative weight assigned to each of the virtual processors of the partition. Unless otherwise constrained by reserves and caps, the scheduler will attempt to weight the run time of the virtual processors scheduled on a given logical processor according to their relative weights. Let’s consider the case where three partitions, each with one virtual processor are being scheduled on a single logical processor, their weights are 100, 200 and 700, no reserves or caps are in effect, and all three of the virtual processors have work to perform (that is, they are not idle). In this case, the fraction of physical CPU capacity provided to the three virtual processors would be approximately 10%, 20%, and 70%.

The CPU weight value is expressed as a decimal value from 1 to 10,000 where 100 (the geometric mean) is the typical value.

By default, a partition’s weight is set to 100.

## Other Scheduling Considerations

### Hyperthreading

Multiple virtual processors can optionally be grouped together and scheduled onto hyperthreads within a single physical processor core. In effect, these virtual processors then act like virtual hyperthreads. When virtual processors are grouped as such, the hypervisor tries to schedule them concurrently on the same physical processor core. This scheduling behavior potentially improves performance and reduces information leakage across partition boundaries.

### NUMA and Affinity

When a virtual processor is run-able, the hypervisor’s scheduler assigns it to a logical processor. The placement is determined based on a variety of factors including workload, reservations, and NUMA topology. In general, the scheduler will attempt to keep a virtual processor scheduled on a logical processor that is topologically closest to the memory being accessed by the virtual processor, in effect minimizing memory access times.

The scheduler also attempts to create as much temporal affinity as possible. That is, it will prefer to run a virtual processor on the same logical processor each time it is scheduled. If the logical processor is oversubscribed, the scheduler may move it to another logical processor.

### Guest Spinlocks

A typical multiprocessor-capable operating system uses locks for enforcing atomicity of certain operations. When running such an operating system inside a virtual machine controlled by the hypervisor these critical sections protected by locks can be extended by intercepts generated by the critical section code. The critical section code may also be preempted by the hypervisor scheduler. Although the hypervisor attempts to prevent such preemptions, they can occur. Consequently, other lock contenders could end up spinning until the lock holder is re-scheduled again and therefore significantly extend the spinlock acquisition time. The hypervisor indicates to the guest OS the number of times a spinlock acquisition should be attempted before indicating an excessive spin situation to the hypervisor. This count is returned in CPUID leaf 0x40000004.

The HvNotifyLongSpinWait hypercall provides an interface for enlightened guests to improve the statistical fairness property of a lock for multiprocessor virtual machines. Through this hypercall, a guest notifies the hypervisor of a long spinlock acquisition. This allows the hypervisor to make better scheduling decisions.

## Scheduler Data Types

The following data types support the scheduler interfaces.

typedef UINT64 HV\_INPUT\_NOTIFY\_LONG\_SPINWAIT ;

typedef HV\_INPUT\_NOTIFY\_LONG\_SPINWAIT, \*PHV\_INPUT\_NOTIFY\_LONG\_SPINWAIT;

## Scheduler Interfaces

### HvNotifyLongSpinWait

The HvNotifyLongSpinWait hypercall is used by a guest OS to notify the hypervisor that the calling virtual processor is attempting to acquire a resource that is potentially held by another virtual processor within the same partition. This scheduling hint improves the scalability of partitions with more than one virtual processor.

**Wrapper Interface**

HV\_STATUS

HvNotifyLongSpinWait(

\_in HV\_INPUT\_NOTIFY\_LONG\_SPINWAIT SpinwaitInfo

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvNotifyLongSpinWait [fast] | | |
|  | Call Code = 0x0008 | |
| 🡆 Input Parameters | | |
| 0 | SpinwaitInfo (4 bytes) | Padding (4 bytes) |

Description

The HvNotifyLongSpinWait hypercall allows a partition to inform the Hypervisor of a long spinlock acquire failure. The hypervisor can use this information to make better scheduling decisions for the notifying virtual processor and its partition.

Input Parameters

SpinwaitInfo – Specifies the accumulated count the guest was spinning.

Output Parameters

None.

Restrictions

None.

Return Values

There is no error status for this hypercall, only HV\_STATUS\_SUCCESS will be returned as this is an advisory hypercall.

# Event Logging

## Overview

The hypervisor provides a general mechanism for recording events within logs and exposing the resulting data, via buffers, to the root partition. Through this mechanism, the root partition can selectively collect various types of events, such as:

* Security logs (for example, policy and rights modifications)
* Diagnostic logs (for example, hardware or software failures)
* Performance analysis information (for example, call traces and profiler data)
* Self-test results

While the hypervisor defines the common framework to be used for buffers and events within buffers, it leaves the definition of the event data itself to the particular application.

### Event Log Buffers

*Event log buffers* are composed of between 1 and 512 pages of memory, making them between 4K and 2MB in size. Buffers are created individually, and the size and limit are determined from the event log buffer group’s geometry (geometry is described in the following section). The pages that constitute each buffer are allocated from the partition’s pool.

When a buffer is created, its pages are allocated and initialized. It remains inaccessible to the guest however, and must therefore be mapped prior to access. Mapping the buffer creates read-only overlay pages in the guest’s address space and returns the GPAs of the pages that constitute the buffer. The buffer remains read-only accessible until it is unmapped. Deletion of the buffer returns the buffer’s pages to the partition’s pool from whence they were allocated.

### Event Log Buffer Groups

Associated with each event log type is an *event log buffer group* that has a fixed set of characteristics. These include the buffer *geometry* and the *completed buffer threshold*. The geometry is defined as the *maximum number of buffers* and the *number of pages per buffer*. All buffers created as part of the group are the same size (that is, they are composed of the same number of pages). A maximum number of 512 buffers may be in the group at any one time. Depending upon the size of the buffers, the hypervisor implementation may not support the creation of the maximum number of buffers.

Prior to use, the event log buffer group must be initialized, at which time the group’s characteristics are established. Initialization of the group may require memory to manage the necessary infrastructure to support the event log type. This will be allocated from the partition’s pool. When the group is no longer needed, event logging should be disabled, all buffers should be unmapped and deleted, and the group subsequently destroyed.

### Local and Global Buffer Classes

The hypervisor supports two event log buffer classes:

*Local buffers* are distributed across and associated with specific logical processors and can be filled without acquiring costly cross-processor locks. The local buffer class is used with performance-critical applications. When deriving the event log buffer group geometry, the guest should consider the number of logical processors and the NUMA topology.

*Global buffers* are shared among all logical processors, requiring their access to be synchronized using global locks. The global buffer class is used for applications that are not performance critical or where the frequency of event generation is low.

The guest should allocate multiple event log buffers for the following reasons:

For local buffers, at least one buffer should be available as the “in use” (currently active) buffer for each active logical processor.

When there is insufficient space available for a new event log entry in the active buffer, the hypervisor will “complete” it. Buffers may also be completed prematurely by the guest requesting the flushing active buffers. If, at the time that an event is ready to be logged, no “in use” buffer is present, then a “free” buffer will be allocated and become the “in use” buffer. If local buffers are being used, only buffers created for that specific logical processor will be used. To avoid interruption in the logging of events, it is important for there to be sufficient free buffers available.

### Event Log Buffer Indices

Buffers are assigned an *event log buffer index* beginning at zero and ending at the maximum number of buffers (established at group initialization) minus one. The index is unique within the event log type and is used by both the hypervisor and the guest to identify each individual buffer in a consistent manner. The index associated with each buffer is assigned by the guest when the buffer is created. An attempt to create a buffer with an index that exceeds the maximum or that matches one assigned to an existing buffer will result in an error.

### Event Log Buffer States

When a buffer is created, it is part of the event log buffer group associated with an event log type. It is initialized in an internal “standby” state. While in this state, the buffer is neither accessible from the guest nor usable by the hypervisor. Mapping the buffer reveals the location of its constituent pages and makes the buffer available to the hypervisor for recording events. Un‑mapping the buffer returns it to the internal “standby” state.

While the event log type is enabled, all related buffers of the group have a guest-visible state described as follows:

*HvEventLogBufferStateFree* – The buffer is currently available to be allocated by the hypervisor as an “in use” buffer. With the exception of the header’s state field, the guest should not read the buffer’s contents. A buffer enters the free state when it is either mapped or released by the guest. It remains in this state until it is either unmapped by the guest (hence, taken out of service) or allocated by the hypervisor as an “in use” buffer. Note that the hypervisor may change the state of this buffer at any time.

*HvEventLogBufferStateInUse* – The buffer is currently active and accepting event messages. Buffers of this type are being written by the hypervisor and may be accessed by the guest in a limited fashion (see section 19.1.12 for details). Buffers leave the “in use” state when they are full or when the guest requests that buffers be flushed. Note that the buffer can become full at any time and the hypervisor could therefore change the state at any time.

*HvEventLogBufferStateComplete* – The buffer is no longer active and contains valid data. Buffers of this type are no longer being written to by the hypervisor and may be read by the guest. Buffers in the “complete” state are on a list waiting for the guest to be notified of their completion. Buffers leave the HvEventLogBufferStateComplete state when either the guest flushes them or the threshold value is reached. The hypervisor changes the state of all “complete” buffers on the list to HvEventLogBufferStateReady and sends the list to the guest with a notification message. While the hypervisor may transition buffers on the completed list to the ready state at any time, the content of the buffer (aside from the state) will not change.

*HvEventLogBufferStateReady* – The buffer is complete (as with the HvEventLogBufferStateComplete state above) and has been sent to the guest with a notification message. Buffers in the “ready” state are under the exclusive control of the guest. Buffers leave the HvEventLogBufferStateReady state when the guest either unmaps them or releases them back to the hypervisor for reuse.

Note that the internal “standby” state is absent in the above description because it is not visible from the guest. While in the standby state, the buffer is not mapped, making the state information inaccessible. When in the “standby” state, the guest may either map the buffer (transitioning it to the HvEventLogBufferStateFree state) or delete the buffer.

depicts how buffers transition between states. Items labeled “guest” are guest actions (a hypercall), and those labeled “hypervisor” are actions taken by the hypervisor as event logging proceeds.

![](data:image/x-emf;base64,AQAAAGwAAAABAAAA1v///0YCAABBAQAAAAAAAAAAAAA+RgAAOCgAACBFTUYAAAEALBQBALUCAAAEAAAAAAAAAAAAAAAAAAAAAAQAAAADAABUAQAADgEAAAAAAAAAAAAAAAAAACAwBQCwHgQARgAAACwAAAAgAAAARU1GKwFAAQAcAAAAEAAAAAIQwNsBAAAAYAAAAGAAAABGAAAAcAAAAGQAAABFTUYrMEACABAAAAAEAAAAAACAPx9ABAAMAAAAAAAAAB5ABQAMAAAAAAAAAB1AAAAUAAAACAAAAAIAAAAjAQAACkAAgCQAAAAYAAAA/////wEAAAAAAMA/6/IkwpPuEESHpLRDIQAAAAgAAABiAAAADAAAAAEAAAAkAAAAJAAAAAAAgD0AAAAAAAAAAAAAgD0AAAAAAAAAAAIAAAAnAAAAGAAAAAEAAAAAAAAA////AAAAAAAlAAAADAAAAAEAAAAlAAAADAAAAAgAAIBWAAAAMAAAAAEAAADW////RgIAAEEBAAAFAAAAGABt/RgAARRUJAEUVCRt/RgAbf0lAAAADAAAAAcAAIAlAAAADAAAAAAAAIAkAAAAJAAAAAAAgEEAAAAAAAAAAAAAgEEAAAAAAAAAAAIAAAAiAAAADAAAAP////9GAAAAHAAAABAAAABFTUYrBEAAAAwAAAAAAAAARgAAAHABAABkAQAARU1GKwhAAAOoAAAAnAAAAAIQwNsQAAAAAAAAABKJx0L0pn5CD4nHQjocKUJDNepClYfHQU59CkOPh8dB/N8fQ3uHx0EZNjFDIhwpQhs2MUPqpn5CGzYxQ+umfkIbNjFD66Z+Qhs2MUPspn5CGzYxQ9EYqkIA4B9DB8XMQlN9CkMHxcxCSjXqQgjFzEIUicdC0xiqQhKJx0Lwpn5CAAMDAwMDAwMDAwMDAwMDgzNAAAEMAAAAAAAAACpAAAAkAAAAGAAAAAAAwEIAAAAAAAAAAAAAwMISicdCB8XMQghAAQFkAAAAWAAAAAIQwNsDAAAAAQAAAAQAAAD37uj/q6oqvNouUT8BAAAAqdrK/zAAAAACEMDbBAAAAAAAAACrqiq8q6oqvNouUT+rqiq82i5RP9ouUT+rqiq82i5RPwABAYEKQAAAJAAAABgAAAABAAAAAQAAAKuqKryrqiq8hdlTP4XZUz8hAAAACAAAAGIAAAAMAAAAAQAAACEAAAAIAAAAOwAAAAgAAAATAAAADAAAAAEAAABVAAAAXAAAAAAAAAAAAAAA//////////8QAAAAZABAAGQAKwB2ABkAiwAZAKAAGQCyACsAsgBAALIAQACyAEAAsgBAALIAVgCgAGcAiwBnAHYAZwBkAFYAZABAADwAAAAIAAAAQwAAAAwAAAABAAAAEwAAAAwAAAABAAAAFQAAAAwAAAAEAAAAFQAAAAwAAAAEAAAARgAAABQAAAAIAAAAVE5QUAYBAABRAAAAQAQAAGQAAAAaAAAAsQAAAGYAAABiAAAAGAAAAAAAAAAAAAAAUAAAAFAAAABQAAAAMAAAAIAAAADAAwAAAAAAAIYA7gBQAAAAUAAAACgAAABQAAAAUAAAAAEAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA////AAAAAAAAAAAAAAAURH////////////+Hw3////////////8WRH////////////+Hw3////////////8YRH////////////+Hw3////////////8aRH////////////+Hw3////////////8dRH////////////+Hw3////////////8fRH////////////+Hw3////////////8hRH////////////+Hw3////////////8jRH////////////+Hw3////////////8mRH////////////+Hw3////////////8oRH////////////+Hw3////////////8qRH////////////+Hw3////////////8sRH////////////+Hw3////////////8vRH////////////+Hw3////////////8xRH////////////+Iw3////////////8xRH////////////+Nw3////////////8xRH////////////+Rw3////////////8xRH////////////+Ww3////////////8xRH////////////+aw3////////////8xRH////////////+fw3////////////8xRH////////////+iw3////////////8wRH////////////+iw3////////////8uRH////////////+iw3////////////8rRH////////////+iw3////////////8pRH////////////+iw3////////////8nRH////////////+iw3////////////8lRH////////////+iw3////////////8iRH////////////+iw3////////////8gRH////////////+iw3////////////8eRH////////////+iw3////////////8cRH////////////+iw3////////////8ZRH////////////+iw3////////////8XRH////////////+iw3////////////8VRH////////////+iw3////////////8TRH////////////+hw3////////////8TRH////////////+dw3////////////8TRH////////////+Yw3////////////8TRH////////////+Uw3////////////8TRH////////////+Pw3////////////8TRH////////////+Lw3////////////8TRH////////////+Hw1EAAAD4GgAAZAAAABoAAACxAAAAZgAAAGIAAAAYAAAAAAAAAAAAAABQAAAAUAAAAFAAAACoAQAA+AEAAAAZAAAAAAAAxgCIAFAAAABQAAAAKAAAAFAAAABQAAAAAQAIAAAAAAAAAAAAAAAAAAAAAABgAAAAAAAAAAAAAAD///8A9u7oAPXu5wD07ecA8+3nAPLt5gDx7eYA8OzlAO/s5QDu7OUA7ezkAO3r5ADs6+QA6+vjAOnr4wDo6uIA5+riAObq4QDl6eEA5OnhAOPp4ADi6OAA4ejfAODo3wDf6N8A3ujeAN3n3gDc590A2+fdANrn3QDZ5twA2ObcANfm3ADW5dsA1eXbANTl2wDT5doA0uTaANHk2QDQ5NkAz+TZAM7k2ADN49gAzOPXAMvj1wDK4tcAyeLWAMji1gDH4tYAxuHVAMXh1QDE4dQAw+HUAMLg1ADB4NMAwODTAL/g0wC+39IAvd/SALzf0QC739EAut7RALne0AC43tAAt93PALXdzwC03c4As93OALLczgCy3M0AsdzNALDczQCv28wArtvMAKzbywCr2ssAqtrKAPbu5wD17ecA7uzkAOrr4wDp6uMA4ungANzn3gDW5tsA0uXaAMzj2ADK49cAxuLVAMTh1QC+4NIAut7QALPczgCv3MwArdvLAAEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU0BTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTQFLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0xNAV9fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX0tLTE0BSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKS0tMTQFeXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eSUpLS0xNAUhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhJSktLTE0BR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHSElKS0tMTQFGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkdISUpLS0xNAV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXUVGR0hJSktLTE0BQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NERUZHSElKS0tMTQFCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQ0RFRkdISUpLS0xNAUJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJDREVGR0hJSktLTE0BQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQkNERUZHSElKS0tMTQFAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEFCQ0RFRkdISUpLS0xNAT8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/P0BAQUJDREVGR0hJSktLTE0BXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFw/QEBBQkNERUZHSElKS0tMTQE9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Pj9AQEFCQ0RFRkdISUpLS0xNAT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0+P0BAQUJDREVGR0hJSktLTE0BPDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PT4/QEBBQkNERUZHSElKS0tMTQE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNAVtbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbWzo7PD0+P0BAQUJDREVGR0hJSktLTE0BODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQE3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNATc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BNjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQE1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTY3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNAVpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWjQ1Njc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzNDU2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQFZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZMjM0NTY3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNATExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTEyMzQ1Njc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMTIzNDU2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQEvLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLzAxMjM0NTY3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNAVhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWC4vMDEyMzQ1Njc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLi8wMTIzNDU2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQFXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNASsrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKyssLS4vMDEyMzQ1Njc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQEpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNASgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQFWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQEFCQ0RFRkdISUpLS0xNASUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BAQUJDREVGR0hJSktLTE0BJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJSYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEBBQkNERUZHSElKS0tMTQEjIyMjIyMj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YmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiT1BRYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYU5PUFFMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExNTk9QUUtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLTE1OT1BRSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSktMTU5PUFFJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlKS0xNTk9QUUhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISUpLTE1OT1BRYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYEhJSktMTU5PUFFGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZHSElKS0xNTk9QUUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRkdISUpLTE1OT1BRREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREREVGR0hJSktMTU5PUFFDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NERUZHSElKS0xNTk9QUV9fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fQ0RFRkdISUpLTE1OT1BRQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUJDREVGR0hJSktMTU5PUFFAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBBQkNERUZHSElKS0xNTk9QUT8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/QEFCQ0RFRkdISUpLTE1OT1BRPj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj9AQUJDREVGR0hJSktMTU5PUFE9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0+P0BBQkNERUZHSElKS0xNTk9QUV5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5ePT4/QEFCQ0RFRkdISUpLTE1OT1BRXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXTw9Pj9AQUJDREVGR0hJSktMTU5PUFE6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo7PD0+P0BBQkNERUZHSElKS0xNTk9QUTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFE3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUTY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRNTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFE0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUTMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUVxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFEuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFErKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKyssLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUSoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbWyorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFEoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUScnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFElJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyQlJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFEiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIjJCUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUSEhISEhISEh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TExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTE1OT1BRS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tMTU5PUFFKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKS0xNTk9QUXd3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d0pLTE1OT1BRSEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhJSktMTU5PUFF2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2SElKS0xNTk9QUUZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkdISUpLTE1OT1BRRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVGR0hJSktMTU5PUFF1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1RUZHSElKS0xNTk9QUXR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dERFRkdISUpLTE1OT1BRc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3NDREVGR0hJSktMTU5PUFFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQkNERUZHSElKS0xNTk9QUUBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEFCQ0RFRkdISUpLTE1OT1BRPz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz9AQUJDREVGR0hJSktMTU5PUFFycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJyP0BBQkNERUZHSElKS0xNTk9QUT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT4/QEFCQ0RFRkdISUpLTE1OT1BRcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXE9Pj9AQUJDREVGR0hJSktMTU5PUFFwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwPD0+P0BBQkNERUZHSElKS0xNTk9QUTo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BROTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFvb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vOTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUW5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubjg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRNjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFE1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUTQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzMzM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFtbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbDIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRa2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2sxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFEvLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUS4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRampqampqampqampqampqampqampqampqampqampqampqampqampqampqamouLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaCwtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUSgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmYoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZCYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFEjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjJCUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUSIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiMkJSYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2MiIyQlJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiISIjJCUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUR8fHx8fHx8f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q6oqvKuqKryF2VM/hdlTPyEAAAAIAAAAYgAAAAwAAAABAAAAIQAAAAgAAAA7AAAACAAAABMAAAAMAAAAAQAAAFUAAABcAAAAAAAAAAAAAAD//////////xAAAABrAZQAawF/AHwBbQCSAW0ApwFtALgBfwC4AZQAuAGUALgBlAC4AZQAuAGqAKcBuwCSAbsAfAG7AGsBqgBrAZQAPAAAAAgAAABDAAAADAAAAAEAAAATAAAADAAAAAEAAAAVAAAADAAAAAQAAAAVAAAADAAAAAQAAABGAAAAFAAAAAgAAABUTlBQBgEAAFEAAABABAAAawEAAG4AAAC3AQAAugAAAGkBAABsAAAAAAAAAAAAAABQAAAAUAAAAFAAAAAwAAAAgAAAAMADAAAAAAAAhgDuAFAAAABQAAAAKAAAAFAAAABQAAAAAQABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAD///8AAAAAAAAAAAAAABRE/////////////4fD/////////////xZE/////////////4fD/////////////xhE/////////////4fD/////////////xpE/////////////4fD/////////////x1E/////////////4fD/////////////x9E/////////////4fD/////////////yFE/////////////4fD/////////////yNE/////////////4fD/////////////yZE/////////////4fD/////////////yhE/////////////4fD/////////////ypE/////////////4fD/////////////yxE/////////////4fD/////////////y9E/////////////4fD/////////////zFE/////////////4jD/////////////zFE/////////////43D/////////////zFE/////////////5HD/////////////zFE/////////////5bD/////////////zFE/////////////5rD/////////////zFE/////////////5/D/////////////zFE/////////////6LD/////////////zBE/////////////6LD/////////////y5E/////////////6LD/////////////ytE/////////////6LD/////////////ylE/////////////6LD/////////////ydE/////////////6LD/////////////yVE/////////////6LD/////////////yJE/////////////6LD/////////////yBE/////////////6LD/////////////x5E/////////////6LD/////////////xxE/////////////6LD/////////////xlE/////////////6LD/////////////xdE/////////////6LD/////////////xVE/////////////6LD/////////////xNE/////////////6HD/////////////xNE/////////////53D/////////////xNE/////////////5jD/////////////xNE/////////////5TD/////////////xNE/////////////4/D/////////////xNE/////////////4vD/////////////xNE/////////////4fDUQAAAJAbAABrAQAAbgAAALcBAAC6AAAAaQEAAGwAAAAAAAAAAAAAAFAAAABQAAAAUAAAAEACAACQAgAAABkAAAAAAADGAIgAUAAAAFAAAAAoAAAAUAAAAFAAAAABAAgAAAAAAAAAAAAAAAAAAAAAAIYAAAAAAAAAAAAAAP///wD37ugA9u3oAPTs6ADz6+gA8urnAPHp5wDv5+cA7ubnAO3l5wDs5OYA6uPmAOni5gDo4eYA5uDmAOXf5gDk3uUA493lAOHb5QDg2uUA39nlAN3Y5ADd1+QA29bkANrV5ADY1OQA19PkANbR4wDV0eMA1M/jANLO4wDRzeMAz8zjAM7L4gDNyuIAzMniAMvI4gDJx+IAyMXhAMfE4QDGw+EAxMLhAMPB4QDCwOEAwL/gAL++4AC+veAAvbzgALu74AC6ueAAubjfALe33wC2tt8AtbXfALS03wCys94AsbLeALCx3gCvsN4Ara/eAKyt3gCrrN0AqavdAKiq3QCnqd0ApqjdAKSn3QCjptwAoqXcAKCj3ACfo9wAnqHcAJ2g2wCbn9sAmp7bAJmd2wCYnNsAlpvbAJWa2gCUmdoAkpfaAPXs6AD06+gA8+rnAPDo5wDs5ecA6+TmAOri5gDn4OYA5t/mAOXe5QDi3OUA3tnlANzW5ADZ1OQA2NPkANfS5ADU0OMA08/jANHO4wDQzeMAz8ziAM7K4gDKx+IAycbiAMfF4QDGxOEAxcPhAMPC4QDBwOEAwL7gALy74AC6uuAAurnfALi43wC1tt8As7PfALKy3gCxsd4Arq/eAKyu3gCrrd0AqqzdAKWn3QCkptwAo6XcAKGk3ACfotwAnaHcAJyg2wCXm9sAlpraAJOY2gABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYVRUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFGEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhE9QUYODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4NOT1BRTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU5PUFFLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0xNTk9QUUpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpLTE1OT1BRgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCSktMTU5PUFGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgUlKS0xNTk9QUYCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIBISUpLTE1OT1BRRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGR0hJSktMTU5PUFF/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f39/f0ZHSElKS0xNTk9QUX5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5+fn5FRkdISUpLTE1OT1BRfX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19fX19REVGR0hJSktMTU5PUFF8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fENERUZHSElKS0xNTk9QUUJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQ0RFRkdISUpLTE1OT1BRQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQUJDREVGR0hJSktMTU5PUFE/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/P0BBQkNERUZHSElKS0xNTk9QUXt7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3t7e3s/QEFCQ0RFRkdISUpLTE1OT1BRenp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6enp6Pj9AQUJDREVGR0hJSktMTU5PUFF5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eXl5eT0+P0BBQkNERUZHSElKS0xNTk9QUXh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHh4eHg8PT4/QEFCQ0RFRkdISUpLTE1OT1BROzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFF3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3d3dzo7PD0+P0BBQkNERUZHSElKS0xNTk9QUXZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnY5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRdXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFE3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUXR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHQ2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRNDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3NzczQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUXJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxcXFxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcHBwcDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUTAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFvb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vb29vby4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUW5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubm4tLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFtbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbSssLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGwqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRa2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2tra2trKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFqampqampqampqampqampqampqampqampqampqampqampqampqaigpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWknKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoaGhoJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFElJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUSQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJSYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnZ2dnIyQlJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZmZiIjJCUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWUhIiMkJSYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkICEiIyQlJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjYx8gISIjJCUmJygpKissLS4vMDEyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTk9QUWJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmIeHyAhIiMkJSYnKCkqKywtLi8wMTIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OT1BRHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR4fICEiIyQlJicoKSorLC0uLzAxMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU5PUFFhYWFh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DAAAAAQAAABGAAAAFAAAAAgAAABUTlBQBgEAAFEAAABABAAAXQAAALMAAACpAAAA/wAAAFsAAACxAAAAAAAAAAAAAABQAAAAUAAAAFAAAAAwAAAAgAAAAMADAAAAAAAAhgDuAFAAAABQAAAAKAAAAFAAAABQAAAAAQABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAD///8AAAAAAAAAAAAAABRE/////////////4fD/////////////xZE/////////////4fD/////////////xhE/////////////4fD/////////////xpE/////////////4fD/////////////x1E/////////////4fD/////////////x9E/////////////4fD/////////////yFE/////////////4fD/////////////yNE/////////////4fD/////////////yZE/////////////4fD/////////////yhE/////////////4fD/////////////ypE/////////////4fD/////////////yxE/////////////4fD/////////////y9E/////////////4fD/////////////zFE/////////////4jD/////////////zFE/////////////43D/////////////zFE/////////////5HD/////////////zFE/////////////5bD/////////////zFE/////////////5rD/////////////zFE/////////////5/D/////////////zFE/////////////6LD/////////////zBE/////////////6LD/////////////y5E/////////////6LD/////////////ytE/////////////6LD/////////////ylE/////////////6LD/////////////ydE/////////////6LD/////////////yVE/////////////6LD/////////////yJE/////////////6LD/////////////yBE/////////////6LD/////////////x5E/////////////6LD/////////////xxE/////////////6LD/////////////xlE/////////////6LD/////////////xdE/////////////6LD/////////////xVE/////////////6LD/////////////xNE/////////////6HD/////////////xNE/////////////53D/////////////xNE/////////////5jD/////////////xNE/////////////5TD/////////////xNE/////////////4/D/////////////xNE/////////////4vD/////////////xNE/////////////4fDUQAAAOgaAABdAAAAswAAAKkAAAD/AAAAWwAAALEAAAAAAAAAAAAAAFAAAABQAAAAUAAAAJgBAADoAQAAABkAAAAAAADGAIgAUAAAAFAAAAAoAAAAUAAAAFAAAAABAAgAAAAAAAAAAAAAAAAAAAAAAFwAAAAAAAAAAAAAAP///wD37ugA9u7nAPXt5wD07ecA8+3mAPLt5gDx7OYA8OzlAO/s5QDu7OUA7evkAOzr5ADr6+MA6uvjAOnq4wDo6uIA5+riAObq4QDl6eEA5OnhAOPp4ADi6eAA4ejgAODo3wDf6N8A3ujeAN3n3gDc594A2+fdANrn3QDZ5t0A2ObcANfm3ADW5tsA1eXbANTl2wDT5doA0uXaANHk2QDP5NkAzuTYAM3j2ADM49gAy+PXAMri1wDJ4tYAyOLWAMbi1QDF4dUAxOHUAMPh1ADC4NMAweDTAMDg0wC/4NIAvt/SAL3f0gC839EAu9/RALre0QC53tAAuN7QALfezwC23c8Atd3PALTdzgCz3c4AstzNALHczQCw3M0Ar9zMAK7bzACt28wArNvLAKvbywCq2soAqdrKAPHt5gDu7OQA4ejfANDk2QDK49cAxOHVAMLg1AC+4NIAut7QALfdzwCz3M4ArdvLAKvaywABAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1OW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbW1tbTU5LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0xNTlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpaWlpLTE1OSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSktMTU5ISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISEhISElKS0xNTkdHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dHR0dISUpLTE1ORkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGRkZGR0hJSktMTU5FRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUZHSElKS0xNTllZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlFRkdISUpLTE1OQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDREVGR0hJSktMTU5CQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkNERUZHSElKS0xNTkFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFCQ0RFRkdISUpLTE1OWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYQUJDREVGR0hJSktMTU4/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/Pz8/P0BBQkNERUZHSElKS0xNTj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4+Pj4/QEFCQ0RFRkdISUpLTE1OV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXV1dXPj9AQUJDREVGR0hJSktMTU48PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PDw8PD0+P0BBQkNERUZHSElKS0xNTjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs8PT4/QEFCQ0RFRkdISUpLTE1OOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozw9Pj9AQUJDREVGR0hJSktMTU46Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo6Ojo7PD0+P0BBQkNERUZHSElKS0xNTlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlZWVlY5Ojs8PT4/QEFCQ0RFRkdISUpLTE1ONzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3Nzc3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU42NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2NjY2Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTjU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVNTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU40NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTlRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFRUVFQzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU4xMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTIyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTjAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAxMjIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDEyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU4vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8vLy8wMTIyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTlNTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1NTU1MuLzAxMjIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS4vMDEyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU4sLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLCwsLC0tLi8wMTIyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTisrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKysrKyssLS0uLzAxMjIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKywtLS4vMDEyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU4pKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSkpKSorLC0tLi8wMTIyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlIpKissLS0uLzAxMjIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCkqKywtLS4vMDEyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU4nJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJycnJygoKSorLC0tLi8wMTIyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYmJiYnKCgpKissLS0uLzAxMjIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJSUlJicoKCkqKywtLS4vMDEyMjM0NTY3ODk6Ozw9Pj9AQUJDREVGR0hJSktMTU4kJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCQkJCUmJygoKSorLC0tLi8wMTIyMzQ1Njc4OTo7PD0+P0BBQkNERUZHSElKS0xNTiMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMkJSYnKCgpKissLS0uLzAxMjIzNDU2Nzg5Ojs8PT4/QEFCQ0RFRkdISUpLTE1OIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIiIyQl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)Figure 2 Event Log Buffer States

The following table summarizes the event logging operations that can be performed based upon the state of the buffer.

|  | Standby | Free | In Use | Complete | Ready |
| --- | --- | --- | --- | --- | --- |
| Initialize Buffer Group | N/A | N/A | N/A | N/A | N/A |
| Create Buffer | N/A | N/A | N/A | N/A | N/A |
| Map Buffer | Places the buffer into the free state | Error | Error | Error | Error |
| Request Flush | N/A | N/A | Places the buffer into the complete state, then … | Places buffers into the ready state and sends a message | N/A |
| Release Buffer | Error | Error | Error | Error | Places the buffer into the free state |
| Unmap Buffer | Error | Places the buffer into the standby state | Error | Error | Places the buffer into the standby state |
| Delete Buffer | Deallocates the buffer | Error | Error | Error | Error |
| Finalize Buffer Group | N/A | N/A | N/A | N/A | N/A |

### Accessing Event Log Buffers

Buffers are allocated using GPA pages deposited in the root partition’s pool. When successfully created, a set of GPA pages are formed into a single, yet inaccessible buffer. The guest obtains the list of the buffer’s GPA pages and enables read-only access to those pages by mapping the buffer. The buffer’s pages remain read-only accessible until the buffer is successfully unmapped.

### Preparing for Event Logging

The hypervisor supports a fixed set of event log types, each using a specific event log buffer class. When preparing to use an event logging type, the guest should consider the class, the expected rate of event generation and the NUMA topology when deriving the event log group’s geometry and threshold.

The guest should set up the event logging environment for the type as follows:

An adequate number of pages should be available in the partition’s pool for both the logging infrastructure as well as the buffers themselves. If the local buffer class will be used, there should be adequate pages available for each NUMA proximity.

The event log buffer group should be initialized. The number of buffers, the size of each buffer and the completed buffer threshold will be established. The number of buffers is a maximum value. There is no requirement that all buffers are created before enabling event logging for the type and buffers can be added to the group while it is active.

An adequate supply of buffers should be created. If the local buffer class will be used, the buffers should be properly distributed across all NUMA proximities, based upon logical processor/proximity relationships.

The buffers should be mapped for read-only access. The hypervisor will not use a buffer until it has been mapped. In addition, the guest does not know where the buffer’s pages are located until they have been mapped.

Once event logging has been set up, the guest may proceed to enable the logging of events.

### Enabling and Disabling Event Logging

Associated with each event log type is a 64-bit *event source mask* that indicates which event sources are enabled. This gives the root partition fine-grained control over which events are of interest. The meaning of the bits within the mask is specific to the event log type.

Event logging is enabled by setting the relevant bits in the event source mask. This selects the set of sources whose events are to be recorded. In order to enable the event logging type, at least one source must be enabled; that is, the event mask must be non-zero.

Event logging is disabled by clearing all bits in the event source mask. This disables the recording of events from all sources for the event log type. The root partition may subsequently re-enable event logging or may finalize (terminate) event logging altogether.

### Logging Events into Buffers

When an event log type is successfully enabled, logging from the specified event sources can begin. As logging proceeds, free buffers are selected on demand from the event log buffer group as the active buffer and placed in the “in use” state. For types that use the global buffer class, at most one buffer will be active at any time. For types that use the local buffer class, at most one buffer per logical processor will be “in use” at any time. Local buffers will be selected based upon its logical processor. When any “in use” buffer cannot hold another entry (or active buffers are flushed by the guest), it is placed in the “complete” state. When the number of buffers in this state reaches the threshold value defined at initialization time, the hypervisor places the buffers into a “ready” state and sends an “event log buffers ready” message to the guest. The message passes a list of buffers that are ready for the guest to examine. Note that the list may contain more buffers than the threshold value setting.

A buffer header will be present at the start of each buffer, possibly followed by a series of event log entries. These entries are added to the buffer in a packed, sequential manner. There may be unused space at the end of the buffer. The buffer remains in the control (read-only) of the guest until it is released back into the buffer group to be placed on the free list for reuse.

If the hypervisor is unable to obtain a free buffer when an event is ready to be recorded, the event will be lost. A mechanism to detect the loss of event messages is provided.

### Event Log Buffers Ready Notification

During normal event log operation, buffers transition to the *HvEventLogBufferStateComplete* state and are placed at the end of a *pending buffer list,* awaiting guest notification. Buffers of the global class are placed onto a single, global list. Buffers of the local class are placed onto a per-logical processor list.

Buffers may be placed into the *HvEventLogBufferStateComplete* state for either of the following reasons:

* The buffer does not contain enough free space to record one more event, or
* The caller has requested that the active buffers for the type be flushed.

When either the count of buffers on a pending buffer list reaches the threshold value or buffers are explicitly flushed by the guest, the hypervisor transitions the buffers to the “ready state and posts an “event log buffers ready” notification message to SINT0 of one of the root partition’s virtual processors. For event log types that use the local buffer class, the virtual processor chosen corresponds with the logical processor that produced the pending buffer list (root virtual processors have a hard affinity with logical processors, as described in section 22.4). Lists of pending global class buffers will be directed to any available virtual processor.

The message posted by the hypervisor includes both the event log type and the index of the first buffer in a *notification buffer list*. Buffers forwarded to the guest on the notification list are removed from the pending list and the count of pending buffers is reset to zero. Note that the number of actual buffers present on a notification list may exceed the threshold value. The hypervisor attempts to report as many ready buffers as possible with each message. Buffer lists are discussed in detail in section 19.1.11.

The guest is expected to respond to the message by reading the content of each of the buffers (perhaps writing the data to disk) and then releasing each of them back to the hypervisor so they will be returned to the free list and reused. Buffers on a notification list may be released to the hypervisor in any order. For a detailed description of the notification message format, see section .

### Completed Buffer Lists

The ready buffers message generated by the hypervisor forwards a list of pending buffers to the guest. The message contains the event log buffer type and the index of the first buffer in the list. The *NextBufferIndex* field of the event log buffer header contains the index of the next buffer in the list. Buffers appear in the order in which they were appended to the list. The last buffer of the list contains the value HV\_EVENTLOG\_BUFFER\_INDEX\_NONE, indicating the end of the list.

### Buffer Access Restrictions

Once a buffer is mapped by the guest, it remains read-only accessible to the guest until it is subsequently unmapped. During this period, the buffer may be in one of several states (as discussed in section 19.1.5). The guest will be able to access buffers that are in the free, in-use, complete and ready states (buffers in the standby state are not mapped and are therefore not accessible).

Buffers in the free state can be allocated by the hypervisor at any time. A free buffer has no meaningful content and therefore is of no interest to the guest. A guest may wish to deallocate a free buffer. Deallocation is a two-step process that involves unmapping, then deleting the buffer. Unmapping the buffer transitions it from the free to the standby state, where it can no longer be allocated for use by the hypervisor. Note that the hypervisor may allocate a free buffer between the time the guest examines the state and attempts to unmap it.

In-use buffers are allocated by the hypervisor and are being filled with individual event entries. Entries are appended to the buffer until it is full, at which time it transitions from the in-use to the complete state. In-use buffers can also be flushed by the guest, placing them into the complete state. While in the in-use state, the buffer’s header is subject to modification at any time. Modification however, will only cause new entries to be appended to the buffer and header updates are performed after the event data is copied. The guest may examine in-use buffers at any time but should be aware that more event data could be appended at any time.

Complete buffers are placed onto a list awaiting guest notification. Buffers in this state no longer accept new entries and are therefore ready for examination. Note that the hypervisor could transition these buffers to the ready state at any time. The NextBufferIndex field of the header is subject to change at any time and should not be examined. The buffer’s content however, will not change.

Ready buffers have either been reported to the guest or notification is in progress (the message may be pending delivery). These buffers are in a guest-controlled state and the hypervisor cannot perform any operations them until they are released. All fields of the buffers in the ready state that have been delivered to the guest are stable and can no longer be changed by the hypervisor.

Since buffers in the in-use and complete state can transition state at any time, it is recommended that guests limit their access to event log buffers in the ready state that have been reported to them using the “complete buffers notification” message (see section 16.5.10). Such buffers remain in the ready state until the exclusive control of the guest until it takes action to cause a change in state.

### Adding and Removing Buffers While Event Logging is Active

The guest may add new buffers to the event log buffer group at any time. New buffers enter the internal “standby” state until they are mapped, at which time they enter service as an available free buffer.

The guest may likewise remove buffers from the event log buffer group, but must do so in a special sequence. Removing a buffer begins by unmapping it. The unmap event log buffer hypercall returns a buffer to the internal “standby” state thereby taking the buffer out of service. The buffer may only be unmapped when it is either in the “free” or “ready” state (a flush operation can be used to bring “in use” buffers to the “completed” state and, subsequently, all “completed” buffers to the “ready” state). An unmapped buffer can be deleted at any time.

### Concluding Event Logging

The conclusion procedure for event logging of a specific type is symmetrical with the preparation procedure. The guest should finalize the event logging environment for the type as follows:

Disable all event sources by calling HvSetEventLogGroupSources, specifying the event source mask as zero.

Flush all active buffers by calling HvFlushEventLogBuffer. If the guest wishes to retain the data, it should wait for the event log buffers ready messages to be delivered. Note that if the local buffer class is being used, a message may be generated for each logical processor. The guest should release the buffers back to the hypervisor to be placed onto the free list by calling HvReleaseEventLogBuffer.

The guest waits until all buffers are free before proceeding.

All mapped buffers are unmapped by calling HvUnmapEventLogBuffer. This revokes the read-only access to the buffer’s pages.

All buffers are deleted by calling HvDeleteEventLogBuffer. Deleting the buffers returns their pages to the partition’s pool from whence they came.

The event log buffer group is finalized by calling HvFinalizeEventLogGroup. Any pages allocated for the group’s infrastructure are returned to the partition’s pool.

Once event logging is concluded, excess free pages may be withdrawn from the partition’s pool.

## Event Logging Data Types

### Event Log Types

The hypervisor event logging interfaces are designed to be extensible and support new types in the future. The supported types are defined by the following enumeration.

typedef enum

{

HvEventLogTypeGlobalSystemEvents = 0x00000000

HvEventLogTypeLocalDiagnostics = 0x00000001,

} HV\_EVENTLOG\_TYPE;

| Event Log Type | Usage | Scope |
| --- | --- | --- |
| Diagnostics | Various internal events that can be used for debugging and performance analysis. | Local |
| System Events | Security auditing, history recording, hardware failures and reconfiguration. | Global |

### Event Enable Flags

Event log events can be selectively enabled or disabled by using a 64-bit mask. The meanings of the bits within this mask depend on the event log type.

// Diagnostic event log type event masks

#define HV\_EVENTLOG\_ENABLE\_DIAG\_ADMIN 0x0000000000000001UI64

#define HV\_EVENTLOG\_ENABLE\_DIAG\_PERF\_GENERAL 0x0000000100000000UI64

// System event log type event masks

#define HV\_EVENTLOG\_ENABLE\_SYSTEM\_SUCCESS 0x0000000000000001UI64

#define HV\_EVENTLOG\_ENABLE\_SYSTEM\_FAILURE 0x0000000000000002UI64

### Event Log Buffer State

Each buffer associated with an enabled event log type is in one of three states, defined by the following enumeration.

typedef enum

{

HvEventLogBufferStateStandby = 0,

HvEventLogBufferStateFree = 1,

HvEventLogBufferStateInUse = 2,

HvEventLogBufferStateComplete = 3,

HvEventLogBufferStateReady = 4

} HV\_EVENTLOG\_BUFFER\_STATE;

When a buffer is in the *HvEventLogBufferStateStandby* state, it is unmapped and not available for use by the hypervisor (note that when in this state the guest cannot access the buffer and therefore cannot see this value set in the state field of the buffer’s header). The buffer enters the *HvEventLogBufferStateFree* state when it is mapped, making it eligible for use. The hypervisor may decide to place it into *HvEventLogBufferStateInUse* state at any time. When a buffer is in *HvEventLogBufferStateInUse* state, the hypervisor may add new event log events to it. When a buffer is placed into the *HvEventLogBufferStateComplete* state, it will no longer be modified by the hypervisor and it will be placed onto the completed buffer list. When the number of buffers on the list reaches the completed buffer threshold, the hypervisor will place all buffers on the list in the *HvEventLogBufferStateReady* state, and notify the guest that it may exclusively read the set of buffers. The message will indicate the start of the buffer list (the buffer header contains the index of the next buffer in the list). The guest should extract any information that it needs from the buffers as quickly and efficiently as possible and release them back to the hypervisor to avoid any potential loss of event data. Releasing a buffer places it in the *HvEventLogBufferStateFree* state, making it immediately available for reuse. Alternatively, the guest may take a buffer out of service by unmapping it, placing it in the *HvEventLogBufferStateStandby* state.

### Event Log Buffer Index

Event log buffers for each type are assigned an event log buffer indexthat is used to identify specific buffers within the type.

typedef UINT32 HV\_EVENTLOG\_BUFFER\_INDEX, \*PHV\_EVENTLOG\_BUFFER\_INDEX;

#define HV\_EVENTLOG\_BUFFER\_INDEX\_NONE 0xffffffff

### Event Log Buffer Header

Each buffer begins with the following header. Event log entry data begins immediately after the buffer header.

typedef struct

{

UINT32 BufferSize;

HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex;

UINT32 EventsLost;

UINT32 ReferenceCounter;

HV\_NANO100\_TIME TimeStamp;

UINT64 Reserved1;

UINT64 Reserved2;

HV\_LOGICAL\_PROCESSOR\_INDEX LogicalProcessor;

HV\_EVENTLOG\_BUFFER\_STATE BufferState;

UINT64 NextBufferOffset;

HV\_EVENTLOG\_TYPE Type;

HV\_EVENTLOG\_BUFFER\_INDEX NextBufferIndex;

UINT64 Reserved3;

UINT64 Reserved4[2];

} HV\_EVENTLOG\_BUFFER\_HEADER;

*BufferSize* is the size of the buffer in bytes*.*

*BufferIndex* is the hypervisor-assigned index of the buffer within this event log type. It was returned to the caller when the buffer was created.

*LogicalProcessor* is used only for local buffers, where it indicates which logical processor is responsible for the buffer. This field is undefined when using global buffers and when the buffer state is “free”.

*EventsLost* is the count of events that have not been recorded since the last event log buffer was placed into the complete state. It indicates that there were no free buffers available for the hypervisor to record these events and they have therefore been lost. When buffers become available, this counter is reset to zero. For event log types that use local buffers, this value is maintained on a logical processor basis.

*TimeStamp* records the time at which the buffer entered the “in use” state. This value represents the number of 100ns time units that have elapsed since the hypervisor was started. It can be used to determine the buffer fill order (that is, which buffers were “complete” first).

*EventCount* is an increasing counter that represents the sequential event number of the first event in the buffer. For event log types that use local buffers, this value is maintained on a logical processor basis. This value may also wrap on overflow.

*BufferState* indicates whether the buffer is currently free, in use, or complete.

*NextBufferOffset* indicates the byte offset into the buffer where the next entry will be written. This field is undefined if the buffer is free. When a buffer is marked “in use”, this value is initially set to the size of the buffer header to indicate that the next entry will be placed immediately after it. The hypervisor guarantees that entries written before this offset, if any, are complete. In other words, this value is updated only after an event has been recorded within the buffer. This allows a guest to safely read the partial content of a buffer before it enters the complete state.

*Type* indicates the Event Log Type that the buffer is related to. The combination of *Type* and *BufferIndex* can be used to identify a specific buffer.

*NextBufferIndex* contains the event log buffer index of the next buffer in the completion list, or the value HV\_EVENTLOG\_BUFFER\_INDEX\_NONE if this is the last one. This field is only valid for buffers in the HvEventLogBufferStateReady state.

### Event Log Entry Header

One or more event log entries immediately follow the buffer header of each complete event log buffer. Each entry is composed of an event log entry header and, optionally, related event data. The total size of the header and any data must be aligned (by padding) to an 8-byte boundary.

typedef enum

{

HV\_EVENTLOG\_ENTRY\_TIME\_REFERENCE = 0,

HV\_EVENTLOG\_ENTRY\_TIME\_TSC = 1

} HV\_EVENTLOG\_ENTRY\_TIME\_BASIS;

typedef union

{

HV\_NANO100\_TIME Reference;

UINT64 Tsc;

} HV\_EVENTLOG\_ENTRY\_TIME;

typedef struct

{

UINT32 Context;

UINT16 Size;

UINT16 Type;

HV\_EVENTLOG\_ENTRY\_TIME TimeStamp;

} HV\_EVENTLOG\_ENTRY\_HEADER;

*Context* is an implementation-specific field available to the event logging type.

*Size* is the size of the entry and is expressed in bytes. It includes the header and any padding bytes required to place the next entry on an 8-byte boundary.

*Type* is the entry type, specific to the event log type. The definitions of the entry types can be found in 0

*TimeStamp* records the time at which the entry was placed into the buffer. The source of the value was specified when the event log buffer group was created.

## Event Logging Interfaces

### HvInitializeEventLogBufferGroup

The HvInitializeEventLogBufferGroup hypercall defines the geometry (that is, the size and count) and establishes a message completion threshold for the event log buffer group associated with a specific type.

**Wrapper Interface**

HV\_STATUS

HvInitializeEventLogBufferGroup(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in UINT32 MaximumBufferCount,

\_\_in UINT32 BufferPages,

\_\_in UINT32 Threshold,

\_\_in HV\_EVENTLOG\_ENTRY\_TIME\_BASIS TimeBasis,

\_\_in HV\_NANO100\_TIME SystemTime

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvInitializeEventLogBufferGroup | | |
|  | Call Code = 0x0060 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | MaximumBufferCount (4 bytes) |
| 8 | BufferPages (4 bytes) | Threshold (4 bytes) |
| 16 | TimeBasis (4 bytes) | SystemTime (8 bytes) |
| 24 | SystemTime | Padding (4 bytes) |

Description

HvInitializeEventLogBufferGroup establishes the buffer size, maximum number of buffers, completion threshold and time basis for the event log buffer group. It allocates pages necessary to support the event logging infrastructure for the type. Pages are allocated from the guest’s pool.

Input Parameters

*Type* specifies the type of the associated group to be initialized.

*MaximumBufferCount* is the maximum number of buffers that can exist for the associated event log buffer group at any time. Valid buffer index values are between zero and *MaximumBufferCount*-1.

*BufferPages* is the number of pages that will constitute each buffer for the associated buffer group.

*Threshold* is the minimum number of buffers that must be completed before the hypervisor will post a buffer completion message.

*TimeBasis* is the source of the event header timestamp. The value HV\_EVENTLOG\_ENTRY\_TIME\_REFERENCE indicates that the source is the hypervisor’s reference time (time since the hypervisor was started). The value HV\_EVENTLOG\_ENTRY\_TIME\_TSC indicates that the source is the TSC of the logical processor posting the event to the buffer.

*SystemTime* is the current system time in 100 ns units.

Output Parameters

None.

Restrictions

The caller must be the root partition.

The event log type must not be previously initialized.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
|  | *BufferPages* specifies that buffers are composed of fewer than 1 or more than 512 pages. |
|  | *MaximumBufferCount* specifies fewer than 1 or more than 512 buffers. |
|  | The *Threshold* parameter specifies an inappropriate value. |
|  | *TimeBasis* does not specify either HV\_EVENTLOG\_ENTRY\_TIME\_REFERENCE or  HV\_EVENTLOG\_ENTRY\_TIME\_TSC. |
| HV\_STATUS\_OBJECT\_IN\_USE | The event log buffer group is already initialized. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the partition’s memory pool is insufficient to perform the operation. |

### HvFinalizeEventLogBufferGroup

The HvFinalizeEventLogBufferGroup hypercall destroys the event log buffer group associated with the specified type.

**Wrapper Interface**

HV\_STATUS

HvFinalizeEventLogBufferGroup(

\_\_in HV\_EVENTLOG\_TYPE Type

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvFinalizeEventLogBufferGroup [fast] | | |
|  | Call Code = 0x0061 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | Padding (4 bytes) |

Description

The HvFinalizeEventLogBufferGroup hypercall is used to destroy the event log buffer group and return all allocated resources for the group’s infrastructure to the partition’s pool. The group should contain no buffers (all buffers should have been unmapped and deleted) when HvFinalizeEventLogBufferGroup is called or an error will be returned.

Input Parameters

*Type* specifies the event log type to be finalized.

Output Parameters

None.

Restrictions

The caller must be the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | There is at least one buffer remaining in the event log group associated with this type. |
| The specified event log type has not been initialized. |
| All sources for the specified event log type are not disabled. |

### HvCreateEventLogBuffer

The HvCreateEventLogBuffer hypercall allocates a single event log buffer for the specified event log buffer group.

**Wrapper Interface**

HV\_STATUS

HvCreateEventLogBuffer(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex,

\_\_in HV\_PROXIMITY\_DOMAIN\_INFO ProximityInfo

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvCreateEventLogBuffer [fast] | | |
|  | Call Code = 0x0062 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | BufferIndex (4 bytes) |
| 8 | ProximityInfo (8 bytes) | |

Description

The HvCreateEventLogBuffer hypercall is used to allocate an event log buffer for the specified event log buffer type. The size of the buffer must have previously been established by a call to HvInitializeEventLogBufferGroup. For types that use local buffers, the hypervisor selects pages for the buffer from the NUMA node described by *ProximityInfo*. The caller assigns the buffer a unique buffer index that may not already be in use and which must be between 0 and the maximum (less one) established at buffer group initialization time.

Input Parameters

*Type* specifies the event log type of the buffer group whose pages are to be mapped.

*ProximityInfo* specifies the NUMA proximity that the buffer’s pages should be allocated from. This is mandatory when allocating local buffers and optional when the event log type uses global buffers. The Proximity Domain specified is described in section 7.2.1.

*BufferIndex* is the buffer index number to be assigned to the created buffer.

Output Parameters

None.

Restrictions

The caller must be the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| The specified proximity domain information is invalid. |
| The specified buffer index is either already in use or is not within the limit as established at event log group initialization time. |
| HV\_STATUS\_OPERATION\_DENIED | The event log buffer group associated with this type has not been initialized. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | Insufficient memory exists for the call to succeed. |

### HvDeleteEventLogBuffer

The HvDeleteEventLogBuffer hypercall deallocates a single event log buffer for the specified event log buffer group.

**Wrapper Interface**

HV\_STATUS

HvDeleteEventLogBuffer(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvDeleteEventLogBuffer [fast] | | |
|  | Call Code = 0x0063 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | BufferIndex (4 bytes) |

Description

The HvDeleteEventLogBuffer hypercall is used to deallocate a specific event log buffer for the specified event log buffer type. The buffer must not be mapped. The pages that constitute the buffer are returned to the partition’s pool from whence they came.

Input Parameters

*Type* specifies the event log type of the buffer group that this buffer is a member of.

*BufferIndex* is the buffer index number associated with the buffer to be deleted.

Output Parameters

None.

Restrictions

The caller must be the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| A buffer with the specified index does not exist. |
| HV\_STATUS\_OPERATION\_DENIED | The event log buffer group has not been initialized. |
| The event log buffer associated with this index and type has not been unmapped. |

### HvMapEventLogBuffer

The HvMapEventLogBuffer hypercall returns the buffer’s GPAs and read-only maps them into the guest’s address space.

**Wrapper Interface**

HV\_STATUS

HvMapEventLogBuffer(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex,

\_\_out PHV\_GPA\_PAGE\_NUMBER GpaPages

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvMapEventLogBuffer | | |
|  | Call Code = 0x0064 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | BufferIndex (4 bytes) |
| 🡄 Output Parameters | | |
| 0 | GpaPages (8 bytes) (Array of HV\_GPA\_PAGE\_NUMBER values, the number determined by the *BufferPages* parameter of the HvIntializeEventLogBufferGroup hypercall) | |

Description

The HvMapEventLogBuffer hypercall is used to read-only map the pages of the specified event log buffer into the guest’s address space and to return the list of GPA pages that constitute the buffer. Successfully-mapped buffers are placed onto the free buffer list, making them available for the hypervisor to use for event logging.

Input Parameters

*Type* specifies the event log type of the buffer group whose pages are to be mapped.

*BufferIndex* is the index of the buffer within the buffer group associated with the specified event log type.

Output Parameters

*GpaPages* is an array of GPA page numbers that constitute the event log buffer. All buffers for a specific event log group are the same size. The size is established by the guest using the HvInitializeEventLogBufferGroup hypercall.

Restrictions

The caller must be the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| A buffer with the specified index does not exist. |
| HV\_STATUS\_OPERATION\_DENIED | The event log buffer group has not been initialized. |
| The event log buffer associated with this index and type is already mapped. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | Insufficient memory exists for the call to succeed. |

### HvUnmapEventLogBuffer

The HvUnmapEventLogBuffer hypercall unmaps an existing event log buffer.

**Wrapper Interface**

HV\_STATUS

HvUnmapEventLogBuffer(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvUnmapEventLogBuffer [fast] | | |
|  | Call Code = 0x0065 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | BufferIndex (4 bytes) |

Description

HvUnmapEventLogBuffer revokes the read-only mappings of the pages that compose the specified buffer. Buffers can only be unmapped if they are in the free or ready state. The buffer is no longer available to be used by the hypervisor to log events.

Input Parameters

*Type* specifies the event log buffer type.

*BufferIndex* identifies the buffer to be unmapped.

Output Parameters

None.

Restrictions

The caller must be the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| HV\_STATUS\_ACCESS\_DENIED | The event log buffer is not in the free or ready state. |

### HvSetEventLogGroupSources

The HvSetEventLogGroupSources hypercall allows the caller to enable or disable specific event sources associated with the specific event log type.

**Wrapper Interface**

HV\_STATUS

HvSetEventLogGroupSources(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in UINT64 EnableFlags

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvSetEventLogGroupSources [fast] | | |
|  | Call Code = 0x0066 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | Padding (4 bytes) |
| 8 | EnableFlags (8 bytes) | |

Description

This hypercall allows the guest to enable and disable a set of up to 64 event sources. When *EnableFlags* is zero, event logging is disabled.

Input Parameters

*Type* specifies the type of the event log buffer.

*EnableFlags* is a 64-bit mask that specifies which event sources are enabled. The meaning of each bit in the mask depends on the event log type. In order for tracing to start, this mask must be non-zero; that is, at least one event source must be enabled.

Output Parameters

None.

Restrictions

The caller is not the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | The event log buffer group is not initialized. |

### HvReleaseEventLogBuffer

The HvReleaseEventLogBuffer hypercall releases the completed buffer to the hypervisor as a free buffer.

**Wrapper Interface**

HV\_STATUS

HvReleaseEventLogBuffer(

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvReleaseEventLogBuffer [fast] | | |
|  | Call Code = 0x0067 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | BufferIndex (4 bytes) |

Description

If the event log buffer specified is not in the “ready” state, an error will be returned. This call informs the hypervisor that the guest is finished processing the buffer. The hypervisor will place the buffer into a free state and make it available for reuse.

Input Parameters

*Type* specifies the type of the event log buffer.

*BufferIndex* specifies the index of the buffer within the event log type.

Output Parameters

None.

Restrictions

The caller must be the root partition.

The buffer specified by *BufferIndex* must be in the “complete” state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| The specified index for an event log buffer of this type is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | The event log buffer group has not been initialized. |
| The specified event log buffer is not in the "ready" state. |

### HvFlushEventLogBuffer

The HvFlushEventLogBuffer hypercall forces an “in-use” event buffer to be marked as “complete” and an “event log buffers ready” notification to be queued.

Wrapper Interface

HV\_STATUS

HvFlushEventLogBuffer (

\_\_in HV\_EVENTLOG\_TYPE Type,

\_\_in HV\_EVENTLOG\_BUFFER\_INDEX BufferIndex

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvFlushEventLogBuffer [fast] | | |
|  | Call Code = 0x0068 | |
| 🡆 Input Parameters | | |
| 0 | Type (4 bytes) | BufferIndex (4 bytes) |

Description

The HvFlushEventLogBuffer hypercall places an “in-use” event buffer into the “complete” state, as if the event buffer had filled up through event logging activity. It allows a caller to reclaim a buffer that is partially filled. The call also queues an “event log buffers ready” notification message to a VP within the root partition. This occurs even if the event buffer threshold has not yet been reached.

Input Parameters

*Type* specifies the type of the event log buffer.

*BufferIndex* specifies the index of the buffer within the event log type.

Output Parameters

None.

Restrictions

The caller must be the root partition.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified event log type is invalid. |
| The specified index for an event log buffer of this type is invalid. |
| HV\_STATUS\_OPERATION\_DENIED | The event log buffer group has not been initialized. |
| The event log buffer is not in the "in use" state. |

# Guest Debugging Support

## Overview

Many operating systems support inter-machine debugging that uses a physical connection to support communication between a debugger running on the *host* system and the component being debugged, running on the *target* system.. The most common approach is to use a NULL modem cable to connect a serial port on the host system to a serial port on the target system. Those ports are then used exclusively for debugging traffic. An alternative is to use an IEEE 1394 cable to connect the two systems. This chapter discusses both approaches, highlighting the differences where applicable.

When more than one guest is debugged simultaneously, each partition’s traffic must be multiplexed across the common physical connection. The hypervisor provides a simple set of interfaces to exchange session-related data across the physical connection. The default port used for the connection is the serial COM port.

The implementation is almost entirely protocol independent. From the hypervisor’s perspective, the debugging data exchanged by the two systems is simply a sequenced stream of opaque data. The exception is an additional multiplex message that translates into an NT debugger break-in message for the client. Non-NT debuggers do not need to handle this message for debugging to continue to work.

Note that because of the volatile nature of the connection, the hypervisor does not guarantee that any or all of the data will be delivered. The host and target systems should be aware of this and implement all appropriate synchronization and recovery mechanisms. Note that the hypervisor must have guest debugging support enabled. To detect the availability of the feature, use a hypervisor CPUID leaf. For details see section .

### Debug Sessions

Each guest that uses the debugging facility manages a single *session* between itself and the external debugger. A partition must possess the *Debugging* privilege to use a debug session (see section 5.2.3). This privilege is granted at partition creation and checked when the partition is initialized to allow the system to reserve resources for the incoming and outgoing debugging data streams. Debugging resources are allocated from the partition’s pool and provide buffers for incoming and outgoing session data. At a minimum, the outgoing buffer is large enough to accept the maximum amount of session data that can be posted by a single hypercall. The debugging privilege also grants access to the debugging hypercalls, HvPostDebugData, HvRetrieveDebugData and HvResetDebugSession.

The debug session is assigned a unique *debugging channel identifier* that identifies the session over the physical connection. The identifier is an assigned partition property value that is 16 bits wide and must either be zero or a non-zero value that is unique on the target system. A debugging channel identifier of zero indicates a channel ID is not assigned, and the debugging hypercalls return an error. For information regarding partition properties, see section .

When the partition is saved, any debugging-related data that has been posted or not yet received is not part of the partition’s save context. When the partition is subsequently restored, the *Debugging* privilege must be granted at partition creation to ensure that the necessary session infrastructure is manifested by partition initialization. The session appears in a reset state, and the guest is expected to sense the loss of synchronization recover accordingly.

When the partition is finalized, the process implicitly resets the debugging session and releases any related resources that were allocated during partition initialization. Any debugging data in the incoming and outgoing buffers is discarded.

### Transmitting Data on a Session

When a guest wishes to transmit data to an external debugger, it does so by *posting* the data to the session. The hypervisor sends the posted data to the host system asynchronously (for serial debugging) or synchronously (for 1394 debugging) to the host system. The hypervisor buffers the outgoing data using the buffer space that was allocated for the session when the partition was initialized. If sufficient free space is not available for all of the data, the hypervisor returns an error and does not post any data.

### Retrieving Data from a Session

The hypervisor retrieves any outstanding received data for the session. The data is buffered using space that was allocated when the partition was initialized. If sufficient free space is not available when data arrives on the connection, the excess bytes are discarded.

The guest should request a buffer of the maximum size permitted by the hypercall. If the buffer has no data, the call returns an error. If the attempt to retrieve data is successful, more data might still be available. The caller should process the retrieved data and attempt to retrieve more. This process should be repeated until the hypercall returns an error indicating that all pending receive data has been retrieved.

### Purging Data from a Session

If a guest detects a loss of synchronization with its remote counterpart, the guest can reset the session, selectively discarding all of the pending received or posted transmitted data for the session.

### Sensing Activity on the Physical Connection

The hypervisor maintains a simple, global count of the number of bytes received over the debugging connection. The hypervisor increments this count each time a byte is received. The counter is permitted to wrap. A changing counter value is a good indication of activity, because it indicates that the session is active and the connection is intact. (Note that a changing counter value does not guarantee an active session, because the hypervisor could misconstrue noise as received data.)

The hypervisor also maintains a local, session-specific copy of the global counter that is updated from the global counter whenever the guest attempts to retrieve data from the session. If a guest wishes to test for incoming activity, it can do so by setting the appropriate options when it calls the post or retrieve hypercalls. If the options are set, the hypercall compares the local and global counters before doing anything else. If the two values match, then no receive activity has occurred on the physical connection since the local counter was last updated, and the hypercall returns an error. Guests can use this feature to detect inactivity and implement a time-out mechanism.

### Hypercall Loops

During a debugging session, guests often enter tight loops as they attempt to post or retrieve session data until they are successful. The hypercalls used for this purpose allow the guest to provide a hint to the hypervisor that it intends to behave this way. The hypervisor is aware of the state of the debug session buffers and might elect to use the hint under either of the following circumstances:

* An attempt by the guest to retrieve data when the incoming data buffer is empty, or
* An attempt by the guest to post data when the outgoing data buffer has insufficient space.

When the hypervisor encounters either circumstance, it can use the hint and its knowledge of the connection and session state to do one of the following:

Return control to the guest, allowing it to make another attempt.

Suspend the virtual processor for the amount of time specified by the caller. The suspension could be terminated early if either of the following are true:

The guest is retrieving debugging data and has received new incoming data.

The guest is posting debugging data and adequate free outgoing buffer space has become available.

## Debugging Data Types

The following data types support the debugging facility.

### Debug Types

The debugging interfaces exchange simple blocks of data between a guest and an external port. The size of the blocks is not fixed, but cannot exceed a maximum size. The options that are accepted by the hypercalls and that guests use to implement debugging are defined as follows:

// Debug channel identifier

typedef UINT16 HV\_DEBUG\_CHANNEL\_IDENTIFIER;

// Maximum size of the payload

#define HV\_DEBUG\_MAXIMUM\_DATA\_SIZE 4088

// Debug options for all calls

typedef UINT32 HV\_DEBUG\_OPTIONS;

// Options flags for HvPostDebugData

#define HV\_DEBUG\_POST\_LOOP 0x00000001

// Options flags for HvRetrieveDebugData

#define HV\_DEBUG\_RETRIEVE\_LOOP 0x00000001

#define HV\_DEBUG\_RETRIEVE\_TEST\_ACTIVITY 0x00000002

// Options flags for HvResetDebugSession

#define HV\_DEBUG\_PURGE\_INCOMING\_DATA 0x00000001

#define HV\_DEBUG\_PURGE\_OUTGOING\_DATA 0x00000002

## Debugging Interfaces

The following sections describe the debug facility hypercall interfaces.

### HvPostDebugData

The HvPostDebugData hypercall posts a block of data for transmission over the connection.

Wrapper Interface

HV\_STATUS

HvPostDebugData(

\_\_in UINT32 Count,

\_\_in HV\_DEBUG\_OPTIONS Options,

\_\_in\_bcount(HV\_DEBUG\_MAXIMUM\_DATA\_SIZE, Count)

PUINT8 Data,

\_\_out PUINT32 PendingCount

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvPostDebugData | | |
|  | Call Code = 0x0069 | |
| 🡆 Input Parameters | | |
| 0 | Count (4 bytes) | Options (4 bytes) |
| 8 | Data (up to HV\_DEBUG\_MAXIMUM\_DATA\_SIZE bytes) | |
| 🡄 Output Parameters | | |
| 0 | PendingCount (4 bytes) | Padding (4 bytes) |

Description

Users call the HvPostDebugData hypercall to post a block of data. The block can be as large as HV\_DEBUG\_MAXIMUM\_DATA\_SIZE bytes in size. If the hypervisor’s outgoing data buffer has enough space for the entire block of data, it is posted for transmission. Otherwise, HvPostDebugData returns an error and no data is posted. Users can call HvPostDebugData with *Count* set to zero to determine whether any pending data is present in the outgoing session buffer.

Input Parameters

*Count* is the number of bytes to be posted. It can range from 0 to HV\_DEBUG\_MAXIMUM\_DATA\_SIZE.

*Options* specifies the following hypercall options:

HV\_DEBUG\_POST\_LOOP indicates that the guest will continue to call HvPostDebugData until it completes successfully.

*Data* is an opaque stream of bytes for the hypervisor to post for transmission.

Output Parameters

*PendingCount* is the total number of bytes in the outgoing session buffer that are pending transmission, including any data posted by this call.

Restrictions

The caller’s partition must possess the *Debugging* privilege.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_FEATURE\_UNAVAILABLE | Guest debugging support is not present or not enabled in the hypervisor. |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition does not possess the *Debugging* privilege. |
| HV\_STATUS\_OPERATION\_DENIED | The partition’s *Debug Channel Identifier* property has not been assigned a unique, non-zero value. |
| HV\_STATUS\_INVALID\_PARAMETER | The *Count* parameter specifies more than HV\_DEBUG\_MAXIMUM\_DATA\_SIZE bytes of data. |
| The area described by *Data* and *Count* crosses a page boundary. |
| *Options* specified a flag other than HV\_DEBUG\_POST\_LOOP. |
| HV\_STATUS\_INSUFFICIENT\_BUFFERS | The outgoing session buffer does not have sufficient space to accept all of *Data.* |

### HvRetrieveDebugData

The HvRetrieveDebugData hypercall retrieves received data.

**Wrapper Interface**

HV\_STATUS

HvRetrieveDebugData(

\_\_in\_out PUINT32 Count,

\_\_in HV\_DEBUG\_OPTIONS Options,

\_\_in HV\_NONA100\_DURATION Timeout,

\_\_out PUINT32 RetrievedCount,

\_\_out PUINT32 RemainingCount,

\_\_out\_bcount(HV\_DEBUG\_MAXIMUM\_DATA\_SIZE, \*Count)

PUINT8 Data

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvRetrieveDebugData | | |
|  | Call Code = 0x006A | |
| 🡆 Input Parameters | | |
| 0 | Count (4 bytes) | Options (4 bytes) |
| 8 | Timeout (8 bytes) | |
| 🡄 Output Parameters | | |
| 0 | RetrievedCount (4 bytes) | RemainingCount (4 bytes) |
| 8 | Data (up to HV\_DEBUG\_MAXIMUM\_DATA\_SIZE bytes) | |

Description

HvRetrieveDebugData retrieves up to HV\_DEBUG\_MAXIMUM\_DATA\_SIZE bytes from the session’s received data stream and returns an error if no data is available. A caller can set *Count* to zero to simply determine whether any received data is present in the incoming session buffer. If data is present, HvRetrieveDebugData returns a success code but does not return any data.

Input Parameters

*Count* specifies the maximum number of bytes to be retrieved. This value can range from 0 to HV\_DEBUG\_MAXIMUM\_DATA\_SIZE.

*Options* specifies the hypercall options:

HV\_DEBUG\_RETRIEVE\_LOOP indicates that if there is no immediately available data, the hypervisor should suspend the virtual processor for the interval specified by *Timeout*.

HV\_DEBUG\_RETRIEVE\_TEST\_ACTIVITY indicates that if no receive activity (for any session) has occurred on the physical connection since the last HvRetrieveDebugData hypercall, the hypercall should return an error. HvRetrieveDebugData performs this test before it does any other processing.

*Timeout* specifies the maximum time, in 100 nanosecond units, that the virtual processor will be suspended while waiting for data to arrive.

Output Parameters

*RetrievedCount* contains the number of bytes returned in *Data*.

*RemainingCount* contains the number of bytes in *Data* that are still pending, after *Count* bytes have been returned.

*Data* contains RetrievedCount bytes data from the beginning of the session’s received data stream.

Restrictions

The caller’s partition must possess the *Debugging* privilege.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_FEATURE\_UNAVAILABLE | Guest debugging support is not present or not enabled in the hypervisor. |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition does not possess the *Debugging* privilege. |
| HV\_STATUS\_OPERATION\_DENIED | The partition’s *Debug Channel Identifier* property has not been assigned a unique, non-zero value. |
| HV\_STATUS\_INVALID\_PARAMETER | The *Count* parameter specifies more than HV\_DEBUG\_MAXIMUM\_DATA\_SIZE bytes of data. |
| The area described by *Data* and *Count* crosses a page boundary. |
| *Options* specified a flag other than HV\_DEBUG\_RETRIEVE\_LOOP or HV\_DEBUG\_RETRIEVE\_TEST\_ACTIVITY. |
| HV\_STATUS\_INACTIVE | *Options* specified HV\_DEBUG\_RETRIEVE\_TEST\_ACTIVITY and the hypervisor has not recorded any receive activity on the physical connection since the guest’s last call to HvRetrieveDebugData. |
| HV\_STATUS\_NO\_DATA | The incoming session buffer has no pending data. |

### HvResetDebugSession

The HvResetDebugSession hypercall resets the incoming or outgoing buffer context.

Wrapper Interface

HV\_STATUS

HvResetDebugSession(

HV\_DEBUG\_OPTIONS Options

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvResetDebugSession [fast] | | |
|  | Call Code = 0x006B | |
| 🡆 Input Parameters | | |
| 0 | Options (4 bytes) | Padding (4 bytes) |

Description

The HvResetDebugSession hypercall directs the hypervisor to purge all internal receive or transmit buffers for the session. Any pending data is lost. Callers use this hypercall to reset the state of the connection.

Input Parameters

*Options* specifies which session buffers should be purged. Callers can set either or both of the following flags:

HV\_DEBUG\_PURGE\_INCOMING\_DATA purges the receive buffer.

HV\_DEBUG\_PURGE\_OUTGOING\_DATA purges the transmit buffer.

Output Parameters

None.

Restrictions

The caller’s partition must possess the *Debugging* privilege.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_FEATURE\_UNAVAILABLE | Guest debugging support is not present or not enabled in the hypervisor. |
| HV\_STATUS\_ACCESS\_DENIED | The caller’s partition does not possess the *Debugging* privilege. |
| HV\_STATUS\_OPERATION\_DENIED | The partition’s *Debug Channel Identifier* property has not been assigned a unique, non-zero value. |
| HV\_STATUS\_INVALID\_PARAMETER | The *Options* parameter includes an unsupported flag or includes neither of the supported flags. |
| HV\_STATUS\_NO\_DATA | No data was purged from the buffers specified by the *Options* flags. |

# Statistics

## Overview

The hypervisor maintains simple statistics counters for various events. The list of events is implementation-specific and likely to differ from one revision to the next. The following definitions help to set the context for further reading:

A *counter* monitors a specific item of the system.

An *object* is a collection of counters that are related to the same measurable entity.

A *counter instance* refers to a particular instantiation of an object that a counter can monitor.

### Global Versus Local Statistics

Two classes of statistics counters are provided: global and local. Counters of the local class are local to a specific partition or objects associated with a partition. Counters of the global class are associated with the entire hypervisor or objects that are not associated with a specific partition.

Global statistics classes include:

Hypervisor

Logical processors

Local statistics classes include:

Partitions

Virtual processors

Additional objects for which counters of global and local classes are gathered may be added in future hypervisor implementations.

### Statistics Page Mappings

Memory for statistics counter instances is allocated automatically by the hypervisor when certain objects are allocated. For example, when a virtual processor is allocated, space for that virtual processor’s associated statistics is also allocated.

One page of memory is allocated to store the counter instances for each object and is called a *stats page*. This allows individual counter types to be mapped into a partition’s address space for easy retrieval. This mapping is performed using overlay pages. For a discussion of overlay pages, see section .

Stats pages mapped into a partition’s GPA space are write protected. Attempts to write to these pages will result in a #MC fault. The hypervisor writes to these pages by using writeback cacheability. For correct behavior, guests are recommended to map these pages with writeback caching as well.

The lifetime of a stats page is the same as the lifetime for its corresponding object. If that object is deleted and its stats page is still mapped within a partition’s GPA space, that stats page mapping is removed. As with any overlay page, removal of a stats page mapping reveals whatever is “beneath” the overlay page.

### Format of Statistics Pages

Statistics pages consist of a sequence of tightly-packed *statistics* *groups*, each prefixed by a header:

typedef UINT16 HV\_STATISTICS\_GROUP\_TYPE;

typedef union

{

UINT32 VersionMajorMinor;

struct

{

UINT32 VersionMinor;

UINT32 VersionMajor;

} Version;

} HV\_STATISTICS\_GROUP\_VERSION;

typedef UINT16 HV\_STATISTICS\_GROUP\_LENGTH;

// Actual header format

typedef struct

{

HV\_STATISTICS\_GROUP\_TYPE Type;

HV\_STATISTICS\_GROUP\_VERSION Version;

HV\_STATISTICS\_GROUP\_LENGTH Length;

} HV\_STATISTICS\_GROUP\_HEADER;

Each statistics group has an associated identifier (TYPE) and version number present within its header. The group identifier is unique within the statistics object. The version number is composed of a major and minor number. The length field indicates how many bytes constitute the group. It does not include the size of the group header and is aligned to an 8-byte boundary. Adding the length to the base address of the group’s data will lead to the next group header.

The last group of data within the page is followed by a header indicating the end of the list, with both the version number and length set to zero. Using this format allows the hypervisor to support a variety of both architectural and implementation-specific counters. See for details of the various statistics group formats.

### Statistics Group Compatibility

As new versions of the hypervisor are released and counters are added to the end of a group, the minor number will be incremented. This will allow existing code to be compatible with future hypervisors. A check to ensure that the group’s major version number matches and that the minor version number is greater-than-or-equal to the compile-time value will suffice.

Should the format of the group ever change, the major number should be incremented. Note that groups containing architectural counters should not change major version numbers from release to release. A particular counter will always be present at its assigned location but may no longer be maintained. In such cases the counter will always read as zero.

// Standard header types that have zero version and length

#define HV\_STATISTICS\_TYPE\_END\_OF\_LIST 0

#define HV\_STATISTICS\_TYPE\_END\_OF\_PAGE 1

// Definitions for the hypervisor counters statistics page

#define HV\_STATISTICS\_TYPE\_HCA\_ID 2

#define HV\_STATISTICS\_TYPE\_HCA\_VERSION ((1 << 16) | 0) // V1.0

// Definitions for the logical processor counters statistics page

#define HV\_STATISTICS\_TYPE\_LPA\_ID 2

#define HV\_STATISTICS\_TYPE\_LPA\_VERSION ((1 << 16) | 0) // V1.0

#define HV\_STATISTICS\_TYPE\_LPV\_ID 3

#define HV\_STATISTICS\_TYPE\_LPV\_VERSION ((1 << 16) | 0) // V1.0

// Definitions for the partition counters statistics page

#define HV\_STATISTICS\_TYPE\_PA\_ID 2

#define HV\_STATISTICS\_TYPE\_PA\_VERSION ((1 << 16) | 0) // V1.0

// Definitions for the virtual processor statistics page

#define HV\_STATISTICS\_TYPE\_VPA\_ID 2

#define HV\_STATISTICS\_TYPE\_VPA\_VERSION ((1 << 16) | 0) // V1.0

#define HV\_STATISTICS\_TYPE\_VPV\_ID 3

#define HV\_STATISTICS\_TYPE\_VPV\_VERSION ((1 << 16) | 0) // V1.0

## Statistics Data Types

### Specifying Statistics Objects

When a caller wishes to access the statistics for an object, it must specify the object type and identify the object instance with the mapping request. The object type is specified using the following enumeration:

typedef enum

{

// Global stats objects

HvStatsObjectHypervisor = 0x00000001,

HvStatsObjectLogicalProcessor = 0x00000002,

// Local stats objects

HvStatsObjectPartition = 0x00010001,

HvStatsObjectVp = 0x00010002

} HV\_STATS\_OBJECT\_TYPE;

The specific statistics object instance is identified using a 16-byte data structure, defined as follows:

typedef union

{

// HvStatsObjectHypervisor

struct

{

UINT64 ReservedZ1[2];

} Hypervisor;

// HvStatsObjectLogicalProcessor

struct

{

HV\_LOGICAL\_PROCESSOR\_INDEX LogicalProcessorIndex;

UINT32 ReservedZ2;

UINT64 ReservedZ3;

} LogicalProcessor;

// HvStatsObjectPartition

struct

{

HV\_PARTITION\_ID PartitionId;

UINT64 ReservedZ4;

} Partition;

// HvStatsObjectVp

struct

{

HV\_PARTITION\_ID PartitionId;

HV\_VP\_INDEX VpIndex;

UINT32 ReservedZ5;

} Vp;

} HV\_STATS\_OBJECT\_IDENTITY;

The flags associated with requests for certain statistics objects are defined as follows:

// Flags used for specifying the stats object when making

// mapping/unmapping stats page hypercalls

typedef UINT16 HV\_STATS\_OBJECT\_FLAG;

#define HvStatsObjectSelfStats 0x0001

## Statistics Interfaces

### HvMapStatsPage

The HvMapStatsPage hypercall maps the specified stats page into the caller’s GPA space.

**Wrapper Interface**

HV\_STATUS

HvMapStatsPage(

\_\_in HV\_STATS\_OBJECT\_TYPE StatsType,

\_\_in HV\_STATS\_OBJECT\_IDENTITY ObjectIdentity,

\_\_in HV\_GPA\_PAGE\_NUMBER TargetGpaPage

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvMapStatsPage | | |
|  | Call Code = 0x006C | |
| 🡆 Input Parameters | | |
| 0 | StatsType (4 bytes) | Padding (4 bytes) |
| 8 | ObjectIdentity (16 bytes) | |
| 16 |
| 24 | TargetGpaPage (8 bytes) | |

Description

Stats pages for global statistics can be mapped only into the GPA space of the root partition.

If the specified GPA page is beyond the bounds of the partition’s valid GPA space, then the call may indicate success, but the stats page will not be accessible. In addition, the partition must have the *AccessStats* privilege.

Input Parameters

*StatsType* specifies the statistics object type.

*ObjectIdentity* specifies the statistics object’s identity by using a union. The specific fields within this union differ depending on the statistics type, as specified in the first parameter. Reserved fields within this parameter must be set to zero.

When *StatsType* is HvStatsObjectHypervisor, *ObjectIdentity* is formatted as follows:

|  |  |
| --- | --- |
| 8 | RsvdZ (16 bytes) |
| 16 |

When *StatsType* is HvStatsObjectLogicalProcessor, *ObjectIdentity* is formatted as follows:

|  |  |  |
| --- | --- | --- |
| 8 | LogicalProcessorIndex (4 bytes) | RsvdZ (4 bytes) |
| 16 | RsvdZ (8 bytes) | |

When *StatsType* is HvStatsObjectPartition, *ObjectIdentity* is formatted as follows:

|  |  |  |  |
| --- | --- | --- | --- |
| 8 | PartitionId (8 bytes) | | |
| 16 | RsvdZ (8 bytes) | StatsObjectFlag (2 bytes) | RsvdZ (2 bytes) |

When *StatsType* is HvStatsObjectVp, *ObjectIdentity* is formatted as follows:

|  |  |  |  |
| --- | --- | --- | --- |
| 8 | PartitionId (8 bytes) | | |
| 16 | VpIndex (4 bytes) | StatsObjectFlag (2 bytes) | RsvdZ (2 bytes) | |

*TargetGpaPage* specifies the location within the caller’s GPA space where the stats page should appear.

Output Parameters

None.

Restrictions

The caller must possess the *AccessStats* privilege.

For global stats types, the caller must be the root partition.

For local stats types, the caller must be the parent of the partition specified by *PartitionId*.

The partition specified by *PartitionId* must be in the “active” state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | For global stats types, the caller is not the root partition.  For local stats types, the caller is not the parent of the specified partition. |
| The caller’s partition privilege flag AccessStats is cleared. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid (used only for partition and VP stats). |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition (used only for VP stats). |
| HV\_STATUS\_OPERATION\_DENIED | The stats page has already been mapped within the caller’s GPA space. It must be unmapped before it is mapped again. |
| HV\_STATUS\_INVALID\_PARAMETER | The specified logical processor index is invalid (used only for logical processor stats). |
| Reserved fields within the *ObjectIdentity* parameter are not set to zero. |
| HV\_STATUS\_INSUFFICIENT\_MEMORY | The number of pages in the memory pool of the caller is insufficient to perform the operation. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |

### HvUnmapStatsPage

The HvUnmapStatsPage hypercall unmaps the specified stats page from the caller’s GPA space.

**Wrapper Interface**

HV\_STATUS

HvUnmapStatsPage(

\_\_in HV\_STATS\_OBJECT\_TYPE StatsType,

\_\_in HV\_STATS\_OBJECT\_IDENTITY ObjectIdentity

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvUnmapStatsPage | | |
|  | Call Code = 0x006D | |
| 🡆 Input Parameters | | |
| 0 | StatsType (4 bytes) | Padding (4 bytes) |
| 8 | ObjectIdentity (16 bytes) | |
| 16 |

Description

Only the root partition may unmap global stats pages. Local stats pages may only be unmapped by the partition or its parent. In addition, the partition must have the *AccessStats* privilege.

Input Parameters

*StatsType* specifies the statistics object type.

*ObjectIdentity* specifies the statistics object’s identity by using a union. The specific fields within this union differ depending on the statistics type, as specified in the first parameter. Reserved fields within this parameter must be set to zero.

When *StatsType* is HvStatsObjectHypervisor, *ObjectIdentity* is formatted as follows:

|  |  |
| --- | --- |
| 8 | RsvdZ (16 bytes) |
| 16 |

When *StatsType* is HvStatsObjectLogicalProcessor, *ObjectIdentity* is formatted as follows:

|  |  |  |
| --- | --- | --- |
| 8 | LogicalProcessorIndex (4 bytes) | RsvdZ (4 bytes) |
| 16 | RsvdZ (8 bytes) | |

When *StatsType* is HvStatsObjectPartition, *ObjectIdentity* is formatted as follows:

|  |  |  |  |
| --- | --- | --- | --- |
| 8 | PartitionId (8 bytes) | | |
| 16 | RsvdZ (8 bytes) | StatsObjectFlag (2 bytes( | RsvdZ (2 bytes) |

When *StatsType* is HvStatsObjectVp, *ObjectIdentity* is formatted as follows:

|  |  |  |  |
| --- | --- | --- | --- |
| 8 | PartitionId (8 bytes) | | |
| 16 | VpIndex (4 bytes) | StatsObjectFlag (2 bytes) | RsvdZ (2 bytes) | |

Output Parameters

None.

Restrictions

The caller must possess the *AccessStats* privilege.

For global stats types, the caller must be the root partition.

For local stats types, the caller must be the parent of the partition specified by *PartitionId.* The partition specified by *PartitionId* must be in the “active” state.

Return Values

| Status code | Error condition |
| --- | --- |
| HV\_STATUS\_ACCESS\_DENIED | For global stats types, the caller is not the root partition.  For local stats types, the caller is not the parent of the specified partition. |
| The caller’s partition privilege flag AccessStats is cleared. |
| HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid (used only for partition and VP stats). |
| HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index does not reference a virtual processor within the specified partition (used only for VP stats). |
| HV\_STATUS\_INVALID\_PARAMETER | The specified logical processor index is invalid (used only for logical processor stats). |
| Reserved fields within the *ObjectIdentity* parameter are not set to zero. |
| HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition is not in the “active” state. |
| HV\_STATUS\_OPERATION\_DENIED | The specified page is not initialized or mapped. |

### Statistics Page Mapping MSRs

A guest wishing to access its own local statistics pages must use the following model-specific registers (MSRs).

| MSR address | Register name | Function | |
| --- | --- | --- | --- |
| 0x400000E0 | HV\_X64\_MSR\_STATS\_PARTITION\_RETAIL\_PAGE | | Map the guest’s local partition-related retail statistics page |
| 0x400000E1 | HV\_X64\_MSR\_STATS\_PARTITION\_INTERNAL\_PAGE | | Map the guest’s local partition-related internal statistics page |
| 0x400000E2 | HV\_X64\_MSR\_STATS\_VP\_RETAIL\_PAGE | | Map the guest’s local virtual processor-related retail statistics page |
| 0x400000E3 | HV\_X64\_MSR\_STATS\_VP\_INTERNAL\_PAGE | | Map the guest’s local virtual processor-related internal statistics page |

Each of the above MSRs has the following format:

| 63:12 | 11:1 | 0 |
| --- | --- | --- |
| Statistics Page Base Address | RsvdP | Enable |

| Bits | Description | Attributes |
| --- | --- | --- |
| 63:12 | Base address (in GPA space) of statistics page to be mapped  (low 12 bits assumed to be zero) | Read/write |
| 11:1 | RsvdP (value should be preserved) | Read/write |
| 0 | Statistics page enable | Read/write |

At virtual processor creation time and upon processor reset, the value of each of the statistics page registers is 0x0000000000000000. Thus, the pages are disabled by default. The guest must enable the chosen page by setting bit 0. If the specified base address is beyond the end of the partition’s GPA space, the statistics page will not be accessible to the guest. When modifying the registers, guests should preserve the value of the reserved bits (1 through 11) for future compatibility.

Restrictions

The caller must possess the *AccessStatsMsr* privilege.

# Booting

## Overview

The mechanism by which the hypervisor is loaded and launched is not architecturally defined. Typically, it will involve a *boot agent* running within ROM-based or flash-based firmware, an on-disk boot loader, or a device driver. This agent is responsible for loading the hypervisor code image, laying it out in memory, gathering various boot-time parameters, and invoking the hypervisor’s entry point in the environment specified by the hypervisor image. Some boot agent implementations may leverage specialized security hardware to perform a *measured* or *verified launch*.

## Pre-boot Requirements

The hypervisor or its boot agent must validate that the following requirements are true before or during a successful hypervisor launch:

* All physical processors support the minimum feature set required by the hypervisor.
* All physical processors have the same number of cores and hyperthreads per core.
* The physical processors support virtualization extensions (for example, Intels® VT or AMD–V™).
* The physical processors are running in monitor mode; that is, the hypervisor is not attempting to be launched on top of an existing hypervisor or VMM.

For a detailed list of the hypervisor pre-boot requirements, see .

## Post-boot Conditions

Immediately after a successful hypervisor launch, control is returned to the boot agent or another component of guest code designated by the boot agent. In any case, this code will be running in a newly-created partition referred to as the *root partition.* At this time, the following conditions are guaranteed to exist:

* The hypervisor knows about all present and potential physical nodes. Each of these nodes has been assigned a unique index.
* The hypervisor knows about the memory access ratios between physical nodes.
* An SPA range has been created for every populated RAM range in the machine.
* All RAM SPA pages are mapped into the root partition’s GPA space with an identity map. All pages that are in use by the hypervisor are mapped with no access rights. All other pages are mapped as readable, writable, and executable.
* The hypervisor knows about all existing and potential logical processors in the system. Each has been given a unique logical processor index. (*Potential* here refers to logical processors that can be hot-plugged at some point in the future.)
* A single partition exists (the root partition). It is in the active state.
* The root partition’s memory pool may or may not be populated with some memory. Code within the root partition can use the hypercall HvGetMemoryBalance to determine how many pages are in the pool.
* One virtual processor is created in the root partition for each existing and potential logical processor in the system.
* None of the root partition’s virtual processors are suspended.
* All virtual processors corresponding to *potential* logical processors are assumed to be in the “offline” power state. All other virtual processors are assumed to be in the “C0” power state.
* After a successful hypervisor boot, the root partition’s virtual processors are left in the same initialization state as their corresponding logical processors before the boot. For example, if a processor was already initialized and executing code, it will continue to execute code after hypervisor boot. If a processor was in the “wait for SIPI” state, it will effectively remain in this state after hypervisor boot.
* The initial register state of a root partition’s virtual processor is not architecturally specified. Typically, it will be derived from the register state of the boot agent before the hypervisor was launched.
* The root partition runs with default scheduling policy parameters (no reserve, an infinite cap, and a weight of 100).
* The hypercall MSRs are visible to the root partition, but the hypercall interface has not yet established within the root partition.
* The SynIC is not yet enabled within the root partition. No synthetic timers are enabled.
* The local APIC is enabled and located in the root partition’s GPA space according to the IA32\_APIC\_BASE MSR of the first logical processor (the boot processor).
* If available (see section ), the APIC MSRs are visible within the root partition.

## Root Partition

The root partition is special in a number of ways. It is “born” with certain properties, capabilities, and privileges that non-root partitions do not possess. These include the following:

* The root partition has no parent partition. Any operations that can be performed only by a parent partition on its children cannot be performed on the root. This includes the following operations:
* Changes to partition state (HvInitializePartition, HvFinalizePartition and HvDeletePartition)
* Modification of partition properties (HvGetPartitionProperty and HvSetPartitionProperty)
* Changes to GPA mappings (HvMapGpaPages may only change access rights to an existing mapping; HvUnmapGpaPages is not allowed)
* Access to VA or GPA space (HvTranslateVirtualAddress, HvReadGpa, or HvWriteGpa)
* Installation of intercepts (HvInstallIntercept)
* Virtual processor creation and deletion (HvCreateVp and HvDeleteVp)
* Access to virtual processor register state (HvGetVpRegisters and HvSetVpRegisters)
* Virtualization of legacy interrupts (HvAssertVirtualInterrupt and HvClearVirtualInterrupt)
* Saving and restoring partition state (HvSavePartitionState and HvRestorePartitionState)
* Management of logical processor states (HvGetLogicalProcessorRunTime) and HvCallParkLogicalProcessors.
* The root partition, by default, can access all physical I/O ports. Other partitions cannot. Note that some hypervisor implementations may restrict access to some I/O ports for security reasons.
* The root partition can access non-virtualized MSRs that directly control the behavior of the hardware. Note that some hypervisor implementations may restrict access to some MSRs for security reasons.
* The virtual processors within the root partition have hard affinities; that is, each virtual processor is strongly bound to a corresponding logical processor.
* By default, all mapped GPAs within the root partition’s GPA space are mapped to the same addresses within the SPA space. This is not necessarily true for overlay pages.
* All hardware interrupts are directed at virtual processors within the root partition. (Future versions of the hypervisor will remove this assumption.)

# 

# System Properties

## Overview

System properties provide a generic way for the root partition to control aspects of the entire hypervisor. After a system property is set, its value is constant unless and until it is again modified by the root partition. System properties are identified by a 32-bit code. System property values are each 128 bits in size.

## System Property Data Types

typedef enum

{

HvSystemProperyPerfCounterConfiguration,

HvSystemPropertyMax

} HV\_SYSTEM\_PROPERTY\_CODE;

typedef union

{

UINT64 GenericProperty[2];

HV\_PERF\_COUNTER\_CONFIGURATION PerfCounter;

} HV\_SYSTEM\_PROPERTY;

## Performance Counter Configuration

The performance counter configuration property specifies the performance counter and the sampling period to use during profiling. Setting this property does not start or stop the profiler – instead the profiler is started and stopped using event log hypercalls. See section 19. Only one event can be set for a given profile run. If the profiler is running when a call is made to set the property, it will fail with an HV\_STATUS\_OPERATION\_DENIED status.

//

// Profile sources that can be configured.

//

typedef enum

{

HvProfileInvalid,

HvProfileCyclesNotHalted,

HvProfileCacheMisses,

HvProfileBranchMispredictions

} HV\_PROFILE\_SOURCE;

typedef struct

{

HV\_PROFILE\_SOURCE ProfileSource;

UINT32 Reserved;

UINT64 Period;

} HV\_PERF\_COUNTER\_CONFIGURATION;

## System Property Interfaces

### HvSetSystemProperty

The HvSetSystemPropery hypercall configures settings that affect the entire system or hypervisor. Such settings are not specific to a particular partition.

**Wrapper Interface**

HV\_STATUS

HvSetSystemProperty(

\_\_in HV\_SYSTEM\_PROPERTY\_CODE PropertyCode,

\_\_in HV\_SYSTEM\_PROPERTY PropertyValue

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvSetSystemProperty | | |
|  | Call Code = 0x006F | |
| 🡆 Input Parameters | | |
| 0 | PropertyCode (4 bytes) | Padding (4 bytes) |
| 8 | PropertyValue (16 bytes) | |

Description

This call provides a generic mechanism to set system properties.

Input Parameters

*PropertyCode* specifies the property the caller is interested in modifying.

*PropertyValue* specifies the new property value.

Output Parameters

None.

Restrictions

The caller must be the root partition.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_UNKNOWN\_PROPERTY | The specified PropertyCode is not a recognized property. |
| HV\_STATUS\_INVALID\_PARAMETER | An attempt was made to modify a read-only partition property or the specified value is invalid for the specified PropertyCode. |
| HV\_STATUS\_OPERATION\_DENIED | The operation is not allowed in the current state. |

### HvGetSystemProperty

The HvGetSystemPropery hypercall returns settings that affect the entire system or hypervisor. Such settings are not specific to a particular partition.

**Wrapper Interface**

HV\_STATUS

HvGetSystemProperty(

\_\_in HV\_SYSTEM\_PROPERTY\_CODE PropertyCode,

\_\_out PHV\_SYSTEM\_PROPERTY PropertyValue

);

Native Interface

|  |  |  |
| --- | --- | --- |
| HvGetSystemProperty | | |
|  | Call Code = 0x0071 | |
| 🡆 Input Parameters | | |
| 0 | PropertyCode (4 bytes) | Padding (4 bytes) |
| 🡄 Output Parameters | | |
| 0 | PropertyValue (16 bytes) | |

Description

This call provides a generic mechanism to get system properties.

Input Parameters

*PropertyCode* specifies the property the caller is interested in querying.

Output Parameters

*PropertyValue* specifies the property’s current value.

Restrictions

The caller must be the root partition.

Return Values

|  |  |
| --- | --- |
| Status code | Error condition |
| HV\_STATUS\_ACCESS\_DENIED | The caller is not the root partition. |
| HV\_STATUS\_UNKNOWN\_PROPERTY | The specified PropertyCode is not a recognized property. |

# Appendix : Interface Guidelines

In the interest of providing a consistent interface for the hypervisor, the following guidelines should be honored for all interfaces:

Names of calls, types, structures, and constants should be clearly marked as being associated with the hypervisor. In general, this is done by prepending “Hv” or “HV”.

Names of calls, types, structures, and constants should be clear and not overly generic (for example, “HvSetFlag” is probably too generic).

All hypervisor types (including enumerations) must have explicitly-defined sizes and values.

Names of types and constants that are platform-specific should include such an indication (for example, by prepending with “HV\_X64”).

In general, abbreviations should be avoided. However, if abbreviations are used, they should be used consistently throughout the interface. The following is a list of approved abbreviations:

CPU (central processing unit)

ID (identifier)

INFO (information)

FP (floating point)

GPA (guest physical address)

HV (hypervisor)

MAX (maximum)

MIN (minimum)

MSR (model-specific register)

RAM (random access memory)

SINT (synthetic interrupt source)

SPA (system physical address)

STATS (statistics)

SYNIC (synthetic interrupt controller)

VP (virtual processor)

In most cases, partition IDs and VP indices should not be implicit; that is, the interface should allow the caller to specify these values rather than assuming the caller wants to operate on the current partition or VP. There are several specific cases where this guideline is ignored.

No hypercalls should include a GPA address input parameter that points to additional input or output parameters. In other words, all parameters should be passed by value, not by reference.

In most cases, page numbers are preferable to addresses. This is especially true when the interface in question operates on memory with page-level granularity.

The verb “Set” should be used only for calls that set a specific value. Generally, such calls should have an accompanying “Get” call.

The name of rep hypercalls should indicate that they operate on multiple parameters (for example, HvSetVpRegisters instead of HvSetVpRegister).

Rep hypercalls should be used only in cases where there is a clear performance need. In other cases, simple hypercalls are preferred.

# Appendix : Hypercall Code Reference

The following is a table of all hypercalls by call code.

| Call Code | Rep Call | Fast  Call | Hypercall | Caller | Partition Privilege Required (if any) |
| --- | --- | --- | --- | --- | --- |
| 0x0001 |  | ✔ | HvSwitchVirtualAddressSpace | Any | UseHypercallForAddressSpaceSwitch |
| 0x0002 |  |  | HvFlushVirtualAddressSpace | Any | UseHypercallFor[Local][Remote]Flush |
| 0x0003 | ✔ |  | HvFlushVirtualAddressList | Any | UseHypercallFor[Local][Remote]Flush |
| 0x0004 |  |  | HvGetLogicalProcessorRunTime | Any | CpuPowerManagement |
|  |  |  |  |  |  |
|  |  |  |  |  |  |
| 0x0005 thru 0x0007 |  |  | Reserved for future use | -- |  |
| 0x0008 |  | ✔ | HvNotifyLongSpinWait | Any | UseHypercallForLongSpinWait |
| 0x0009 |  |  | HvParkLogicalProcessors | Any | CpuPowerManagement |
| 0x000A  through  0x003F |  |  | Reserved for future use | -- |  |
| 0x0040 |  |  | HvCreatePartition | Any | CreatePartitions |
| 0x0041 |  | ✔ | HvInitializePartition | Parent |  |
| 0x0042 |  | ✔ | HvFinalizePartition | Parent |  |
| 0x0043 |  | ✔ | HvDeletePartition | Parent |  |
| 0x0044 |  |  | HvGetPartitionProperty | Parent / Root |  |
| 0x0045 |  |  | HvSetPartitionProperty | Parent / Root |  |
| 0x0046 |  |  | HvGetPartitionId | Any | AccessPartitionId |
| 0x0047 |  |  | HvGetNextChildPartition | Parent |  |
| 0x0048 | ✔ |  | HvDepositMemory | Parent / Self | AccessMemoryPool |
| 0x0049 | ✔ |  | HvWithdrawMemory | Parent / Self | AccessMemoryPool |
| 0x004A |  |  | HvGetMemoryBalance | Parent / Self | AccessMemoryPool |
| 0x004B | ✔ |  | HvMapGpaPages | Parent / Root |  |
| 0x004C | ✔ |  | HvUnmapGpaPages | Parent |  |
| 0x004D |  |  | HvInstallIntercept | Parent |  |
| 0x004E |  |  | HvCreateVp | Parent |  |
| 0x004F |  | ✔ | HvDeleteVp | Parent |  |
| 0x0050 | ✔ |  | HvGetVpRegisters | Parent / Self |  |
| 0x0051 | ✔ |  | HvSetVpRegisters | Parent |  |
| 0x0052 |  |  | HvTranslateVirtualAddress | Parent |  |
| 0x0053 |  |  | HvReadGpa | Parent |  |
| 0x0054 |  |  | HvWriteGpa | Parent |  |
| 0x0055 |  |  | HvAssertVirtualInterrupt | Parent |  |
| 0x0056 |  | ✔ | HvClearVirtualInterrupt | Parent |  |
| 0x0057 |  |  | HvCreatePort | Parent / Self | CreatePort |
| 0x0058 |  | ✔ | HvDeletePort | Parent / Self |  |
| 0x0059 |  |  | HvConnectPort | Parent / Self | ConnectPort |
| 0x005A |  |  | HvGetPortProperty | Parent / Self |  |
| 0x005B |  | ✔ | HvDisconnectPort | Parent / Self |  |
| 0x005C |  |  | HvPostMessage | Any | PostMessages |
| 0x005D |  | ✔ | HvSignalEvent | Any | SignalEvents |
| 0x005E |  |  | HvSavePartitionState | Parent |  |
| 0x005F |  |  | HvRestorePartitionState | Parent | CreatePartitions |
| 0x0060 |  |  | HvInitializeEventLogBufferGroup | Root |  |
| 0x0061 |  | ✔ | HvFinalizeEventLogBufferGroup | Root |  |
| 0x0062 |  | ✔ | HvCreateEventLogBuffer | Root |  |
| 0x0063 |  | ✔ | HvDeleteEventLogBuffer | Root |  |
| 0x0064 |  |  | HvMapEventLogBuffer | Root |  |
| 0x0065 |  | ✔ | HvUnmapEventLogBuffer | Root |  |
| 0x0066 |  | ✔ | HvSetEventLogGroupSources | Root |  |
| 0x0067 |  | ✔ | HvReleaseEventLogBuffer | Root |  |
| 0x0068 |  | ✔ | HvFlushEventLogBuffer | Root |  |
| 0x0069 |  |  | HvPostDebugData | Any | Debugging |
| 0x006A |  |  | HvRetrieveDebugData | Any | Debugging |
| 0x006B |  | ✔ | HvResetDebugSession | Any | Debugging |
| 0x006C |  |  | HvMapStatsPage | Parent[[4]](#footnote-4) | AccessStats |
| 0x006D |  |  | HvUnmapStatsPage | Parent | AccessStats |
| 0x006E | ✔ |  | HvCallMapSparseGpaPages | Parent / Root |  |
| 0x006F |  |  | HvCallSetSystemProperty | Root | ConfigureProfiler |
| 0x0070 |  |  | HvCallSetPortProperty | Parent / Self | CreatePort |

# 

# Appendix : Hypercall Status Code Reference

The following is a table of all hypercall return codes.

| Status Code | Status Name | Meaning |
| --- | --- | --- |
| 0x0000 | HV\_STATUS\_SUCCESS | The operation succeeded. |
| 0x0001 |  | Reserved.[[5]](#endnote-1) |
| 0x0002 | HV\_STATUS\_INVALID\_HYPERCALL\_CODE | The hypercall code was not recognized. |
| 0x0003 | HV\_STATUS\_INVALID\_HYPERCALL\_INPUT | The rep count was incorrect (for example, a non-zero rep count was passed to a non-rep call or a zero rep count was passed to a rep call) or a reserved bit in the specified hypercall input value was non-zero. |
| 0x0004 | HV\_STATUS\_INVALID\_ALIGNMENT | The specified input and/or output GPA pointers were not aligned to 8 bytes or the specified input and/or output parameters lists spanned a page boundary. |
| 0x0005 | HV\_STATUS\_INVALID\_PARAMETER | One or more input parameters were invalid. |
| 0x0006 | HV\_STATUS\_ACCESS\_DENIED | The caller did not possess sufficient access rights to perform the requested operation. |
| 0x0007 | HV\_STATUS\_INVALID\_PARTITION\_STATE | The specified partition’s state was not appropriate for the requested operation. |
| 0x0008 | HV\_STATUS\_OPERATION\_DENIED | The operation could not be performed. (The actual cause depends on the operation.) |
| 0x0009 | HV\_STATUS\_UNKNOWN\_PROPERTY | The specified partition property ID is not a recognized property. |
| 0x000A | HV\_STATUS\_PROPERTY\_VALUE\_OUT\_OF\_RANGE | The specified value of a partition property is out of range or violates an invariant. |
| 0x000B | HV\_STATUS\_INSUFFICIENT\_MEMORY | Insufficient memory exists for the call to succeed. |
| 0x000C | HV\_STATUS\_PARTITION\_TOO\_DEEP | The maximum partition depth has been exceeded for the partition hierarchy. |
| 0x000D | HV\_STATUS\_INVALID\_PARTITION\_ID | The specified partition ID is invalid. |
| 0x000E | HV\_STATUS\_INVALID\_VP\_INDEX | The specified VP index is invalid. |
| 0x000F |  | Reserved |
| 0x0010 |  | Reserved |
| 0x0011 | HV\_STATUS\_INVALID\_PORT\_ID | The specified port ID is not unique or does not exist. |
| 0x0012 | HV\_STATUS\_INVALID\_CONNECTION\_ID | The specified connection ID is not unique or does not exist. |
| 0x0013 | HV\_STATUS\_INSUFFICIENT\_BUFFERS | The target port does not have sufficient buffers for the caller to post a message. |
| 0x0014 | HV\_STATUS\_NOT\_ACKNOWLEDGED | An external interrupt has not previously been asserted and acknowledged by the virtual processor prior to clearing it. |
| 0x0015 |  | Reserved |
| 0x0016 | HV\_STATUS\_ACKNOWLEDGED | An external interrupt cannot be asserted because a previously-asserted external interrupt was acknowledged by the virtual processor and has not yet been cleared. |
| 0x0017 | HV\_STATUS\_INVALID\_SAVE\_RESTORE\_STATE | The initial call to HvSavePartitionState or HvRestorePartitionState specifying HV\_SAVE\_RESTORE\_STATE\_START was not made at the beginning of the save/restore process. |
| 0x0018 | HV\_STATUS\_INVALID\_SYNIC\_STATE | The operation could not be performed because a required feature of the SynIC was disabled. |
| 0x0019 | HV\_STATUS\_OBJECT\_IN\_USE | The operation could not be performed because the object or value was either already in use or being used for a purpose that would not permit it. |
| 0x001A | HV\_STATUS\_INVALID\_PROXIMITY\_DOMAIN\_INFO | The *Flags* field included an invalid mask value in the proximity domain information. |
| The *Id* field contained an invalid ACPI node ID in the proximity domain information. |
| 0x001B | HV\_STATUS\_NO\_DATA | An attempt to retrieve data failed because none was available. |
| 0x001C | HV\_STATUS\_INACTIVE | The physical connection being used for debugging has not recorded any receive activity since the last operation. |
| 0x001D | HV\_STATUS\_NO\_RESOURCES | A resource is unavailable for allocation. This may indicate that there is a resource shortage or that an implementation limitation may have been reached. |
| 0x001E | HV\_STATUS\_FEATURE\_UNAVAILABLE | A hypervisor feature is not available to the caller. |
| 0x001F | HV\_STATUS\_PARTIAL\_PACKET | The debug packet returned is only a partial packet due to an I/O error. |
| 0x0020 | HV\_STATUS\_PROCESSOR\_FEATURE\_SSE3\_NOT\_SUPPORTED | The processor feature SSE3 is not supported. |
| 0x0021 | HV\_STATUS\_PROCESSOR\_FEATURE\_LAHFSAHF\_NOT\_SUPPORTED | The processor feature LAHFSAHF is not supported. |
| 0x0022 | HV\_STATUS\_PROCESSOR\_FEATURE\_SSSE3\_NOT\_SUPPORTED | The processor feature SSSE3 is not supported. |
| 0x0023 | HV\_STATUS\_PROCESSOR\_FEATURE\_SSE4\_1\_NOT\_SUPPORTED | The processor feature SSE4.1 is not supported. |
| 0x0024 | HV\_STATUS\_PROCESSOR\_FEATURE\_SSE4\_2\_NOT\_SUPPORTED | The processor feature SSE4.2 is not supported. |
| 0x0025 | HV\_STATUS\_PROCESSOR\_FEATURE\_SSE4A\_NOT\_SUPPORTED | The processor feature SSE4a is not supported. |
| 0x0026 | HV\_STATUS\_PROCESSOR\_FEATURE\_SSE5\_NOT\_SUPPORTED | The processor feature SSE5 is not supported. |
| 0x0027 | HV\_STATUS\_PROCESSOR\_FEATURE\_POPCNT\_NOT\_SUPPORTED | The processor feature POPCNT is not supported. |
| 0x0028 | HV\_STATUS\_PROCESSOR\_FEATURE\_CMPXCHG16B\_NOT\_SUPPORTED | The processor feature CMPXCHG16B is not supported. |
| 0x0029 | HV\_STATUS\_PROCESSOR\_FEATURE\_ALTMOVCR8\_NOT\_SUPPORTED | The processor feature ALTMOVCR8 is not supported. |
| 0x002A | HV\_STATUS\_PROCESSOR\_FEATURE\_LZCNT\_NOT\_SUPPORTED | The processor feature LZCNT is not supported. |
| 0x002B | HV\_STATUS\_PROCESSOR\_FEATURE\_MISALIGNED\_SSE\_NOT\_SUPPORTED | The processor feature MISALIGNED SSE3 is not supported. |
| 0x002C | HV\_STATUS\_PROCESSOR\_FEATURE\_MMX\_EXT\_NOT\_SUPPORTED | The processor feature MMX EXT is not supported. |
| 0x002D | HV\_STATUS\_PROCESSOR\_FEATURE\_3DNOW\_NOT\_SUPPORTED | The processor feature 3DNow is not supported. |
| 0x002E | HV\_STATUS\_PROCESSOR\_FEATURE\_EXTENDED\_3DNOW\_NOT\_SUPPORTED | The processor feature Extended 3DNow is not supported. |
| 0x002F | HV\_STATUS\_PROCESSOR\_FEATURE\_PAGE\_1GB\_NOT\_SUPPORTED | The processor feature PAHGE 1GB is not supported. |
| 0x0030 | HV\_STATUS\_PROCESSOR\_CACHE\_LINE\_FLUSH\_SIZE\_INCOMPATIBLE | The processor’s cache line flush size is not supported. |
| 0x0031 | HV\_STATUS\_PROCESSOR\_FEATURE\_XSAVE\_NOT\_SUPPORTED | The processor feature XSAVE is not supported. |
| 0x0032 | HV\_STATUS\_PROCESSOR\_FEATURE\_XSAVEOPT\_NOT\_SUPPORTED | The processor feature XSAVEOPT is not supported. |
| 0x0033 | HV\_STATUS\_PROCESSOR\_FEATURE\_XSAVE\_LEGACY\_SSE\_NOT\_SUPPORTED | The processor feature XSAVE Legacy SSE is not supported. |
| 0x0034 | HV\_STATUS\_PROCESSOR\_FEATURE\_XSAVE\_AVX\_NOT\_SUPPORTED | The processor feature XSAVE AVX is not supported. |
| 0x0035 | HV\_STATUS\_PROCESSOR\_FEATURE\_XSAVE\_UNKNOWN\_FEATURE\_NOT\_SUPPORTED | The processor unknown XSAVE feature is not supported. |
| 0x0036 | HV\_STATUS\_PROCESSOR\_XSAVE\_SAVE\_AREA\_INCOMPATIBLE | The processor’s XSAVE area is not supported. |
| 0x0037 | HV\_STATUS\_INCOMPATIBLE\_PROCESSOR | The processor architecture is not supported. |

# Appendix : Architectural CPUID

The table below contains a list of architected CPUID leaves and how they are virtualized by the hypervisor. *Passthrough* means that the hardware value is used and passed through to the guest. That value will be the same on all logical processors in the system.

| Index and Register | Name | Start bit | End bit | Virtualized Value  Root | Virtualized Value  Non-Root |
| --- | --- | --- | --- | --- | --- |
| 0x00000000 |  |  |  |  |  |
| EAX | Maximum valid standard CPUID index | 0 | 31 | Minimum of the hardware value on the current logical processor and 0x00000006 | Minimum of the hardware value across all logical processors and 0x00000006 |
| EBX | Processor vendor string | 0 | 31 | Passthrough | Passthrough |
| ECX | Processor vendor string | 0 | 31 | Passthrough | Passthrough |
| EDX | Processor vendor string | 0 | 31 | Passthrough | Passthrough |
| 0x00000001 |  |  |  |  |  |
| EAX | Stepping | 0 | 3 | Passthrough | Passthrough minimum stepping across all logical processors |
|  | Base Model | 4 | 7 | Passthrough | Passthrough |
|  | Base Family | 8 | 11 | Passthrough | Passthrough |
|  | Processor Type | 12 | 13 | Passthrough | Passthrough |
|  | RsvdZ | 14 | 15 | Cleared | Cleared |
|  | Extended Model | 16 | 19 | Passthrough | Passthrough |
|  | Extended Family | 20 | 27 | Passthrough | Passthrough |
|  | RsvdZ | 28 | 31 | Cleared | Cleared |
| EBX | Miscellaneous Information |  |  |  |  |
|  | Brand Identifier | 0 | 7 | Passthrough | Passthrough value received from processor 0 |
|  | CL Flush size | 8 | 15 | Passthrough | Passthrough |
|  | Maximum LPs in a physical package | 16 | 23 | Passthrough | If HT is disabled, set to number of cores. If HT enabled, passthrough. |
|  | Initial APIC ID | 24 | 31 | Passthrough | Return value of the HvX64RegisterInitialApicId |
| ECX | Feature Flags / Identifiers |  |  |  |  |
|  | SSE3 | 0 | 0 | Passthrough | Set if set on all, otherwise 0 |
|  | RsvdZ | 1 | 2 | Cleared | Cleared |
|  | MONITOR | 3 | 3 | Cleared | Cleared |
|  | DS-CPL | 4 | 4 | Cleared | Cleared |
|  | VMX | 5 | 5 | Cleared | Cleared |
|  | RsvdZ | 6 | 6 | Cleared | Cleared |
|  | EST | 7 | 7 | Passthrough | Cleared |
|  | TM2 | 8 | 8 | Passthrough | Cleared |
|  | SSSE3 | 9 | 9 | Passthrough | Set if set on all, otherwise 0 |
|  | CNXTID | 10 | 10 | Passthrough | Cleared |
|  | RsvdZ | 11 | 12 | Cleared | Cleared |
|  | CMPXCHG16B | 13 | 13 | Passthrough | Set if set on all, otherwise 0 |
|  | xTPR | 14 | 14 | Passthrough | Cleared |
|  | PDCM | 15 | 15 | Cleared | Cleared |
|  | RsvdZ | 16 | 17 | Cleared | Cleared |
|  | DCA | 18 | 18 | Cleared | Cleared |
|  | SSE4.1 | 19 | 19 | Passthrough | Set if set on all, otherwise 0 |
|  | SSE4.2 | 20 | 20 | Passthrough | Set if set on all, otherwise 0 |
|  | RsvdZ | 21 | 22 | Cleared | Cleared |
|  | Population Count | 23 | 23 | Passthrough | Set if set on all, otherwise 0 |
|  | RsvdZ | 24 | 30 | Cleared | Cleared |
|  | Hypervisor Present | 31 | 31 | Set | Set |
| EDX | Feature Information |  |  |  |  |
|  | FPU | 0 | 0 | Set | Set |
|  | VME | 1 | 1 | Set | Set |
|  | DE | 2 | 2 | Set | Set |
|  | PSE | 3 | 3 | Set | Set |
|  | TSC | 4 | 4 | Set | Set |
|  | MSR | 5 | 5 | Set | Set |
|  | PAE | 6 | 6 | Set | Set |
|  | MCE | 7 | 7 | Set | Set |
|  | CMPXCHG8B | 8 | 8 | Set | Set |
|  | APIC | 9 | 9 | Set | Set |
|  | RsvdZ | 10 | 10 | Cleared | Cleared |
|  | SEP | 11 | 11 | Set | Set |
|  | MTRR | 12 | 12 | Set | Set |
|  | PGE | 13 | 13 | Set | Set |
|  | MCA | 14 | 14 | Set | Set |
|  | CMOV | 15 | 15 | Set | Set |
|  | PAT | 16 | 16 | Set | Set |
|  | PSE-36 | 17 | 17 | Set | Set |
|  | PSN | 18 | 18 | Cleared | Cleared |
|  | CLFSH | 19 | 19 | Set | Set |
|  | RsvdZ | 20 | 20 | Cleared | Cleared |
|  | DS | 21 | 21 | Cleared | Cleared |
|  | ACPI | 22 | 22 | Passthrough | Cleared |
|  | MMX | 23 | 23 | Set | Set |
|  | FXSR | 24 | 24 | Set | Set |
|  | SSE | 25 | 25 | Set | Set |
|  | SSE2 | 26 | 26 | Set | Set |
|  | SS | 27 | 27 | Passthrough | Set if set on all, otherwise 0 |
|  | HTT | 28 | 28 | Passthrough | Cleared |
|  | TM | 29 | 29 | Passthrough | Cleared |
|  | RsvdZ | 30 | 30 | Cleared | Cleared |
|  | PBE | 31 | 31 | Passthrough | Cleared |
| 0x00000002 |  |  |  |  |  |
| EAX | Cache and TLB descriptors | 0 | 31 | Passthrough on Intel  Cleared on AMD | Passthrough value from processor 0 on Intel  Cleared on AMD |
| EBX | Cache and TLB descriptors | 0 | 31 | Passthrough on Intel  Cleared on AMD | Passthrough value from processor 0 on Intel  Cleared on AMD |
| ECX | Cache and TLB descriptors | 0 | 31 | Passthrough on Intel  Cleared on AMD | Passthrough value from processor 0 on Intel  Cleared on AMD |
| EDX | Cache and TLB descriptors | 0 | 31 | Passthrough on Intel  Cleared on AMD | Passthrough value from processor 0 on Intel  Cleared on AMD |
| 0x00000003 |  |  |  |  |  |
| EAX | Processor serial number | 0 | 31 | Cleared | Cleared |
| EBX | Processor serial number | 0 | 31 | Cleared | Cleared |
| ECX | Processor serial number | 0 | 31 | Cleared | Cleared |
| EDX | Processor serial number | 0 | 31 | Cleared | Cleared |
| 0x00000004 |  |  |  |  |  |
| EAX | Cache type | 0 | 4 | Passthrough | Passthrough value from processor 0 |
|  | Cache level | 5 | 7 | Passthrough | Passthrough value from processor 0 |
|  | Self-initializing cache level | 8 | 8 | Passthrough | Passthrough value from processor 0 |
|  | Fully Associative | 9 | 9 | Passthrough | Passthrough value from processor 0 |
|  | Write back invalidate / invalidate | 10 | 10 | Passthrough | Passthrough value from processor 0 |
|  | Cache inclusiveness | 11 | 11 | Passthrough | Passthrough value from processor 0 |
|  | RsvdZ | 12 | 13 | Cleared | Cleared |
|  | Maximum threads in cache | 14 | 25 | Passthrough | If HT disabled, scale the value down by the number of threads / core. If HT is enabled, set to the number of logical processors per package. |
|  | Max cores per package | 26 | 31 | Passthrough | If HT enabled, scale up this value by number of threads per core. If HT is disabled, passthrough the value from processor 0. |
| EBX | Coherency line size | 0 | 11 | Passthrough | Passthrough value from processor 0 |
|  | Physical line partitions | 12 | 21 | Passthrough | Passthrough value from processor 0 |
|  | Ways of associativity | 22 | 31 | Passthrough | Passthrough value from processor 0 |
| ECX | Number of sets | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x00000005 | MONITOR / MWAIT |  |  |  |  |
| EAX |  | 0 | 15 | Cleared | Cleared |
|  | RsvdZ | 16 | 31 | Cleared | Cleared |
| EBX | MonLineSizeMax | 0 | 15 | Cleared | Cleared |
|  | RsvdZ | 16 | 31 | Cleared | Cleared |
| ECX | EMX | 0 | 0 | Cleared | Cleared |
|  | IBE | 1 | 1 | Cleared | Cleared |
|  | RsvdZ | 2 | 31 | Cleared | Cleared |
| EDX | C0SubstatesUsingMwait | 0 | 3 | Cleared | Cleared |
|  | C1SubstatesUsingMwait | 4 | 7 | Cleared | Cleared |
|  | C2SubstatesUsingMwait | 8 | 11 | Cleared | Cleared |
|  | C3SubstatesUsingMwait | 12 | 15 | Cleared | Cleared |
|  | C4SubstatesUsingMwait | 16 | 19 | Cleared | Cleared |
|  | RsvdZ | 20 | 31 | Cleared | Cleared |
| 0x00000006 | Power management feature enumeration function |  |  |  |  |
| EAX | Digital Temperature Sensor Supported. | 0 | 0 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
|  | Dynamic Acceleration Enabled | 1 | 1 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
|  | Constant Rate Timer | 2 | 2 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
|  | RsvdZ | 3 | 31 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
| EBX | Thermal Threshold Count | 0 | 3 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
|  | RsvdZ | 4 | 31 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
| ECX | Hardware Coordination Feeback | 0 | 0 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
|  | RsvdZ | 1 | 31 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
| EDX | RsvdZ | 0 | 31 | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared | Passthrough for the partition possessing the CpuPowerManagement privilege , otherwise Cleared |
| 0x00000007 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x00000008 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x00000009 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x0000000A |  |  |  |  |  |
| EAX | Architectural Perfmon | 0 | 31 | Cleared | Cleared |
| EBX | Architectural Perfmon | 0 | 31 | Cleared | Cleared |
| ECX | Architectural Perfmon | 0 | 31 | Cleared | Cleared |
| EDX | Architectural Perfmon | 0 | 31 | Cleared | Cleared |
| 0x40000000 | Hypervisor CPUID leaf range and vendor ID signature |  |  |  |  |
| EAX | The maximum input value for hypervisor CPUID information | 0 | 31 | At least 0x40000005 | At least 0x40000005 |
| EBX | Hypervisor Vendor ID Signature | 0 | 31 | 0x7263694D—“Micr” | 0x7263694D—“Micr” |
| ECX | Hypervisor Vendor ID Signature | 0 | 31 | 0x666F736F—“osof” | 0x666F736F—“osof” |
| EDX | Hypervisor Vendor ID Signature | 0 | 31 | 0x76482074—“t Hv” | 0x76482074—“t Hv” |
| 0x40000001 | Hypervisor vendor-neutral interface identification |  |  |  |  |
| EAX | Hypervisor Interface Signature | 0 | 31 | 0x31237648—“Hv#1” | 0x31237648—“Hv#1” |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x40000002 | Operating system identity |  |  |  |  |
| EAX | Build Number | 0 | 31 |  |  |
| EBX | Minor Version | 0 | 15 |  |  |
|  | Major Version | 16 | 31 |  |  |
| ECX | Service Pack | 0 | 31 |  |  |
| EDX | Service Number | 0 | 23 |  |  |
|  | Service Branch | 24 | 31 |  |  |
| 0x40000003 |  |  |  |  |  |
| EAX | Feature identification - partition privileges |  |  |  |  |
|  | VP Runtime available | 0 | 0 |  |  |
|  | Partition Reference Counter available | 1 | 1 |  |  |
|  | Basic SynIC MSRs available | 2 | 2 |  |  |
|  | Synthetic Timer MSRs available | 3 | 3 |  |  |
|  | APIC access MSRs available | 4 | 4 |  |  |
|  | Hypercall MSRs available | 5 | 5 |  |  |
|  | Access virtual processor index MSR available | 6 | 6 |  |  |
|  | Virtual System Reset MSR available | 7 | 7 |  |  |
|  | Access statistics pages MSRs available | 8 | 8 |  |  |
|  | RsvdZ | 9 | 31 | Cleared | Cleared |
| EBX | Feature identification - partition creation flags |  |  |  |  |
|  | CreatePartitions | 0 | 0 |  |  |
|  | AccessPartitionId | 1 | 1 |  |  |
|  | AccessMemoryPool | 2 | 2 |  |  |
|  | AdjustMessageBuffers | 3 | 3 |  |  |
|  | PostMessages | 4 | 4 |  |  |
|  | SignalEvents | 5 | 5 |  |  |
|  | CreatePort | 6 | 6 |  |  |
|  | ConnectPort | 7 | 7 |  |  |
|  | AccessStats | 8 | 8 |  |  |
|  | RsvdZ | 9 | 10 |  |  |
|  | Debugging | 11 | 11 |  |  |
|  | CpuPowerManagement | 12 | 12 |  |  |
|  | RsvdZ | 13 | 31 | Cleared | Cleared |
| ECX | Feature identification - power management |  |  |  |  |
|  | Maximum processor power state. 0 is C0, 1 is C1, 2 is C2, 3 is C3. | 0 | 3 |  |  |
|  | RsvdZ | 4 | 31 | Cleared | Cleared |
| EDX | Feature identification - miscellaneous |  |  |  |  |
|  | The MWAIT instruction is available | 0 | 0 |  |  |
|  | Guest debugging support is available | 1 | 1 |  |  |
|  | Performance monitor support is available | 2 | 2 |  |  |
|  | RsvdZ | 3 | 31 | Cleared | Cleared |
| 0x40000004 |  |  |  |  |  |
| EAX | Implementation recommendations |  |  |  |  |
|  | Recommend using hypercall for address space switches rather than MOV to CR3 instruction | 0 | 0 |  |  |
|  | Recommend using hypercall for local TLB flushes rather than INVLPG or MOV to CR3 instructions | 1 | 1 |  |  |
|  | Recommend using hypercall for remote TLB flushes rather than inter-processor interrupts | 2 | 2 |  |  |
|  | Recommend using MSRs for accessing APIC registers EOI, ICR and TPR rather than their memory-mapped counterparts. | 3 | 3 |  |  |
|  | Recommend using the hypervisor-provided MSR to initiate a system RESET. | 4 | 4 |  |  |
|  | Recommend using relaxed timing for this partition. | 5 | 5 |  |  |
|  | RsvdZ | 6 | 31 | Cleared | Cleared |
| EBX | Recommended number of attempts to retry a spinlock failure before notifying the hypervisor about the failures. 0xFFFFFFFF indicates never to retry. | 0 | 31 |  |  |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x40000005 | Implementation limits |  |  |  |  |
| EAX | The maximum number of virtual processors supported. | 0 | 31 |  |  |
| EBX | The maximum number of logical processors supported. | 0 | 31 |  |  |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x40000006 | Implementation hardware features. |  |  |  |  |
| EAX | Support for APIC overlay assist is detected and in use. | 0 | 0 |  |  |
|  | Support for MSR bitmaps is detected and in use. | 1 | 1 |  |  |
|  | Support for architectural performance counters is detected and in use. | 2 | 2 |  |  |
|  | RsvdZ for future Intel-specific features. | 3 | 31 |  |  |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ for future AMD-specific features | 0 | 31 | Cleared | Cleared |
| 0x40000007 - 0x4000007F |  |  |  |  |  |
| EAX - EDX | Reserved for future hypervisor use |  |  |  |  |
| 0x40000080 - 0x400000FF |  |  |  |  |  |
| EAX - EDX | Reserved for use of intercept handlers in the parent partition |  |  |  |  |
| 0x80000000 |  |  |  |  |  |
| EAX | Highest Extended CPUID Leaf | 0 | 31 | Minimum of the hardware value on current logical processor and 0x8000001A | Minimum of the hardware value across all logical processors in the system and 0x8000001A |
| EBX | Processor vendor string | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| ECX | Processor vendor string | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EDX | Processor vendor string | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| 0x80000001 |  |  |  |  |  |
| EAX | Stepping | 0 | 3 | Passthrough | Minimum across all logical processors |
|  | Base Model | 4 | 7 | Passthrough | Passthrough value from processor 0 |
|  | Base Family | 8 | 11 | Passthrough | Passthrough value from processor 0 |
|  | Processor Type | 12 | 13 | Passthrough | Passthrough value from processor 0 |
|  | RsvdZ | 14 | 15 | Cleared | Cleared |
|  | Extended Model | 16 | 19 | Passthrough | Passthrough |
|  | Extended Family | 20 | 27 | Passthrough | Passthrough |
|  | RsvdZ | 28 | 31 | Cleared | Cleared |
| EBX | Brand ID | 0 | 15 | Passthrough | Passthrough |
|  | RsvdZ | 16 | 27 | Cleared | Cleared |
|  | Package Type | 28 | 31 | Passthrough | Passthrough |
| ECX | Extended feature flag / feature identifiers |  |  |  |  |
|  | LahfSahf | 0 | 0 | Passthrough | Set if set on all, otherwise 0 |
|  | CmpLegacy | 1 | 1 | Passthrough | Set if set on all, otherwise 0 |
|  | SVM | 2 | 2 | Cleared | Cleared |
|  | ExtApicSpace | 3 | 3 | Cleared | Cleared |
|  | AltMovCr8 | 4 | 4 | Passthrough | Set if set on all, otherwise 0 |
|  | ABM | 5 | 5 | Passthrough | Set if set on all, otherwise 0 |
|  | SSE4A | 6 | 6 | Passthrough | Set if set on all, otherwise 0 |
|  | MisAlignSse | 7 | 7 | Passthrough | Set if set on all, otherwise 0 |
|  | 3DNowPrefetch | 8 | 8 | Passthrough | Set if set on all, otherwise 0 |
|  | OSVW | 9 | 9 | Passthrough | Set if set on all, otherwise 0 |
|  | RsvdZ | 10 | 10 | Cleared | Cleared |
|  | SSE5 | 11 | 11 | Passthrough | Set if set on all, otherwise 0 |
|  | SKINIT | 12 | 12 | Cleared | Cleared |
|  | WDT | 13 | 13 | Passthrough | Cleared |
|  | RsvdZ | 14 | 31 | Cleared | Cleared |
| EDX | Extended feature flags |  |  |  |  |
|  | FPU | 0 | 0 | Set | Set |
|  | VME | 1 | 1 | Passthrough | Set if set on all, otherwise 0 |
|  | DE | 2 | 2 | Set | Set |
|  | PSE | 3 | 3 | Set | Set |
|  | TSC | 4 | 4 | Set | Set |
|  | MSR | 5 | 5 | Set | Set |
|  | PAE | 6 | 6 | Set | Set |
|  | MCE | 7 | 7 | Set | Set |
|  | CMPXCHG8B | 8 | 8 | Set | Set |
|  | APIC | 9 | 9 | Set | Set |
|  | RsvdZ | 10 | 10 | Cleared | Cleared |
|  | SysCallSysRet | 11 | 11 | Set | Set |
|  | MTRR | 12 | 12 | Set | Set |
|  | PGE | 13 | 13 | Set | Set |
|  | MCA | 14 | 14 | Set | Set |
|  | CMOV | 15 | 15 | Set | Set |
|  | PAT | 16 | 16 | Set | Set |
|  | PSE36 | 17 | 17 | Passthrough | Set if set on all, otherwise 0 |
|  | RsvdZ | 18 | 18 | Cleared | Cleared |
|  | RsvdZ | 19 | 19 | Cleared | Cleared |
|  | Execute disabled / No execute | 20 | 20 | Set | Set |
|  | RsvdZ | 21 | 21 | Cleared | Cleared |
|  | MmxExt | 22 | 22 | Passthrough | Set if set on all, otherwise 0 |
|  | MMX | 23 | 23 | Set | Set |
|  | FXSR | 24 | 24 | Set | Set |
|  | FFXSR | 25 | 25 | Passthrough | Set if set on all, otherwise 0 |
|  | Page1GB | 26 | 26 | Cleared | Cleared |
|  | RDTSCP | 27 | 27 | Passthrough | Cleared |
|  | RsvdZ | 28 | 28 | Cleared | Cleared |
|  | LM | 29 | 29 | Set | Set |
|  | 3DNowExt | 30 | 30 | Passthrough | Set if set on all, otherwise 0 |
|  | 3DNow | 31 | 31 | Passthrough | Set if set on all, otherwise 0 |
| 0x80000002 | Processor name string identifier |  |  |  |  |
| EAX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EBX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| ECX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EDX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| 0x80000003 | Processor name string identifier |  |  |  |  |
| EAX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EBX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| ECX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EDX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| 0x80000004 | Processor name string identifier |  |  |  |  |
| EAX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EBX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| ECX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| EDX | Processor Brand String | 0 | 31 | Passthrough | Passthrough value from processor 0 |
| 0x80000005 | L1 Cache and TLB identifiers (all registers) |  |  |  |  |
| EAX | L1ITlb2and4MSize | 0 | 7 | Passthrough | Passthrough |
|  | L1ITlb2and4MAssoc | 8 | 15 | Passthrough | Passthrough |
|  | L1DTlb2and4MSize | 16 | 23 | Passthrough | Passthrough |
|  | L1DTlb2and4MAssoc | 24 | 31 | Passthrough | Passthrough |
| EBX | L1ITlb4KSize | 0 | 7 | Passthrough | Passthrough |
|  | L1ITlb4KAssoc | 8 | 15 | Passthrough | Passthrough |
|  | L1DTlb4KSize | 16 | 23 | Passthrough | Passthrough |
|  | L1DTlb4KAssoc | 24 | 31 | Passthrough | Passthrough |
| ECX | L1DcLineSize | 0 | 7 | Passthrough | Passthrough |
|  | L1DcLinesPerTag | 8 | 15 | Passthrough | Passthrough |
|  | L1DcAssoc | 16 | 23 | Passthrough | Passthrough |
|  | L1DcSize | 24 | 31 | Passthrough | Passthrough |
| EDX | L1IcLineSize | 0 | 7 | Passthrough | Passthrough |
|  | L1IcLinesPerTag | 8 | 15 | Passthrough | Passthrough |
|  | L1IcAssoc | 16 | 23 | Passthrough | Passthrough |
|  | L1IcSize | 24 | 31 | Passthrough | Passthrough |
| 0x80000006 | L2 Cache and L2 TLB identifiers |  |  |  |  |
| EAX | L2ITlb2and4MSize | 0 | 11 | Passthrough | Passthrough |
|  | L2ITlb2and4MAssoc | 12 | 15 | Passthrough | Passthrough |
|  | L2DTlb2and4MSize | 16 | 27 | Passthrough | Passthrough |
|  | L2DTlb2and4MAssoc | 28 | 31 | Passthrough | Passthrough |
| EBX | L2ITlb4KSize | 0 | 11 | Passthrough | Passthrough |
|  | L2ITlb4KAssoc | 12 | 15 | Passthrough | Passthrough |
|  | L2DTlb4KSize | 16 | 27 | Passthrough | Passthrough |
|  | L2DTlb4KAssoc | 28 | 31 | Passthrough | Passthrough |
| ECX | L2 Line Size | 0 | 7 | Passthrough | Passthrough |
|  | L2 Lines per tag | 8 | 11 | Passthrough | Passthrough |
|  | L2 Associativity | 12 | 15 | Passthrough | Passthrough |
|  | L2 cache size in kilobytes | 16 | 31 | Passthrough | Passthrough |
| EDX | L3LineSize | 0 | 7 | Passthrough | Passthrough |
|  | L3LinesPerTag | 8 | 11 | Passthrough | Passthrough |
|  | L3Assoc | 12 | 15 | Passthrough | Passthrough |
|  | RsvdZ | 16 | 17 | Passthrough | Passthrough |
|  | L3Size | 18 | 31 | Passthrough | Passthrough |
| 0x80000007 | Advanced Power Management Information |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | TS | 0 | 0 | Passthrough | Cleared |
|  | FID | 1 | 1 | Passthrough | Cleared |
|  | VID | 2 | 2 | Passthrough | Cleared |
|  | TTP | 3 | 3 | Passthrough | Cleared |
|  | TM | 4 | 4 | Passthrough | Cleared |
|  | STC | 5 | 5 | Passthrough | Cleared |
|  | 100MhzSteps | 6 | 6 | Passthrough | Cleared |
|  | HwPState | 7 | 7 | Passthrough | Cleared |
|  | TscInvariant | 8 | 8 | Passthrough | Cleared |
|  | RsvdZ | 9 | 31 | Cleared | Cleared |
| 0x80000008 |  |  |  |  |  |
| EAX | PhysAddrSize | 0 | 7 | Passthrough | The size of the physical GPA space that is supported |
|  | LinAddrSize | 8 | 15 | Passthrough | The size of the virtual GPA space that is supported |
|  | RsvdZ | 16 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | NC | 0 | 7 | Passthrough | Passthrough |
|  | RsvdZ | 8 | 11 | Cleared | Cleared |
|  | ApicIdCoreIdSize | 12 | 15 | Passthrough | Passthrough |
|  | RsvdZ | 16 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000009 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x8000000A | SVM revision and feature identification |  |  |  |  |
| EAX | SvmRev | 0 | 7 | Cleared | Cleared |
|  | RsvdZ | 8 | 31 | Cleared | Cleared |
| EBX | NASID | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | NP | 0 | 0 | Cleared | Cleared |
|  | LBRVirt | 1 | 1 | Cleared | Cleared |
|  | SVML | 2 | 2 | Cleared | Cleared |
|  | NRIPS | 3 | 3 | Cleared | Cleared |
|  | RsvdZ | 4 | 31 | Cleared | Cleared |
| 0x8000000B |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x8000000C |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x8000000D |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x8000000E |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x8000000F |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000010 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000011 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000012 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000013 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000014 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000015 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000016 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000017 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000018 |  |  |  |  |  |
| EAX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x80000019 |  |  |  |  |  |
| EAX | L1ITlb1GSize | 0 | 11 | Cleared | Cleared |
|  | L1ITlb1GAssoc | 12 | 15 | Cleared | Cleared |
|  | L1DTlb1GSize | 16 | 27 | Cleared | Cleared |
|  | L1DTlb1GAssoc | 28 | 31 | Cleared | Cleared |
| EBX | L2ITlb1GSize | 0 | 11 | Cleared | Cleared |
|  | L2ITlb1GAssoc | 12 | 15 | Cleared | Cleared |
|  | L2DTlb1GSize | 16 | 27 | Cleared | Cleared |
|  | L2DTlb1GAssoc | 28 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |
| 0x8000001A |  |  |  |  |  |
| EAX | FP128 | 0 | 0 | Passthrough | Set if set on all, otherwise 0 |
|  | MOVU | 1 | 1 | Passthrough | Set if set on all, otherwise 0 |
|  | RsvdZ | 2 | 31 | Cleared | Cleared |
| EBX | RsvdZ | 0 | 31 | Cleared | Cleared |
| ECX | RsvdZ | 0 | 31 | Cleared | Cleared |
| EDX | RsvdZ | 0 | 31 | Cleared | Cleared |

# Appendix : Boot-time CPUID Feature Requirements

The table below contains a list of CPUID values required for the hypervisor to boot.

| Index and Register | Name | Start bit | End bit | Virtualized feature set definition between processors in the same system |
| --- | --- | --- | --- | --- |
| 0x00000000 |  |  |  |  |
| EAX | Maximum valid standard CPUID index | 0 | 31 | Hardware-specific  If the underlying hardware is AMD-based, the value must be at least 0x00000001.  If the underlying hardware is Intel-based, the value must be at least 0x00000004. |
| EBX | Processor vendor string | 0 | 31 | Must match |
| ECX | Processor vendor string | 0 | 31 | Must match |
| EDX | Processor vendor string | 0 | 31 | Must match |
| 0x00000001 |  |  |  |  |
| EAX | Stepping | 0 | 3 | -- |
|  | Base Model | 4 | 7 | Must match |
|  | Base Family | 8 | 11 | Must match |
|  | Processor Type | 12 | 13 | Must match |
|  | RsvdZ | 14 | 15 | -- |
|  | Extended Model | 16 | 19 | Must match |
|  | Extended Family | 20 | 27 | Must match |
|  | RsvdZ | 28 | 31 | -- |
| EBX | Miscellaneous Information |  |  |  |
|  | Brand Identifier | 0 | 7 | -- |
|  | CL Flush size | 8 | 15 | Must match |
|  | Maximum LPs in a physical package | 16 | 23 | Must match |
|  | Initial APIC ID | 24 | 31 | -- |
| ECX | Feature Flags / Identifiers |  |  |  |
|  | SSE3 | 0 | 0 | -- |
|  | RsvdZ | 1 | 2 | -- |
|  | MONITOR | 3 | 3 | -- |
|  | DS-CPL | 4 | 4 | -- |
|  | VMX | 5 | 5 | Must be set everywhere on Intel,  Ignore on AMD |
|  | RsvdZ | 6 | 6 | -- |
|  | EST | 7 | 7 | -- |
|  | TM2 | 8 | 8 | -- |
|  | SSSE3 | 9 | 9 | -- |
|  | CNXTID | 10 | 10 | -- |
|  | RsvdZ | 11 | 12 | -- |
|  | CMPXCHG16B | 13 | 13 | -- |
|  | xTPR | 14 | 14 | -- |
|  | RsvdZ | 15 | 22 | -- |
|  | Population Count | 23 | 23 | -- |
|  |  |  | 25 |  |
|  | XSAVE | 26 | 26 | Must match |
|  | OSXSAVE | 27 | 27 | Must match |
|  | RsvdZ | 28 | 30 | -- |
|  | Hypervisor Present | 31 | 31 | Must be cleared everywhere |
| EDX | Feature Information |  |  |  |
|  | FPU | 0 | 0 | Must be set everywhere |
|  | VME | 1 | 1 | Must be set everywhere |
|  | DE | 2 | 2 | Must be set everywhere |
|  | PSE | 3 | 3 | Must be set everywhere |
|  | TSC | 4 | 4 | Must be set everywhere |
|  | MSR | 5 | 5 | Must be set everywhere |
|  | PAE | 6 | 6 | Must be set everywhere |
|  | MCE | 7 | 7 | Must be set everywhere |
|  | CMPXCHG8B | 8 | 8 | Must be set everywhere |
|  | APIC | 9 | 9 | Must be set everywhere |
|  | RsvdZ | 10 | 10 | -- |
|  | SEP | 11 | 11 | Must be set everywhere |
|  | MTRR | 12 | 12 | Must be set everywhere |
|  | PGE | 13 | 13 | Must be set everywhere |
|  | MCA | 14 | 14 | Must be set everywhere |
|  | CMOV | 15 | 15 | Must be set everywhere |
|  | PAT | 16 | 16 | Must be set everywhere |
|  | PSE-36 | 17 | 17 | Must be set everywhere |
|  | PSN | 18 | 18 | -- |
|  | CLFSH | 19 | 19 | Must be set everywhere |
|  | RsvdZ | 20 | 20 | -- |
|  | DS | 21 | 21 | -- |
|  | ACPI | 22 | 22 | -- |
|  | MMX | 23 | 23 | Must be set everywhere |
|  | FXSR | 24 | 24 | Must be set everywhere |
|  | SSE | 25 | 25 | Must be set everywhere |
|  | SSE2 | 26 | 26 | Must be set everywhere |
|  | SS | 27 | 27 | -- |
|  | HTT | 28 | 28 | -- |
|  | TM | 29 | 29 | -- |
|  | RsvdZ | 30 | 30 | -- |
|  | PBE | 31 | 31 | -- |
| 0x80000000 |  |  |  |  |
| EAX | Highest Extended CPUID Leaf | 0 | 31 | Hardware specific.  If the underlying hardware is AMD-based, the value must be at least 0x80000008.  If the underlying hardware is Intel based, the value must be at least 0x80000001 |
| EBX | Processor vendor string | 0 | 31 | -- |
| ECX | Processor vendor string | 0 | 31 | -- |
| EDX | Processor vendor string | 0 | 31 | -- |
| 0x80000001 |  |  |  |  |
| EAX | Stepping | 0 | 3 | -- |
|  | Base Model | 4 | 7 | Must match |
|  | Base Family | 8 | 11 | Must match |
|  | Processor Type | 12 | 13 | Must match |
|  | RsvdZ | 14 | 15 | -- |
|  | Extended Model | 16 | 19 | Must match |
|  | Extended Family | 20 | 27 | Must match |
|  | RsvdZ | 28 | 31 | -- |
| EBX | Brand ID | 0 | 15 | Must match |
|  | RsvdZ | 16 | 27 | -- |
|  | Package Type | 28 | 31 | Must match |
| ECX | Extended feature flag / feature identifiers |  |  |  |
|  | LahfSahf | 0 | 0 | -- |
|  | CmpLegacy | 1 | 1 | -- |
|  | SVM | 2 | 2 | Must be set everywhere on AMD,  Ignore on Intel |
|  | ExtApicSpace | 3 | 3 | -- |
|  | AltMovCr8 | 4 | 4 | -- |
|  | ABM | 5 | 5 | -- |
|  | SSE4A | 6 | 6 | -- |
|  | MisAlignSse | 7 | 7 | -- |
|  | 3DNowPrefetch | 8 | 8 | -- |
|  | OSVW | 9 | 9 | -- |
|  | RsvdZ | 10 | 10 | -- |
|  | SSE5 | 11 | 11 | -- |
|  | SKINIT | 12 | 12 | -- |
|  | WDT | 13 | 13 | -- |
|  | RsvdZ | 14 | 31 | -- |
| EDX | Extended feature flags |  |  |  |
|  | FPU | 0 | 0 | Must be set everywhere |
|  | VME | 1 | 1 | Must be set everywhere |
|  | DE | 2 | 2 | Must be set everywhere |
|  | PSE | 3 | 3 | Must be set everywhere |
|  | TSC | 4 | 4 | Must be set everywhere |
|  | MSR | 5 | 5 | Must be set everywhere |
|  | PAE | 6 | 6 | Must be set everywhere |
|  | MCE | 7 | 7 | Must be set everywhere |
|  | CMPXCHG8B | 8 | 8 | Must be set everywhere |
|  | APIC | 9 | 9 | Must be set everywhere |
|  | RsvdZ | 10 | 10 | -- |
|  | SysCallSysRet | 11 | 11 | Must be set everywhere |
|  | MTRR | 12 | 12 | Must be set everywhere |
|  | PGE | 13 | 13 | Must be set everywhere |
|  | MCA | 14 | 14 | Must be set everywhere |
|  | CMOV | 15 | 15 | Must be set everywhere |
|  | PAT | 16 | 16 | Must be set everywhere |
|  | PSE36 | 17 | 17 | Must be set everywhere |
|  | RsvdZ | 18 | 19 | -- |
|  | Execute disabled / No execute | 20 | 20 | Must be set everywhere |
|  | RsvdZ | 21 | 21 | -- |
|  | MmxExt | 22 | 22 | -- |
|  | MMX | 23 | 23 | Must be set everywhere |
|  | FXSR | 24 | 24 | Must be set everywhere |
|  | FFXSR | 25 | 25 | -- |
|  | Page1GB | 26 | 26 | -- |
|  | RDTSCP | 27 | 27 | -- |
|  | RsvdZ | 28 | 28 | -- |
|  | LM | 29 | 29 | Must be set everywhere |
|  | 3DNowExt | 30 | 30 | -- |
|  | 3DNow | 31 | 31 | -- |
| 0x80000005 | L1 Cache and TLB identifiers (all registers) |  |  |  |
| EAX | L1ITlb2and4MSize | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | L1ITlb2and4MAssoc | 8 | 15 | Must match on AMD,  Ignore on Intel |
|  | L1DTlb2and4MSize | 16 | 23 | Must match on AMD,  Ignore on Intel |
|  | L1DTlb2and4MAssoc | 24 | 31 | Must match on AMD,  Ignore on Intel |
| EBX | L1ITlb4KSize | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | L1ITlb4KAssoc | 8 | 15 | Must match on AMD,  Ignore on Intel |
|  | L1DTlb4KSize | 16 | 23 | Must match on AMD,  Ignore on Intel |
|  | L1DTlb4KAssoc | 24 | 31 | Must match on AMD,  Ignore on Intel |
| ECX | L1DcLineSize | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | L1DcLinesPerTag | 8 | 15 | Must match on AMD,  Ignore on Intel |
|  | L1DcAssoc | 16 | 23 | Must match on AMD,  Ignore on Intel |
|  | L1DcSize | 24 | 31 | Must match on AMD,  Ignore on Intel |
| EDX | L1IcLineSize | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | L1IcLinesPerTag | 8 | 15 | Must match on AMD,  Ignore on Intel |
|  | L1IcAssoc | 16 | 23 | Must match on AMD,  Ignore on Intel |
|  | L1IcSize | 24 | 31 | Must match on AMD,  Ignore on Intel |
| 0x80000006 | L2 Cache and L2 TLB identifiers |  |  |  |
| EAX | L2ITlb2and4MSize | 0 | 11 | Must match on AMD,  Ignore on Intel |
|  | L2ITlb2and4MAssoc | 12 | 15 | Must match on AMD,  Ignore on Intel |
|  | L2DTlb2and4MSize | 16 | 27 | Must match on AMD,  Ignore on Intel |
|  | L2DTlb2and4MAssoc | 28 | 31 | Must match on AMD,  Ignore on Intel |
| EBX | L2ITlb4KSize | 0 | 11 | Must match on AMD,  Ignore on Intel |
|  | L2ITlb4KAssoc | 12 | 15 | Must match on AMD,  Ignore on Intel |
|  | L2DTlb4KSize | 16 | 27 | Must match on AMD,  Ignore on Intel |
|  | L2DTlb4KAssoc | 28 | 31 | Must match on AMD,  Ignore on Intel |
| ECX | L2 Line Size | 0 | 7 | Must match |
|  | L2 Lines per tag | 8 | 11 | Must match |
|  | L2 Associativity | 12 | 15 | Must match |
|  | L2 cache size in kilobytes | 16 | 31 | Must match |
| EDX | L3LineSize | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | L3LinesPerTag | 8 | 11 | Must match on AMD,  Ignore on Intel |
|  | L3Assoc | 12 | 15 | Must match on AMD,  Ignore on Intel |
|  | RsvdZ | 16 | 17 | Must match on AMD,  Ignore on Intel |
|  | L3Size | 18 | 31 | Must match on AMD,  Ignore on Intel |
| 0x80000008 |  |  |  |  |
| EAX | PhysAddrSize | 0 | 7 | Must be less than or equal to 48 |
|  | LinAddrSize | 8 | 15 | -- |
|  | RsvdZ | 16 | 31 | -- |
| EBX | RsvdZ | 0 | 31 | -- |
| ECX | NC | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | RsvdZ | 8 | 11 | -- |
|  | ApicIdCoreIdSize | 12 | 15 | Must match on AMD,  Ignore on Intel |
|  | RsvdZ | 16 | 31 | -- |
| EDX | RsvdZ | 0 | 31 | -- |
| 0x8000000A | SVM revision and feature identification |  |  |  |
| EAX | SvmRev | 0 | 7 | Must match on AMD,  Ignore on Intel |
|  | RsvdZ | 8 | 31 | -- |
| EBX | NASID | 0 | 31 | Must match on AMD,  Ignore on Intel |
| ECX | RsvdZ | 0 | 31 | -- |
| EDX | NP | 0 | 0 | -- |
|  | LBRVirt | 1 | 1 | -- |
|  | SVML | 2 | 2 | -- |
|  | NRIPS | 3 | 3 | -- |
|  | RsvdZ | 4 | 31 | -- |

# Appendix : Architectural MSRs

The table below contains a list of architectural MSRs and how they are handled by the hypervisor. The default behavior for MSRs not listed here is passthrough to hardware for root, and #GP for non-root.

| MSR Number | MSR Name | Root | Non-Root |
| --- | --- | --- | --- |
| 0x010 | X64\_MSR\_TIME\_STAMP\_COUNTER | Virtual TSC | Virtualize |
| 0x01B | X64\_MSR\_APIC\_BASE | Virtualize | Virtualize |
| 0x0FE | X64\_MSR\_MTRRCAP | Passthrough | Virtualize |
| 0x174 | X64\_MSR\_SYSENTER\_CS | Virtualize | Virtualize |
| 0x175 | X64\_MSR\_SYSENTER\_ESP | Virtualize | Virtualize |
| 0x176 | X64\_MSR\_SYSENTER\_EIP | Virtualize | Virtualize |
| 0x179 | X64\_MSR\_MCG\_CAP | Passthrough | 0 |
| 0x17A | X64\_MSR\_MCG\_STATUS | Virtualize | Virtualize |
| 0x1D9 | X64\_MSR\_DEBUG\_CTL | Virtualize | Virtualize |
| 0x200 | X64\_MSR\_MTRR\_PHYSBASE0 | Passthrough | Virtualize |
| 0x201 | X64\_MSR\_MTRR\_PHYSMASK0 | Passthrough | Virtualize |
| 0x202 | X64\_MSR\_MTRR\_PHYSBASE1 | Passthrough | Virtualize |
| 0x203 | X64\_MSR\_MTRR\_PHYSMASK1 | Passthrough | Virtualize |
| 0x204 | X64\_MSR\_MTRR\_PHYSBASE2 | Passthrough | Virtualize |
| 0x205 | X64\_MSR\_MTRR\_PHYSMASK2 | Passthrough | Virtualize |
| 0x206 | X64\_MSR\_MTRR\_PHYSBASE3 | Passthrough | Virtualize |
| 0x207 | X64\_MSR\_MTRR\_PHYSMASK3 | Passthrough | Virtualize |
| 0x208 | X64\_MSR\_MTRR\_PHYSBASE4 | Passthrough | Virtualize |
| 0x209 | X64\_MSR\_MTRR\_PHYSMASK4 | Passthrough | Virtualize |
| 0x20A | X64\_MSR\_MTRR\_PHYSBASE5 | Passthrough | Virtualize |
| 0x20B | X64\_MSR\_MTRR\_PHYSMASK5 | Passthrough | Virtualize |
| 0x20C | X64\_MSR\_MTRR\_PHYSBASE6 | Passthrough | Virtualize |
| 0x20D | X64\_MSR\_MTRR\_PHYSMASK6 | Passthrough | Virtualize |
| 0x20E | X64\_MSR\_MTRR\_PHYSBASE7 | Passthrough | Virtualize |
| 0x20F | X64\_MSR\_MTRR\_PHYSMASK7 | Passthrough | Virtualize |
| 0x250 | X64\_MSRMTRR\_FIX64K\_00000 | Passthrough | Virtualize |
| 0x258 | X64\_MSR\_MTRR\_FIX16K\_80000 | Passthrough | Virtualize |
| 0x259 | X64\_MSR\_MTRR\_FIX16K\_A0000 | Passthrough | Virtualize |
| 0x268 | X64\_MSR\_MTRR\_FIX4K\_C0000 | Passthrough | Virtualize |
| 0x269 | X64\_MSR\_MTRR\_FIX4K\_C8000 | Passthrough | Virtualize |
| 0x26A | X64\_MSR\_MTRR\_FIX4K\_D0000 | Passthrough | Virtualize |
| 0x26B | X64\_MSR\_MTRR\_FIX4K\_D8000 | Passthrough | Virtualize |
| 0x26C | X64\_MSR\_MTRR\_FIX4K\_E0000 | Passthrough | Virtualize |
| 0x26D | X64\_MSR\_MTRR\_FIX4K\_E8000 | Passthrough | Virtualize |
| 0x26E | X64\_MSR\_MTRR\_FIX4K\_F0000 | Passthrough | Virtualize |
| 0x26F | X64\_MSR\_MTRR\_FIX4K\_F8000 | Passthrough | Virtualize |
| 0x277 | X64\_MSR\_CR\_PAT | Virtualize | Virtualize |
| 0x2FF | X64\_MSR\_MTRR\_DEF\_TYPE | Passthrough | Virtualize |
| 0xC0000080 | X64\_MSR\_EFER | Virtualize | Virtualize |
| 0xC0000081 | X64\_MSR\_STAR | Virtualize | Virtualize |
| 0xC0000082 | X64\_MSR\_LSTAR | Virtualize | Virtualize |
| 0xC0000083 | X64\_MSR\_CSTAR | Virtualize | Virtualize |
| 0xC0000084 | X64\_MSR\_SFMASK | Virtualize | Virtualize |
| 0xC0000100 | X64\_MSR\_FS\_BASE | Virtualize | Virtualize |
| 0xC0000101 | X64\_MSR\_GS\_BASE | Virtualize | Virtualize |
| 0xC0000102 | X64\_MSR\_KERNEL\_GS\_BASE | Virtualize | Virtualize |

# Appendix : Vendor-Specific MSRs

**G.1 AMD-specific MSRs**

The table below contains a list of AMD-specific MSRs and how they are handled by the hypervisor.

The default behavior for MSRs not listed here is passthrough to hardware for root, and #GP for non-root.

| MSR Number | MSR Name | Root | Non-Root |
| --- | --- | --- | --- |
| 0xC0010010 | AMD\_MSR\_SYSCFG | Passthrough | Reads return 0. Writes ignored. |
| 0xC001001F | AMD\_MSR\_NB\_CFG | Passthrough | Reads return 0. Writes ignored. |

**G.2 Intel-specific MSRs**

The table below contains a list of Intel-specific MSRs and how they are handled by the hypervisor.

The default behavior for MSRs not listed here is passthrough to hardware for root, and #GP for non-root.

| MSR Number | MSR Name | Root | Non-Root |
| --- | --- | --- | --- |
| 0x006 | INTEL\_MSR\_MONITOR\_FILTER\_SIZE | #GP | #GP |
| 0x017 | INTEL\_MSR\_PLATFORM\_ID | Passthrough | Reads return 0.  Writes ignored. |
| 0x03A | INTEL\_MSR\_FEATURE\_CONTROL | #GP | #GP |
| 0x079 | INTEL\_MSR\_BIOS\_UPDT\_TRIG | Virtualize | Reads return 0.  Writes ignored. |
| 0x08B | INTEL\_MSR\_BIOS\_SIGN\_ID | Passthrough | Reads return 8FFFFFFF indicating the best possible patch is already loaded.  Writes ignored. |
| 0x1A0 | INTEL\_MSR\_MISC\_ENABLE | Passthrough | Reads passthrough,  Writes ignored |
| 0x38D | INTEL\_MSR\_PERF\_CAPABILITIES | #GP | #GP |
| 0x38E | INTEL\_MSR\_PER\_GLOBAL\_STATUS | #GP | #GP |
| 0x38F | INTEL\_MSR\_PERF\_GLOBAL\_CTRL | #GP | #GP |
| 0x390 | INTEL\_MSR\_PERF\_GLOBAL\_OVF\_CTRL | #GP | #GP |
| 0x3F1 | INTEL\_MSR\_PEBS\_ENABLE | #GP | #GP |
| 0x480 – 0x48A | INTEL\_MSR\_VMX\_CAPSx | #GP | #GP |
| 0x600 | INTEL\_MSR\_DS\_AREA | #GP | #GP |

# Appendix : Hypervisor Synthetic MSRs

The following is a table of new MSR values defined by the hypervisor.

| MSR Number | MSR Name | Privilege Required (if any) | Access | Description |
| --- | --- | --- | --- | --- |
| 0x40000000 | HV\_X64\_MSR\_GUEST\_OS\_ID | AccessHypercallMsrs | R/W | Used to identify the guest OS running in the partition. See section . |
| 0x40000001 | HV\_X64\_MSR\_HYPERCALL | AccessHypercallMsrs | R/W | Used to enable the hypercall interface. See section . |
| 0x40000002 | HV\_X64\_MSR\_VP\_INDEX | AccessVpIndex | R | Specifies the virtual processor’s index. See section . |
| 0x40000003 | HV\_X64\_MSR\_RESET | AccessSystemResetMsr | R/W | Used to perform a hypervisor-controlled reboot operation. See section . |
| 0x40000010 | HV\_X64\_MSR\_VP\_RUNTIME | AccessVpRuntimeMsr | R | Specifies the virtual processor’s run time in 100ns units. See section . |
| 0x40000020 | HV\_X64\_MSR\_TIME\_REF\_COUNT | AccessPartitionReferenceCounter | R | Partition-wide reference counter. See section . |
| 0x40000021 | HV\_X64\_MSR\_REFERENCE\_TSC | AccessPartitionReferenceCounter | R | Partition-wide reference time stamp counter. See section 15.4 |
| 0x40000070 | HV\_X64\_MSR\_EOI | AccessApicMsrs | W | Fast access to the APIC EOI register. See section . |
| 0x40000071 | HV\_X64\_MSR\_ICR | AccessApicMsrs | R/W | Fast access to the APIC ICR high and ICR low registers. See section . |
| 0x40000072 | HV\_X64\_MSR\_TPR | AccessApicMsrs | R/W | Fast access to the APIC TPR register (use CR8 in 64-bit mode). See section . |
| 0x40000073 | HV\_X64\_MSR\_APIC\_ASSIST\_PAGE | AccessApicMsrs | R/W | Enables lazy EOI processing. See section . |
| 0x40000080 | HV\_X64\_MSR\_SCONTROL | AccessSynicMsrs | R/W | Used to control specific behaviors of the synthetic interrupt controller. See section . |
| 0x40000081 | HV\_X64\_MSR\_SVERSION | AccessSynicMsrs | R | Specifies the SynIC version. See section . |
| 0x40000082 | HV\_X64\_MSR\_SIEFP | AccessSynicMsrs | R/W | Controls the base address of the synthetic interrupt event flag page. See section . |
| 0x40000083 | HV\_X64\_MSR\_SIMP | AccessSynicMsrs | R/W | Controls the base address of the synthetic interrupt parameter page. See section . |
| 0x40000084 | HV\_X64\_MSR\_EOM | AccessSynicMsrs | W | Indicates the end of message in the SynIC. See section . |
| 0x40000090 | HV\_X64\_MSR\_SINT0 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 0. See section . |
| 0x40000091 | HV\_X64\_MSR\_SINT1 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 1. See section . |
| 0x40000092 | HV\_X64\_MSR\_SINT2 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 2. See section . |
| 0x40000093 | HV\_X64\_MSR\_SINT3 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 3. See section . |
| 0x40000094 | HV\_X64\_MSR\_SINT4 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 4. See section . |
| 0x40000095 | HV\_X64\_MSR\_SINT5 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 5. See section . |
| 0x40000096 | HV\_X64\_MSR\_SINT6 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 6. See section . |
| 0x40000097 | HV\_X64\_MSR\_SINT7 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 7. See section . |
| 0x40000098 | HV\_X64\_MSR\_SINT8 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 8. See section . |
| 0x40000099 | HV\_X64\_MSR\_SINT9 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 9. See section . |
| 0x4000009A | HV\_X64\_MSR\_SINT10 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 10. See section . |
| 0x4000009B | HV\_X64\_MSR\_SINT11 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 11. See section . |
| 0x4000009C | HV\_X64\_MSR\_SINT12 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 12. See section . |
| 0x4000009D | HV\_X64\_MSR\_SINT13 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 13. See section . |
| 0x4000009E | HV\_X64\_MSR\_SINT14 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 14. See section . |
| 0x4000009F | HV\_X64\_MSR\_SINT15 | AccessSynicMsrs | R/W | Configures synthetic interrupt source 15. See section . |
| 0x400000B0 | HV\_X64\_MSR\_STIMER0\_CONFIG | AccessSyntheticTimerMsrs | R/W | Configuration register for synthetic timer 0. See section . |
| 0x400000B1 | HV\_X64\_MSR\_STIMER0\_COUNT | AccessSyntheticTimerMsrs | R/W | Expiration time or period for synthetic timer 0. See section . |
| 0x400000B2 | HV\_X64\_MSR\_STIMER1\_CONFIG | AccessSyntheticTimerMsrs | R/W | Configuration register for synthetic timer 1. See section . |
| 0x400000B3 | HV\_X64\_MSR\_STIMER1\_COUNT | AccessSyntheticTimerMsrs | R/W | Expiration time or period for synthetic timer 1. See section . |
| 0x400000B4 | HV\_X64\_MSR\_STIMER2\_CONFIG | AccessSyntheticTimerMsrs | R/W | Configuration register for synthetic timer 2. See section . |
| 0x400000B5 | HV\_X64\_MSR\_STIMER2\_COUNT | AccessSyntheticTimerMsrs | R/W | Expiration time or period for synthetic timer 2. See section . |
| 0x400000B6 | HV\_X64\_MSR\_STIMER3\_CONFIG | AccessSyntheticTimerMsrs | R/W | Configuration register for synthetic timer 3. See section . |
| 0x400000B7 | HV\_X64\_MSR\_STIMER3\_COUNT | AccessSyntheticTimerMsrs | R/W | Expiration time or period for synthetic timer 3. See section . |
| 0x400000C1 | HV\_X64\_MSR\_POWER\_STATE\_TRIGGER\_C1 | CpuPowerManagement | R | Trigger the transition to power state C1 |
| 0x400000C2 | HV\_X64\_MSR\_POWER\_STATE\_TRIGGER\_C2 | CpuPowerManagement | R | Trigger the transition to power state C2 |
| 0x400000C3 | HV\_X64\_MSR\_POWER\_STATE\_TRIGGER\_C3 | CpuPowerManagement | R | Trigger the transition to power state C3 |
| 0x400000D1 | HV\_X64\_MSR\_POWER\_STATE\_CONFIG\_C1 | CpuPowerManagement | R/W | Configure the recipe for power state transitions to C1 |
| 0x400000D2 | HV\_X64\_MSR\_POWER\_STATE\_CONFIG\_C2 | CpuPowerManagement | R/W | Configure the recipe for power state transitions to C2 |
| 0x400000D3 | HV\_X64\_MSR\_POWER\_STATE\_CONFIG\_C3 | CpuPowerManagement | R/W | Configure the recipe for power state transitions to C3 |
| 0x400000E0 | HV\_X64\_MSR\_STATS\_PARTITION\_RETAIL\_PAGE | AccessStatsMsr | R/W | Map the guest’s retail partition statistics page |
| 0x400000E1 | HV\_X64\_MSR\_STATS\_PARTITION\_INTERNAL\_PAGE | AccessStatsMsr | R/W | Map the guest’s internal partition statistics page |
| 0x400000E2 | HV\_X64\_MSR\_STATS\_VP\_RETAIL\_PAGE | AccessStatsMsr | R/W | Map the guest’s retail VP statistics page |
| 0x400000E3 | HV\_X64\_MSR\_STATS\_VP\_INTERNAL\_PAGE | AccessStatsMsr | R/W | Map the guest’s internal VP statistics page |
| 0x400000F0 | HV\_X64\_MSR\_GUEST\_IDLE | AccessGuestIdleMsr | R | Trigger the guest’s transition to the idle power state |

# Appendix : Event Log Entries

Note: In the table below, rows with bold entries in the “Event Name” column and 8-byte entries in the “Event Number” column represent event groups, not individual events. Not all groups have events defined, but the groups are included for completeness.

| Event Name | Event Number | Event Description | | Parameters | |
| --- | --- | --- | --- | --- | --- |
| Boot Manager | 0x0000010000000000 |  | | | |
|  | | | | | |
| Dispatch Manager | 0x0000000000000200 |  | | | |
|  | | | | | |
| Hypercall | 0x0000000000000400 |  | | | |
| HV\_TR\_HC\_HYPERCALL | 0x1A | Hypercall | | Guest RIP,  Input Value | |
|  | | | | | |
| Intercept Manager | 0x0000000000000800 |  | | | |
| HV\_TR\_IM\_GUEST\_EXCEPTION | 0x20 | Guest exception | | Guest RIP,  Type | |
| HV\_TR\_IM\_MSR\_READ | 0x21 | MSR Read | | Guest RIP,  Index,  Value (0 if unsuccessful) | |
| HV\_TR\_IM\_MSR\_WRITE | 0x22 | MSR Write | | Guest RIP,  Index,  Value | |
| HV\_TR\_IM\_ CR\_READ | 0x23 | Control Register Read | | Guest RIP,  Index,  Value (0 if unsuccessful) | |
| HV\_TR\_IM\_CR\_WRITE | 0x24 | Control Register Write | | Guest RIP,  Index,  Value | |
| HV\_TR\_IM\_HLT\_INSTRUCTION | 0x25 | Hlt Instruction | | Guest RIP | |
| HV\_TR\_IM\_MWAIT\_INSTRUCTION | 0x26 | MWAIT Instruction | | Guest RIP,  MWAIT Extension,  MWAIT Hint | |
| HV\_TR\_IM\_CPUID\_INSTRUCTION | 0x27 | CPUID Instruction | | Guest RIP,  EAX Input,  ECX Input | |
| HV\_TR\_IM\_IO\_PORT\_READ | 0x28 | IO Port Read | | Guest RIP,  Port,  Value (0 if unsuccessful) | |
| HV\_TR\_IM\_IO\_PORT\_WRITE | 0x29 | IO Port Write | | Guest RIP,  Port,  Value | |
| HV\_TR\_IM\_EXTERNAL\_INTERRUPT | 0x2A | External interrupt | | Vector | |
| HV\_TR\_IM\_ INTERRUPT\_PENDING | 0x2B | Virtual Interrupt Pending | | Guest RIP | |
| HV\_TR\_IM\_GUEST\_SHUTDOWN | 0x2C | Guest Shutdown | | Guest RIP | |
| HV\_TR\_IM\_EMULATED\_INSTRUCTION | 0x2D | Emulated Instruction | | Guest RIP | |
| HV\_TR\_IM\_NMI\_INTERRUPT | 0x2E | Non-Maskable Interrupt | | Guest RIP | |
| HV\_TR\_IM\_INVLPG\_INSTRUCTION | 0x2F | INVLPG Instruction | | Guest RIP,  Guest VA | |
| HV\_TR\_IM\_IRET\_INSTRUCTION | 0x30 | IRET Instruction | | Guest RIP | |
| HV\_TR\_IM\_TASK\_SWITCH | 0x31 | Task Switch | | Guest RIP | |
| HV\_TR\_IM\_INVD\_INSTRUCTION | 0x32 | INVD Instruction | | Guest RIP | |
| HV\_TR\_IM\_ DR\_ACCESS | 0x33 | Debug Register Access | | Guest RIP | |
| HV\_TR\_IM\_FERR\_FREEZE | 0x34 | FERR Freeze | | Guest RIP | |
| HV\_TR\_IM\_REAL\_MODE\_INTERRUPT | 0x35 | Emulated Real-Mode Interrupt | | Guest RIP | |
| HV\_TR\_IM\_MEMORY\_INTERCEPT | 0x36 | Memory Intercept | | Guest RIP,  GPA Intercept Type | |
| HV\_TR\_IM\_REFLECTED EXCEPTION | 0x37 | Reflected Guest Exception | | Guest RIP,  Exception Type | |
|  | | | | | |
| Instruction Completion | 0x0000000000001000 |  | | | |
|  | | | | | |
| Object Manager | 0x0000000000002000 |  | | | |
| HV\_TR\_OB\_CREATE\_PARTITION | 0x40 | Create Partition | | Partition ID,  Hv Status | |
| HV\_TR\_OB\_DELETE\_PARTITION | 0x41 | Delete Partition | | Partition ID,  Hv Status | |
|  | | | | | |
| Partition Manager | 0x0000000000004000 |  | | | |
| HV\_TR\_PT\_REFERENCE\_TIME | 0x46 | Partition Reference Time | | | TscOffset, ReferenceTimeOffset |
|  | | | | | |
| Virtual Processor Manager | 0x0000000000008000 |  | | | |
| HV\_TR\_VP\_CREATE\_VP | 0x4B | Create Virtual Processor | | Partition ID,  Thread ID,  Hv Status | |
| HV\_TR\_VP\_DELETE\_VP | 0x4C | Delete Virtual Processor | | Partition ID,  Thread ID,  Hv Status | |
|  | | | | | |
| Synthetic Interrupt Controller | 0x0000000000010000 |  | | | |
|  | | | | | |
| Synthetic Timers | 0x0000000000020000 |  | | | |
|  | | | | | |
| Address Manager – Guest Virtual Addresses | 0x0000000000040000 |  | | | |
| HV\_TR\_AM\_GVA\_GROW\_VIRTUAL\_TLB | 0x70 | Grow Virtual TLB | | Old Count,  New Count | |
| HV\_TR\_AM\_GVA\_SHRINK\_VIRTUAL\_TLB | 0x71 | Shrink Virtual TLB | | Old Count,  New Count,  Free Count | |
| HV\_TR\_AM\_GVA\_FLUSH\_VIRTUAL\_TLB | 0x72 | FlushVirtual TLB | | Partition ID | |
|  | | | | | |
| Address Manager | 0x0000000000080000 |  | | | |
|  | | | | | |
| Virtual Abstraction Layer | 0x0000000000100000 |  | | | |
|  | | | | | |
| Virtualization Manager | 0x0000000000200000 |  | | | |
|  | | | | | |
| Scheduler | 0x0000000000400000 |  | | | |
| HV\_TR\_SCH\_CONTEXT\_SWITCH | 0xB0 | Context Switch | | Partition ID,  Thread ID,  TSC Offset (perf only) | |
|  | | | | | |
| Threads | 0x0000000000800000 |  | | | |
|  | | | | | |
| Timers | 0x0000000001000000 |  | | | |
|  | | | | | |
| Kernel | 0x0000000002000000 |  | | | |
|  | | | | | |
| Memory Manager | 0x0000000004000000 |  | | | |
|  | | | | | |
| Profiler | 0x0000000008000000 |  | | | |
| HV\_TR\_SCH\_PROFILER\_SAMPLE | 0xF0 | Context Switch | | Partition ID,  Thread ID,  TSC Offset (perf only) | |
| HV\_TR\_PROFILER\_HV\_MODULE | 0Xf1 | |  | | |

# Appendix : Statistics Counter Definitions

The hypervisor exposes statistics counters to guests, upon request, using overlay pages in the partition’s GPA space. Each page is a packed collection of statistics groups. Each group is prefixed by a header that contains the group’s identification, the format version and the size of the group in bytes (not including the header). The array of values within the group (after the header) may be accessed using a zero-based index into the array. Index zero references the group’s header. Each 64-bit value is of a particular type.

This appendix describes each supported statistics page and the groups that it may contain. Statistics Counters are discussed in detail in chapter .

**J.1 Hypervisor Counters**Hypervisor Counters describe hypervisor-wide states and items. There is only one Hypervisor Counter Object in the system consisting of only architectural values. As counters of a global class they can only be accessed by the root partition.

**J.1.1 Architected Hypervisor Counters Group**

**Group identifier:** HV\_STATISTICS\_TYPE\_HCA\_ID **Group version:** HV\_STATISTICS\_TYPE\_HCA\_VERSION

| Architected Hypervisor Counters  (accessible from the root partition only) | | | |
| --- | --- | --- | --- |
| Index | Counter | Type | Description |
| 1 | Logical Processors | UINT64 | The number of logical processors present in the system. |
| 2 | Partitions | UINT64 | The number of partitions (virtual machines) present in the system. |
| 3 | Total Pages | UINT64 | The number of bootstrap and deposited pages in the hypervisor. |
| 4 | Virtual Processors | UINT64 | The number of virtual processors present in the system. |
| 5 | Monitored Notifications | UINT64 | The number of monitored notifications registered with the hypervisor. |

**J.2 Logical Processor Counters**

Counters for Logical Processor Objects provide information about logical processor-related states and items. The number of Logical Processor Objects matches the number of logical processors in the system and consists of both architectural and release-specific values. As counters of a global class they can be only accessed by the root partition.

**J.2.1 Architected Logical Processor Counters Group**

**Group identifier:** HV\_STATISTICS\_TYPE\_LPA\_ID **Group version:** HV\_STATISTICS\_TYPE\_LPA\_VERSION

| Architected Logical Processor Counters  (accessible from the root partition only) | | | |
| --- | --- | --- | --- |
| Index | Counter | Type | Description |
| 1 | Global Time | UINT64 | The global time on this logical processor. |
| 2 | Total Run Time | UINT64 | The total time (in 100ns) spent by the processor in guest and hypervisor code. |
| 3 | Hypervisor Run Time | UINT64 | The total time (in 100ns) spent by the processor in hypervisor code. |
| 4 | Hardware Interrupts/sec | UINT64 | The rate of hardware interrupts on the processor (excluding hypervisor interrupts). |
| 5 | Context Switches/sec | UINT64 | The rate of virtual processor context switches on the processor. |
| 6 | Inter-Processor Interrupts/sec | UINT64 | The rate of hypervisor inter-processor interrupts delivered to the processor. |
| 7 | Scheduler Interrupts/sec | UINT64 | The rate of hypervisor scheduler interrupts on the processor. |
| 8 | Timer Interrupts/sec | UINT64 | The rate of hypervisor timer interrupts on the processor. |
| 9 | Inter-Processor Interrupts Sent/sec | UINT64 | The rate of hypervisor inter-processor interrupts sent by the processor. |
| 10 | Processor Halts/sec | UINT64 | The rate of entries into a halt state by the processor. |
| 11 | Monitor Transition Cost | UINT64 | The hardware cost of transitions into the hypervisor. |
| 12 | Context Switch Time | UINT64 | The total time (in nanoseconds) spent switching between virtual processors. |
| 13 | C1 Transitions/sec | UINT64 | The rate at which that CPU enters the C1 low-power idle state. |
| 14 | % C1 Time | UINT64 | The percentage of time the processor spends in the C1 low-power idle state. % C1 Time is a subset of the total processor idle time. |
| 15 | C2 Transitions/sec | UINT64 | The rate at which that CPU enters the C2 low-power idle state. |
| 16 | % C2 Time | UINT64 | The percentage of time the processor spends in the C2 low-power idle state. % C2 Time is a subset of the total processor idle time. |
| 17 | C3 Transitions/sec | UINT64 | The rate at which that CPU enters the C3 low-power idle state. |
| 18 | % C3 Time |  | The percentage of time the processor spends in the C3 low-power idle state. % C3 Time is a subset of the total processor idle time. |

**J.3 Partition Counters**

Counters for Partition Objects provide information about states and items for a particular partition. The number of Partition Objects matches the number of partitions present in the system consisting of only architectural values. As counters of a local class they can be accessed by a partition and its parent. In Version 1, the root partition is the parent partition.

**J.3.1 Architected Partition Counters Group**

**Group identifier:** HV\_STATISTICS\_TYPE\_PA\_ID **Group version:** HV\_STATISTICS\_TYPE\_PA\_VERSION

| Architected Partition Counters  (accessible from a partition and its parent) | | | |
| --- | --- | --- | --- |
| Index | Counter | Type | Description |
| 1 | Virtual Processors | UINT64 | The number of virtual processors present in the partition. |
|  |  |  |  |

**J.3.2 Release-specific Partition Counters Group**

**Group identifier:** HV\_STATISTICS\_TYPE\_PV\_ID  
**Group version:**  HV\_STATISTICS\_TYPE\_PV\_VERSION

| Release-specific Partition Counters  (accessible from a partition and its parent) | | | |
| --- | --- | --- | --- |
| Index | Counter | Type | Description |
| 1 | Virtual TLB Pages | UINT64 | The number of pages used by the virtual TLB of the partition. |
| 2 | Address Spaces | UINT64 | The number of address spaces in the virtual TLB of the partition. |
| 3 | Deposited Pages | UINT64 | The number of pages deposited into the partition. |
| 4 | GPA Pages | UINT64 | The number of pages present in the GPA space of the partition (zero for root partition). |
| 5 | GPA Space Modifications/sec | UINT64 | The rate of modifications to the GPA space of the partition. |
| 6 | Virtual TLB Flush Entries/sec | UINT64 | The rate of flushes of the entire virtual TLB. |
| 7 | Recommended Virtual TLB Size | UINT64 | The recommended number of pages to be deposited for the virtual TLB. |

**J.4 Virtual Processor Counters**

Counters for Virtual Processor Objects provide information about states and items for a particular virtual processor. The number of Virtual Processor Objects matches the number of virtual processors present in a given partition and consists of both architectural and release-specific values. As counters of a local class they can be accessed by the partition and its parent. In Version 1, the root partition is the parent partition.

**J.4.1 Architected Virtual Processor Counters Group**

**Group identifier:** HV\_STATISTICS\_TYPE\_VPA\_ID **Group version:** HV\_STATISTICS\_TYPE\_VPA\_VERSION

| Architected Virtual Processor Counters  (accessible from a partition and its parent) | | | |
| --- | --- | --- | --- |
| Index | Counter | Type | Description |
| 1 | Total Run Time | UINT64 | The total run time (in 100ns) spent by the processor in guest and hypervisor code. |
| 2 | Hypervisor Run Time | UINT64 | The total time (in 100ns) spent by the virtual processor in hypervisor code. |

**J.4.2 Release-specific Virtual Processor Counters Group**

**Group Identifier:** HV\_STATISTICS\_TYPE\_VPV\_ID **Group version:** HV\_STATISTICS\_TYPE\_VPV\_VERSION

| Release-specific Virtual Processor Counters  (accessible from a partition and its parent) | | | |
| --- | --- | --- | --- |
| Index | Counter | Type | Description |
| 1 | Hypercalls/sec | UINT64 | The rate of the hypercalls made by guest code on the virtual processor. |
| 2 | Hypercalls Cost | UINT64 | The average time (in nanoseconds) spent processing a hypercall. |
| 3 | Page Invalidations/sec | UINT64 | The rate of INVLPG instructions executed by guest code on the virtual processor. |
| 4 | Page Invalidations Cost | UINT64 | The average time (in nanoseconds) spent processing an INVLPG instruction. |
| 5 | Control Register Accesses/sec | UINT64 | The rate of control register accesses by guest code on the virtual processor. |
| 6 | Control Register Accesses Cost | UINT64 | The average time (in nanoseconds) spent processing a control register access. |
| 7 | IO Instructions/sec | UINT64 | The rate of IO instructions executed by guest code on a virtual processor. |
| 8 | IO Instructions Cost | UINT64 | The average time (in nanoseconds) spent processing an IO instruction. |
| 9 | HLT Instructions/sec | UINT64 | The rate of HLT instructions executed by guest code on a virtual processor. |
| 10 | HLT Instructions Cost | UINT64 | The average time (in nanoseconds) spent processing an HLT instruction. |
| 11 | MWAIT Instructions/sec | UINT64 | The rate of MWAIT instructions executed by guest code on a virtual processor. |
| 12 | MWAIT Instructions Cost | UINT64 | The average time (in nanoseconds) spent processing an MWAIT instruction. |
| 13 | CPUID Instructions/sec | UINT64 | The rate of CPUID instructions executed by guest code on a virtual processor. |
| 14 | CPUID Instructions Cost | UINT64 | The average time (in nanoseconds) spent processing an CPUID instruction. |
| 15 | MSR Accesses/sec | UINT64 | The rate of MSR instructions executed by guest code on a virtual processor. |
| 16 | MSR Accesses Cost | UINT64 | The average time (in nanoseconds) spent processing an MSR instruction. |
| 17 | Other Intercepts/sec | UINT64 | The rate of other intercepts triggered by guest code on a virtual processor. |
| 18 | Other Intercepts Cost | UINT64 | The average time (in nanoseconds) spent processing other intercepts. |
| 19 | External Interrupts/sec | UINT64 | The rate of external interrupts received by the hypervisor while executing guest code on the virtual processor. |
| 20 | External Interrupts Cost | UINT64 | The average time (in nanoseconds) spent processing an external interrupt. |
| 21 | Pending Interrupts/sec | UINT64 | The rate of intercepts due to a task priority (TPR) reduction by guest code on the virtual processor. |
| 22 | Pending Interrupts Cost | UINT64 | The average time (in nanoseconds) spent processing a pending interrupt intercept. |
| 23 | Emulated Instructions/sec | UINT64 | The rate of emulated instructions while executing guest code on the virtual processor. |
| 24 | Emulated Instructions Cost | UINT64 | The average time (in nanoseconds) spent emulating an instruction. |
| 25 | Debug Register Accesses/sec | UINT64 | The rate of debug register accesses by guest code on the virtual processor. |
| 26 | Debug Register Accesses Cost | UINT64 | The average time (in nanoseconds) spent handling a debug register access. |
| 27 | Page Fault Intercepts/sec | UINT64 | The rate of page fault exceptions intercepted by the hypervisor while executing guest code on the virtual processor. |
| 28 | Page Fault Intercepts Cost | UINT64 | The average time (in nanoseconds) spent processing a page fault intercept. |
| 29 | Guest Page Table Maps/sec | UINT64 | The rate of map operations for guest page table pages. |
| 30 | Large Page TLB Fills/sec | UINT64 | The rate of virtual TLB misses on large pages. |
| 31 | Small Page TLB Fills/sec | UINT64 | The rate of virtual TLB misses on 4K pages. |
| 32 | Reflected Guest Page Faults/sec | UINT64 | The rate of page fault exceptions delivered to the guest. |
| 33 | APIC MMIO Accesses/sec | UINT64 | The rate of APIC MMIO register accesses by guest code on the virtual processor. |
| 34 | IO Intercept Messages/sec | UINT64 | The rate of IO intercept messages sent to the parent partition. |
| 35 | Memory Intercept Messages/sec | UINT64 | The rate of memory intercept messages sent to the parent partition. |
| 36 | APIC EOI Accesses/sec | UINT64 | The rate of APIC EOI register writes by guest code on the virtual processor. |
| 37 | Other Messages/sec | UINT64 | The rate of other intercept messages sent to the parent partition. |
| 38 | Page Table Allocations/sec | UINT64 | The rate of page table allocations in the virtual TLB. |
| 39 | Logical Processor Migrations/sec | UINT64 | The rate of migrations by the virtual processor to a different logical processor. |
| 40 | Address Space Evictions/sec | UINT64 | The rate of address space evictions in the virtual TLB. |
| 41 | Address Space Switches/sec | UINT64 | The rate of address space switches by guest code on the virtual processor. |
| 42 | Address Domain Flushes/sec | UINT64 | The rate of explicit flushes of the virtual TLB by guest code on the virtual processor. |
| 43 | Address Space Flushes/sec | UINT64 | The rate of explicit flushes of one address space by guest code on the virtual processor. |
| 44 | Global GVA Range Flushes/sec | UINT64 | The rate of explicit flushes of a virtual address range in all address spaces by guest code on the virtual processor. |
| 45 | Local Flushed GVA Ranges/sec | UINT64 | The rate of explicit flushes of a virtual address range in one address space by guest code on the virtual processor. |
| 46 | Page Table Evictions/sec | UINT64 | The rate of page table evictions in the virtual TLB. |
| 47 | Page Table Reclamations/sec | UINT64 | The rate of reclamations of unreferenced page tables in the virtual TLB. |
| 48 | Page Table Resets/sec | UINT64 | The rate of page table resets In the virtual TLB. |
| 49 | Page Table Validations/sec | UINT64 | The rate of page table validations to remove state entries in the virtual TLB. |
| 50 | APIC TPR Accesses/sec | UINT64 | The rate of APIC TPR accesses by guest code on the virtual processor. |
| 51 | Page Table Write Intercepts/sec | UINT64 | The rate of write intercepts on guest page tables by guest code on the virtual processor. |
| 52 | Synthetic Interrupts/sec | UINT64 | The rate of synthetic interrupts delivered to the virtual processor. |
| 53 | Virtual Interrupts/sec | UINT64 | The rate of interrupts (including synthetic interrupts) delivered to the virtual processor. |
| 54 | APIC IPIs Sent/sec | UINT64 | The rate of APIC inter-processor interrupts (including to self) sent to the virtual processor. |
| 55 | APIC Self IPIs Sent/sec | UINT64 | The rate of APIC interrupts sent by the virtual processor to itself. |
| 56 | GPA Space Hypercalls/sec | UINT64 | The rate of Guest Physical Address Space hypercalls made by guest code on the virtual processor. |
| 57 | Logical Processor Hypercalls/sec | UINT64 | The rate of Logical Processor hypercalls made by guest code on the virtual processor. |
| 58 | Long Spin Wait Hypercalls/sec | UINT64 | The rate of Long Spin Wait hypercalls made by guest code on the virtual processor. |
| 59 | Other Hypercalls/sec | UINT64 | The rate of other hypercalls made by guest code on the virtual processor. |
| 60 | Synthetic Interrupt Hypercalls/sec | UINT64 | The rate of Synthetic Interrupt hypercalls made by guest code on the virtual processor. |
| 61 | Virtual Interrupt Hypercalls/sec | UINT64 | The rate of Virtual Interrupt hypercalls made by guest code on the virtual processor. |
| 62 | Virtual MMU Hypercalls/sec | UINT64 | The rate of Virtual MMU hypercalls made by guest code on the virtual processor. |
| 63 | Virtual Processor Hypercalls/sec | UINT64 | The rate of Virtual Processor hypercalls made by guest code on the virtual processor. |

**Index**

A

AccessPartitionId

in MSR 7

Active *See* Partition State

C

CPUID

Instruction 5

Microsoft leafs 6

0x40000000 6

0x40000001 6

0x40000002 6

0x40000003 6

0x40000004 8

0x40000005 8

standard leafs 5

0x40000000 5

0x40000001 5

CreatePartitions

in MSR 7

E

Endianness 4

enum 4

Exceptions

#GP

AccessApicMsrs privilege 24

AccessHypercallMsrs privilege 24

AccessSynicMsrs privilege 24

AccessSyntheticTimers privilege 24

AccessVpIndex privilege 24

AccessVpRuntime privilege 24

feature not present 66

hypercall interface 18

read from APIC EOI MSR 110

reference counter MSR 151

segment registers 80

write to SVERSION MSR 124

#MC

SIM and SIEF pages 126

statistics page mappings 207

#UD

hypercall environment 12

MONITOR instruction 14, 88

MWAIT instruction 14, 88

double/triple faults 91

F

Finalized *See* Partition State

Finalizing *See* Partition State

G

GPA *See* guest physical address

guest physical address 3

guest virtual address 3

GVA *See* guest virtual address

H

HV\_ADDRESS\_SPACE\_ID 95

HV\_CACHE\_TYPE 96

HV\_FLUSH\_FLAGS 96

HV\_GPA 3

HV\_GPA\_PAGE\_NUMBER 3

HV\_GVA 3

HV\_GVA\_PAGE\_NUMBER 3

HV\_INTERCEPT\_ACCESS\_TYPE\_MASK 67

HV\_INTERCEPT\_DESCRIPTOR 67

HV\_INTERCEPT\_PARAMETERS 67

HV\_INTERCEPT\_TYPE 67

HV\_LOGICAL\_PROCESSOR\_INDEX 43

HV\_MAP\_GPA\_FLAGS 57, 172

HV\_NANO100\_TIME 45

HV\_PARTITION\_ID 21, 179

HV\_PARTITION\_ID\_INVALID 21

HV\_PARTITION\_PRIVILEGE\_MASK 23

HV\_PARTITION\_PROPERTY 21

HV\_PERF\_COUNTER\_CONFIGURATION 217

HV\_REGISTER\_NAME 76

HV\_REGISTER\_VALUE 76

HV\_SPA 3

HV\_SPA\_PAGE\_NUMBER 3

HV\_STATS\_OBJECT\_IDENTITY 210

HV\_STATS\_OBJECT\_TYPE 209

HV\_STATUS 3

HV\_SYSTEM\_PROPERTY 217

HV\_TRACE\_BUFFER\_HEADER 188

HV\_TRACE\_BUFFER\_STATE 187

HV\_TRACE\_TYPE 187

HV\_VP\_INDEX 72

HvAllocateTraceBufferGroup 190, 195, 203

HvAssertVirtualInterrupt 115

HvClearVirtualInterrupt 117

HvConnectPort 137

HvCreatePartition 31

HvCreatePort 134

HvCreateVp 81

HvDeallocateTraceBufferGroup 191, 192, 193, 194

HvDeletePartition 34

HvDeletePort 136

HvDeleteVp 82

HvDepositMemory 51

HvDisconnectPort 141

HvEnableTraceEvents 195

HvFinalizePartition 33, 218, 219

HvFlushVirtualAddressList 100

HvFlushVirtualAddressSpace 99

HvFreeTraceBuffer 196

HvGetLogicalProcessorRunTime 46

HvGetMemoryBalance 53

HvGetNextChildPartition 38

HvGetPartitionId 37

HvGetPartitionProperty 35

HvGetSystemProperty 218

HvGetVpRegisters 83

HvInitializePartition 32

HvInstallIntercept 68

HvMapGpaPages 57, 61

HvMapStatsPage 210

HvParkLogicalProcessors 47

HvPostMessage 142

HvReadGpa 105

HvRestorePartitionState 174

HvSavePartitionState 172

HvSetLogicalProcessorRunTimeGroup 179

HvSetPartitionProperty 36

HvSetSystemProperty 218

HvSetVpRegisters 84

HvSwitchVirtualAddressSpace 98

HvTranslateVirtualAddress 102

HvUnmapGpaPages 59

HvUnmapStatsPage 212

HvWithdrawMemory 52

HvWriteGpa 106

hypercall inputvalue 13

hypercall output value 15

hypercall page 17

hypercalls 5, 11

rep 11

simple 11

I

intercept 65

L

logical processors 42

M

memory pool 49

memory-mapped registers 5

message buffer 119

MNF *See* Monitored Notification Faciliity

monitor page 122

monitored notification 122

Monitored Notification Facility 122

MSR 5

HV\_X64\_MSR\_GUEST\_OS\_ID 9, 18

HV\_X64\_MSR\_HYPERCALL 17

HV\_X64\_MSR\_TIMER\_REF\_COUNT 7

HV\_X64\_MSR\_VP\_RUNTIME 7

values

0x40000000 9

N

native interface 15

O

overlay pages 56

restrictions 13

P

Partition State

Active 26

Finalized 26

Finalizing 26

Uninitialized 25

PHV\_GPA\_PAGE\_NUMBER 3

physical node 42

pointers 4

processor interrupts 5

Proximity Domains 50

R

Reference Counter 147

rep count 11

reps completed count 11

reserved values 1

root partition 215

Rsvd 1

RsvdP 1

RsvdZ 1

S

SPA *See* system physical address

SPA page ranges 41

SPA space *See* system physical address space

status

HV\_STATUS\_INVALID\_ALIGNMENT 17

HV\_STATUS\_INVALID\_HYPERCALL\_CODE 17

HV\_STATUS\_INVALID\_HYPERCALL\_INPUT 17

HV\_STATUS\_SUCCESS 17

Synthetic Timers 147

system physical address 3

system physical address space 41

System properties 217

T

TLFS (Top Level Functional Specification) 1

U

Uninitialized *See* Partition State

V

versioning 8

W

WinHv.sys 15

wrapper interface 15

1. Some implementations may restrict this partition privilege to the root partition. [↑](#footnote-ref-1)
2. This feature set is a subset of all processor features as exposed by CPUID. Other features are either hidden from non-root guests or are required to be enabled for Hyper-V to launch. [↑](#footnote-ref-2)
3. Mda is also known as MessageDestinationAddress. [↑](#footnote-ref-3)
4. Only the root partition may map global statistics pages. [↑](#footnote-ref-4)
5. Status code is currently referenced by a released version of Microsoft Windows ® [↑](#endnote-ref-1)