| [Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
 Re: Proposal for Porting Xen to Armv8-R64 - DraftA
 
To: Wei Chen <Wei.Chen@xxxxxxx>, "xen-devel@xxxxxxxxxxxxxxxxxxxx"	<xen-devel@xxxxxxxxxxxxxxxxxxxx>, "julien@xxxxxxx" <julien@xxxxxxx>, Stefano Stabellini <sstabellini@xxxxxxxxxx>From: Ayan Kumar Halder <ayan.kumar.halder@xxxxxxxxxx>Date: Thu, 24 Feb 2022 11:52:07 +0000Arc-authentication-results: i=1; mx.microsoft.com 1; spf=pass (sender ip is 149.199.80.198) smtp.rcpttodomain=arm.com smtp.mailfrom=xilinx.com; dmarc=pass (p=none sp=none pct=100) action=none header.from=xilinx.com; dkim=none (message not signed); arc=noneArc-message-signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=xVrjslPv3gt8iFlmxkUL7Svwo3yvaf6VhUzZ7WVUCsQ=; b=PyEmFDouJvCGz3hu/391ulGmT+zoO+Ics0dOJl+5HPPXUrVOBxD4yusjcH8xuPlxr15VEQcvBLTROLY6skrUj2DeQEAaro0Q5vEx12rBTxUV6dJCFMqA3q/xwr1YmundQnMk7tFtyyzPT5k2td0NR6mY4CQ+t0iCk6Ix1AWdPZxrdh9vBfAJ70xY3H43FMdeMAK7Cwq4xQImHKWjaP4ic7P6Ld6SPQYuwsEIm4KixSYyjfuxVUKL5OagVlsMSgkCwxJMMn30M7PhfRYh0usZydeGvaRym7gDjOtjfNJmPtoEdEBegi1Y1FJKO1owH+Ml9MaqAFXNuGdsSm9CItinDA==Arc-seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=abQ1ypetRZK3kvIjehD/4iqKAoCEsH6TOv+kzm+bqFh/TTKKqHbLVqAUB/On5O43qnW3lrwnB+FHMhfurOah6UplnW0Z+jsr1KX2U68f+HJZaR7SqZB82IlI2UwxveGQwbW8Q95vG/Hb1T1qTGErAlE+CfCHBFNBhVB1Mekci1lishNMAmIZQrTeZ1lfME6a5SyXKMfEV8/wMhnsIq7ZGJbGGUwyafEsEgIRPip4RdHs0sXwk0+QvPwd8nZ1y6S57WPrVjZDRQYNcULMZbifKwyn37fegxF2F10HJLZfShxLTrk616z7BbXLJRu6W+aU323y/UhOS7BnHwG4ykC4Ww==Cc: Bertrand Marquis <Bertrand.Marquis@xxxxxxx>, Penny Zheng	<Penny.Zheng@xxxxxxx>, Henry Wang <Henry.Wang@xxxxxxx>, nd <nd@xxxxxxx>Delivery-date: Thu, 24 Feb 2022 11:52:26 +0000List-id: Xen developer discussion <xen-devel.lists.xenproject.org> 
 
Hi Wei,
This is a nice writeup. I have a few initial queries.
On 24/02/2022 06:01, Wei Chen wrote:
 
# Proposal for Porting Xen to Armv8-R64
This proposal will introduce the PoC work of porting Xen to Armv8-R64,
which includes:
- The changes of current Xen capability, like Xen build system, memory
   management, domain management, vCPU context switch.
- The expanded Xen capability, like static-allocation and direct-map.
***Notes:***
1. ***This proposal only covers the work of porting Xen to Armv8-R64***
    ***single CPU. Xen SMP support on Armv8-R64 relates to Armv8-R***
    ***Trusted-Frimware (TF-R). This is an external dependency,***
    ***so we think the discussion of Xen SMP support on Armv8-R64***
    ***should be started when single-CPU support is complete.***
2. ***This proposal will not touch xen-tools. In current stage,***
    ***Xen on Armv8-R64 only support dom0less, all guests should***
    ***be booted from device tree.***
## 1. Essential Background
### 1.1. Armv8-R64 Profile
The Armv-R architecture profile was designed to support use cases that
have a high sensitivity to deterministic execution. (e.g. Fuel Injection,
Brake control, Drive trains, Motor control etc)
Arm announced Armv8-R in 2013, it is the latest generation Arm architecture
targeted at the Real-time profile. It introduces virtualization at the highest
security level while retaining the Protected Memory System Architecture (PMSA)
based on a Memory Protection Unit (MPU). In 2020, Arm announced Cortex-R82,
which is the first Arm 64-bit Cortex-R processor based on Armv8-R64.
 
Is there some good document explaining the difference between MPU and 
MMU ? And when do we need one vs other. 
 
- The latest Armv8-R64 document can be found here:
   [Arm Architecture Reference Manual Supplement - Armv8, for Armv8-R AArch64 
architecture profile](https://developer.arm.com/documentation/ddi0600/latest/).
- Armv-R Architecture progression:
   Armv7-R -> Armv8-R AArch32 -> Armv8 AArch64
   The following figure is a simple comparison of "R" processors based on
   different Armv-R Architectures.
   

- The Armv8-R architecture evolved additional features on top of Armv7-R:
     - An exception model that is compatible with the Armv8-A model
     - Virtualization with support for guest operating systems
         - PMSA virtualization using MPUs In EL2.
- The new features of Armv8-R64 architecture
     - Adds support for the 64-bit A64 instruction set, previously Armv8-R
       only supported A32.
     - Supports up to 48-bit physical addressing, previously up to 32-bit
       addressing was supported.
     - Optional Arm Neon technology and Advanced SIMD
     - Supports three Exception Levels (ELs)
         - Secure EL2 - The Highest Privilege, MPU only, for firmware, 
hypervisor
         - Secure EL1 - RichOS (MMU) or RTOS (MPU)
         - Secure EL0 - Application Workloads
     - Optionally supports Virtual Memory System Architecture at S-EL1/S-EL0.
       This means it's possible to run rich OS kernels - like Linux - either
       bare-metal or as a guest.
- Differences with the Armv8-A AArch64 architecture
     - Supports only a single Security state - Secure. There is not Non-Secure
       execution state supported.
 
If so, then I guess there is no Trustzone kind of protection available. 
I mean where application is normal world can request for data to be 
processed in secure world (by switching the NS bit on AXI). 
Also, does Armv8-R support Trustzone controller 400 which helps to 
partition memory into different protected enclaves based on NSAID ? 
(Apologies if my queries are irrelevant, I am asking this purely out of 
my own interest :) ) 
 So, can it support running both RTOS and Linux as guests ? My 
understanding is no as we can't enable MPU (for RTOS) and MMU (for 
Linux) at the same time. There needs to be two separate images of Xen. 
Please confirm.
     - EL3 is not supported, EL2 is mandatory. This means secure EL2 is the
       highest EL.
     - Supports the A64 ISA instruction
         - With a small set of well-defined differences
     - Provides a PMSA (Protected Memory System Architecture) based
       virtualization model.
         - As opposed to Armv8-A AArch64's VMSA based Virtualization
         - Can support address bits up to 52 if FEAT_LPA is enabled,
           otherwise 48 bits.
         - Determines the access permissions and memory attributes of
           the target PA.
         - Can implement PMSAv8-64 at EL1 and EL2
             - Address translation flat-maps the VA to the PA for EL2 Stage 1.
             - Address translation flat-maps the VA to the PA for EL1 Stage 1.
             - Address translation flat-maps the IPA to the PA for EL1 Stage 2.
     - PMSA in EL1 & EL2 is configurable, VMSA in EL1 is configurable.
### 1.2. Xen Challenges with PMSA Virtualization
Xen is PMSA unaware Type-1 Hypervisor, it will need modifications to run
with an MPU and host multiple guest OSes.
- No MMU at EL2:
     - No EL2 Stage 1 address translation
         - Xen provides fixed ARM64 virtual memory layout as basis of EL2
           stage 1 address translation, which is not applicable on MPU system,
           where there is no virtual addressing. As a result, any operation
           involving transition from PA to VA, like ioremap, needs modification
           on MPU system.
     - Xen's run-time addresses are the same as the link time addresses.
         - Enable PIC (position-independent code) on a real-time target
           processor probably very rare.
     - Xen will need to use the EL2 MPU memory region descriptors to manage
       access permissions and attributes for accesses made by VMs at EL1/0.
         - Xen currently relies on MMU EL1 stage 2 table to manage these
           accesses.
- No MMU Stage 2 translation at EL1:
     - A guest doesn't have an independent guest physical address space
     - A guest can not reuse the current Intermediate Physical Address
       memory layout
     - A guest uses physical addresses to access memory and devices
     - The MPU at EL2 manages EL1 stage 2 access permissions and attributes
- There are a limited number of MPU protection regions at both EL2 and EL1:
     - Architecturally, the maximum number of protection regions is 256,
       typical implementations have 32.
     - By contrast, Xen does not need to consider the number of page table
       entries in theory when using MMU.
- The MPU protection regions at EL2 need to be shared between the hypervisor
   and the guest stage 2.
     - Requires careful consideration - may impact feature 'fullness' of both
       the hypervisor and the guest
     - By contrast, when using MMU, Xen has standalone P2M table for guest
       stage 2 accesses.
 
## 2. Proposed changes of Xen
### **2.1. Changes of build system:**
- ***Introduce new Kconfig options for Armv8-R64***:
   Unlike Armv8-A, because lack of MMU support on Armv8-R64,
 
But Armv8-R64 supports VMSA (Refer
ARM DDI 0600A.d ID120821, B1.2.2,
Virtual Memory System Architecture, VMSAv8-64). So it should support 
MMU, isn't it ? 
- Ayan
 
we may not
   expect one Xen binary to run on all machines. Xen images are not common
   across Armv8-R64 platforms. Xen must be re-built for different Armv8-R64
   platforms. Because these platforms may have different memory layout and
   link address.
     - `ARM64_V8R`:
       This option enables Armv8-R profile for Arm64. Enabling this option
       results in selecting MPU. This Kconfig option is used to gate some
       Armv8-R64 specific code except MPU code, like some code for Armv8-R64
       only system ID registers access.
     - `ARM_MPU`
       This option enables MPU on ARMv8-R architecture. Enabling this option
       results in disabling MMU. This Kconfig option is used to gate some
       ARM_MPU specific code. Once when this Kconfig option has been enabled,
       the MMU relate code will not be built for Armv8-R64. The reason why
       not depends on runtime detection to select MMU or MPU is that, we don't
       think we can use one image for both Armv8-R64 and Armv8-A64. Another
       reason that we separate MPU and V8R in provision to allow to support MPU
       on 32bit Arm one day.
     - `XEN_START_ADDRESS`
       This option allows to set the custom address at which Xen will be
       linked. This address must be aligned to a page size. Xen's run-time
       addresses are the same as the link time addresses. Different platforms
       may have differnt memory layout. This Kconfig option provides users
       the ability to select proper link addresses for their boards.
       ***Notes: Fixed link address means the Xen binary could not be***
       ***relocated by EFI loader. So in current stage, Xen could not***
       ***be launched as an EFI application on Armv8-R64.***
     - `ARM_MPU_NORMAL_MEMORY_START` and `ARM_MPU_NORMAL_MEMORY_END`
       `ARM_MPU_DEVICE_MEMORY_START` and `ARM_MPU_DEVICE_MEMORY_END`
       These Kconfig options allow to set memory regions for Xen code, data
       and device memory. Before parsing memory information from device tree,
       Xen will use the values that stored in these options to setup boot-time
       MPU configuration. Why we need a boot-time MPU configuration?
       1. More deterministic: Arm MPU supports background regions,
          if we don't configure the MPU regions and don't enable MPU.
          We can enable MPU background regions. But that means all RAM
          is RWX. Random values in RAM or maliciously embedded data can
          be exploited. Using these Kconfig options allow users to have
          a deterministic RAM area to execute code.
       2. More compatible: On some Armv8-R64 platforms, if the MPU is
          disabled, the `dc zva` instruction will make the system halt.
          And this instruction will be embedded in some built-in functions,
          like `memory set`. If we use `-ddont_use_dc` to rebuild GCC,
          the built-in functions will not contain `dc zva`. However, it is
          obviously unlikely that we will be able to recompile all GCC
          for ARMv8-R64.
       3. One optional idea:
           We can map `XEN_START_ADDRESS` to `XEN_START_ADDRESS + 2MB` or
           `XEN_START_ADDRESS` to `XEN_START_ADDRESS + image_end` for
           MPU normal memory. It's enough to support Xen run in boot time.
- ***Define new system registers for compilers***:
   Armv8-R64 is based on Armv8.4. That means we will use some Armv8.4
   specific system registers. As Armv8-R64 only have secure state, so
   at least, `VSTCR_EL2` and `VSCTLR_EL2` will be used for Xen. And the
   first GCC version that supports Armv8.4 is GCC 8.1. In addition to
   these, PMSA of Armv8-R64 introduced lots of MPU related system registers:
   `PRBAR_ELx`, `PRBARx_ELx`, `PRLAR_ELx`, `PRLARx_ELx`, `PRENR_ELx` and
   `MPUIR_ELx`. But the first GCC version to support these system registers
   is GCC 11. So we have two ways to make compilers to work properly with
   these system registers.
   1. Bump GCC version to GCC 11.
      The pros of this method is that, we don't need to encode these
      system registers in macros by ourselves. But the cons are that,
      we have to update Makefiles to support GCC 11 for Armv8-R64.
      1.1. Check the GCC version 11 for Armv8-R64.
      1.2. Add march=armv8r to CFLAGS for Armv8-R64.
      1.3. Solve the confliction of march=armv8r and mcpu=generic
     These changes will affect common Makefiles, not only Arm Makefiles.
     And GCC 11 is new, lots of toolchains and Distro haven't supported it.
   2. Encode new system registers in macros ***(preferred)***
         ```
         /* Virtualization Secure Translation Control Register */
         #define VSTCR_EL2  S3_4_C2_C6_2
         /* Virtualization System Control Register */
         #define VSCTLR_EL2 S3_4_C2_C0_0
         /* EL1 MPU Protection Region Base Address Register encode */
         #define PRBAR_EL1  S3_0_C6_C8_0
         ...
         /* EL2 MPU Protection Region Base Address Register encode */
         #define PRBAR_EL2  S3_4_C6_C8_0
         ...
         ```
      If we encode all above system registers, we don't need to bump GCC
      version. And the common CFLAGS Xen is using still can be applied to
      Armv8-R64. We don't need to modify Makefiles to add specific CFLAGS.
### **2.2. Changes of the initialization process**
In general, we still expect Armv8-R64 and Armv8-A64 to have a consistent
initialization process. In addition to some architecture differences, there
is no more than reusable code that we will distinguish through CONFIG_ARM_MPU
or CONFIG_ARM64_V8R. We want most of the initialization code to be reusable
between Armv8-R64 and Armv8-A64.
- We will reuse the original head.s and setup.c of Arm. But replace the
   MMU and page table operations in these files with configuration operations
   for MPU and MPU regions.
- We provide a boot-time MPU configuration. This MPU configuration will
   support Xen to finish its initialization. And this boot-time MPU
   configuration will record the memory regions that will be parsed from
   device tree.
   In the end of Xen initialization, we will use a runtime MPU configuration
   to replace boot-time MPU configuration. The runtime MPU configuration will
   merge and reorder memory regions to save more MPU regions for guests.
   

- Defer system unpausing domain.
   When Xen initialization is about to end, Xen unpause guests created
   during initialization. But this will cause some issues. The unpause
   action occurs before free_init_memory, however the runtime MPU configuration
   is built after free_init_memory.
   So if the unpaused guests start executing the context switch at this
   point, then its MPU context will base on the boot-time MPU configuration.
   Probably it will be inconsistent with runtime MPU configuration, this
   will cause unexpected problems (This may not happen in a single core
   system, but on SMP systems, this problem is foreseeable, so we hope to
   solve it at the beginning).
### **2.3. Changes to reduce memory fragmentation**
In general, memory in Xen system can be classified to 4 classes:
`image sections`, `heap sections`, `guest RAM`, `boot modules (guest Kernel,
initrd and dtb)`
Currently, Xen doesn't have any restriction for users how to allocate
memory for different classes. That means users can place boot modules
anywhere, can reserve Xen heap memory anywhere and can allocate guest
memory anywhere.
In a VMSA system, this would not be too much of a problem, since the
MMU can manage memory at a granularity of 4KB after all. But in a
PMSA system, this will be a big problem. On Armv8-R64, the max MPU
protection regions number has been limited to 256. But in typical
processor implementations, few processors will design more than 32
MPU protection regions. Add in the fact that Xen shares MPU protection
regions with guest's EL1 Stage 2. It becomes even more important
to properly plan the use of MPU protection regions.
- An ideal of memory usage layout restriction:

1. Reserve proper MPU regions for Xen image (code, rodata and data + bss).
2. Reserve one MPU region for boot modules.
    That means the placement of all boot modules, include guest kernel,
    initrd and dtb, will be limited to this MPU region protected area.
3. Reserve one or more MPU regions for Xen heap.
    On Armv8-R64, the guest memory is predefined in device tree, it will
    not be allocated from heap. Unlike Armv8-A64, we will not move all
    free memory to heap. We want Xen heap is dertermistic too, so Xen on
    Armv8-R64 also rely on Xen static heap feature. The memory for Xen
    heap will be defined in tree too. Considering that physical memory
    can also be discontinuous, one or more MPU protection regions needs
    to be reserved for Xen HEAP.
4. If we name above used MPU protection regions PART_A, and name left
    MPU protection regions PART_B:
    4.1. In hypervisor context, Xen will map left RAM and devices to PART_B.
         This will give Xen the ability to access whole memory.
    4.2. In guest context, Xen will create EL1 stage 2 mapping in PART_B.
         In this case, Xen just need to update PART_B in context switch,
         but keep PART_A as fixed.
***Notes: Static allocation will be mandatory on MPU based systems***
**A sample device tree of memory layout restriction**:
```
chosen {
     ...
     /*
      * Define a section to place boot modules,
      * all boot modules must be placed in this section.
      */
     mpu,boot-module-section = <0x10000000 0x10000000>;
     /*
      * Define a section to cover all guest RAM. All guest RAM must be located
      * within this section. The pros is that, in best case, we can only have
      * one MPU protection region to map all guest RAM for Xen.
      */
     mpu,guest-memory-section = <0x20000000 0x30000000>;
     /*
      * Define a memory section that can cover all device memory that
      * will be used in Xen.
      */
     mpu,device-memory-section = <0x80000000 0x7ffff000>;
     /* Define a section for Xen heap */
     xen,static-mem = <0x50000000 0x20000000>;
     domU1 {
         ...
         #xen,static-mem-address-cells = <0x01>;
         #xen,static-mem-size-cells = <0x01>;
         /* Statically allocated guest memory, within mpu,guest-memory-section 
*/
         xen,static-mem = <0x30000000 0x1f000000>;
         module@11000000 {
             compatible = "multiboot,kernel\0multiboot,module";
             /* Boot module address, within mpu,boot-module-section */
             reg = <0x11000000 0x3000000>;
             ...
         };
         module@10FF0000 {
                 compatible = "multiboot,device-tree\0multiboot,module";
                 /* Boot module address, within mpu,boot-module-section */
                 reg = <0x10ff0000 0x10000>;
                 ...
         };
     };
};
```
### **2.4. Changes of memory management**
Xen is coupled with VMSA, in order to port Xen to Armv8-R64, we have to
decouple Xen from VMSA. And give Xen the ability to manage memory in PMSA.
1. ***Use buddy allocator to manage physical pages for PMSA***
    From the view of physical page, PMSA and VMSA don't have any difference.
    So we can reuse buddy allocator on Armv8-R64 to manage physical pages.
    The difference is that, in VMSA, Xen will map allocated pages to virtual
    addresses. But in PMSA, Xen just convert the pages to physical address.
2. ***Can not use virtual address for memory management***
    As Armv8-R64 only has PMSA in EL2, Xen loses the ability of using virtual
    address to manage memory. This brings some problems, some virtual address
    based features could not work well on Armv8-R64, like `FIXMAP`, 
`vmap/vumap`,
    `ioremap` and `alternative`.
    But the functions or macros of these features are used in lots of common
    code. So it's not good to use `#ifdef CONFIG_ARM_MPU` to gate relate code
    everywhere. In this case, we propose to use stub helpers to make the changes
    transparently to common code.
    1. For `FIXMAP`, we will use `0` in `FIXMAP_ADDR` for all fixmap operations.
       This will return physical address directly of fixmapped item.
    2. For `vmap/vumap`, we will use some empty inline stub helpers:
         ```
         static inline void vm_init_type(...) {}
         static inline void *__vmap(...)
         {
             return NULL;
         }
         static inline void vunmap(const void *va) {}
         static inline void *vmalloc(size_t size)
         {
             return NULL;
         }
         static inline void *vmalloc_xen(size_t size)
         {
             return NULL;
         }
         static inline void vfree(void *va) {}
         ```
    3. For `ioremap`, it depends on `vmap`. As we have make `vmap` to always
       return `NULL`, they could not work well on Armv8-R64 without changes.
       `ioremap` will return input address directly.
         ```
         static inline void *ioremap_attr(...)
         {
             /* We don't have the ability to change input PA cache attributes */
             if ( CACHE_ATTR_need_change )
                 return NULL;
             return (void *)pa;
         }
         static inline void __iomem *ioremap_nocache(...)
         {
             return ioremap_attr(start, len, PAGE_HYPERVISOR_NOCACHE);
         }
         static inline void __iomem *ioremap_cache(...)
         {
             return ioremap_attr(start, len, PAGE_HYPERVISOR);
         }
         static inline void __iomem *ioremap_wc(...)
         {
             return ioremap_attr(start, len, PAGE_HYPERVISOR_WC);
         }
         void *ioremap(...)
         {
             return ioremap_attr(pa, len, PAGE_HYPERVISOR_NOCACHE);
         }
         ```
     4. For `alternative`, it depends on `vmap` too. We will simply disable
        it on Armv8-R64 in current stage. How to implement `alternative`
        on Armv8-R64 is better to be discussed after basic functions of Xen
        on Armv8-R64 work well.
        But simply disable `alternative` will make `cpus_have_const_cap` always
        return false.
         ```
         * System capability check for constant cap */
         #define cpus_have_const_cap(num) ({                \
                register_t __ret;                           \
                                                            \
                asm volatile (ALTERNATIVE("mov %0, #0",     \
                                          "mov %0, #1",     \
                                          num)              \
                              : "=r" (__ret));              \
                                                            \
                 unlikely(__ret);                           \
                 })
         ```
         So, before we have an PMSA `alternative` implementation, we have to
         implement a separate `cpus_have_const_cap` for Armv8-R64:
         ```
         #define cpus_have_const_cap(num) cpus_have_cap(num)
         ```
### **2.5. Changes of guest management**
Armv8-R64 only supports PMSA in EL2, but it supports configurable
VMSA or PMSA in EL1. This means Xen will have a new type guest on
Armv8-R64 - MPU based guest.
1. **Add a new domain type - MPU_DOMAIN**
    When user want to create a guest that will be using MPU in EL1, user
    should add a `mpu` property in device tree `domU` node, like following
    example:
     ```
     domU2 {
         compatible = "xen,domain";
         direct-map;
         mpu; --> Indicates this domain will use PMSA in EL1.
         ...
     };
     ```
     Corresponding to `mpu` property in device tree, we also need to introduce
     a new flag `XEN_DOMCTL_CDF_INTERNAL_mpu` for domain to mark itself as an
     MPU domain. This flag will be used in domain creation and domain doing
     vCPU context switch.
     1. Domain creation need this flag to decide enable PMSA or VMSA in EL1.
     2. vCPU context switch need this flag to decide save/restore MMU or MPU
        related registers.
2. **Add MPU registers to vCPU save EL1 MPU context**
    Current Xen only support MMU based guest, so it hasn't considered to
    save/restore MPU context. In this case, we need to add MPU registers
    to `arch_vcpu`:
     ```
     struct arch_vcpu
     {
     #ifdef CONFIG_ARM_MPU
         /* Virtualization Translation Control Register */
         register_t vtcr_el2;
         /* EL1 MPU regions' registers */
         pr_t mpu_regions[CONFIG_ARM_MPU_EL1_PROTECTION_REGIONS];
     #endif
     }
     ```
     Armv8-R64 can support max to 256 MPU regions. But that's just theoretical.
     So we don't want to define `pr_t mpu_regions[256]`, this is a memory waste
     in most of time. So we decided to let the user specify through a Kconfig
     option. `CONFIG_ARM_MPU_EL1_PROTECTION_REGIONS` default value can be `32`,
     it's a typical implementation on Armv8-R64. Users will recompile Xen when
     their platform changes. So when the MPU changes, respecifying the MPU
     protection regions number will not cause additional problems.
3. **MPU based P2M table management**
    Armv8-R64 EL2 doesn't have EL1 stage 2 address translation. But through
    PMSA, it still has the ability to control the permissions and attributes
    of EL1 stage 2. In this case, we still hope to keep the interface
    consistent with MMU based P2M as far as possible.
    p2m->root will point to an allocated memory. In Armv8-A64, this memory
    is used to save the EL1 stage 2 translation table. But in Armv8-R64,
    this memory will be used to store EL2 MPU protection regions that are
    used by guest. During domain creation, Xen will prepare the data in
    this memory to make guest can access proper RAM and devices. When the
    guest's vCPU will be scheduled in, this data will be written to MPU
    protection region registers.
### **2.6. Changes of exception trap**
As Armv8-R64 has compatible exception mode with Armv8-A64, so we can reuse most
of Armv8-A64's exception trap & handler code. But except the trap based on EL1
stage 2 translation abort.
In Armv8-A64, we use `FSC_FLT_TRANS`
```
     case FSC_FLT_TRANS:
         ...
         if ( is_data )
         {
             enum io_state state = try_handle_mmio(regs, hsr, gpa);
             ...
         }
```
But for Armv8-R64, we have to use `FSC_FLT_PERM`
```
     case FSC_FLT_PERM:
         ...
         if ( is_data )
         {
             enum io_state state = try_handle_mmio(regs, hsr, gpa);
             ...
         }
```
### **2.5. Changes of device driver**
1. Because Armv8-R64 only has single secure state, this will affect some
devices that have two secure state, like GIC. But fortunately, most
vendors will not link a two secure state GIC to Armv8-R64 processors.
Current GIC driver can work well with single secure state GIC for Armv8-R64.
2. Xen should use secure hypervisor timer in Secure EL2. We will introduce
a CONFIG_ARM_SECURE_STATE to make Xen to use secure registers for timer.
### **2.7. Changes of virtual device**
Currently, we only support pass-through devices in guest. Because event
channel, xen-bus, xen-storage and other advanced Xen features haven't been
enabled in Armv8-R64.
--
Cheers,
Wei Chen
 
 
 |