Skip to content

Commit

Permalink
Add FlatPageTableLib
Browse files Browse the repository at this point in the history
Description

Creates a new library, FlatPageTableLib, which works on X64 and AARCH64
platforms and converts the page table to a "flat" version. The flat
version is a one-dimensional array where each entry is an address,
a length, and attributes. The library will walk the page/translation
table and combine blocks/leaves with the same attributes into a single
entry in the flat array. The attributes mask for each architecture is
defined in the header and includes both the upper and lower block/leaf
attributes. On both X64 and AARCH64, the hierarchical inheritance of
attributes is factored into the determination of block/leaf attributes.
This allows the consumer of the library to easily check the attributes
of any region in the page/translation table.

- [x] Impacts functionality?
  - **Functionality** - Does the change ultimately impact how firmware functions?
  - Examples: Add a new library, publish a new PPI, update an algorithm, ...
- [ ] Impacts security?
  - **Security** - Does the change have a direct security impact on an application,
    flow, or firmware?
  - Examples: Crypto algorithm change, buffer overflow fix, parameter
    validation improvement, ...
- [ ] Breaking change?
  - **Breaking change** - Will anyone consuming this change experience a break
    in build or boot behavior?
  - Examples: Add a new library class, move a module to a different repo, call
    a function in a new library class in a pre-existing module, ...
- [ ] Includes tests?
  - **Tests** - Does the change include any explicit test code?
  - Examples: Unit tests, integration tests, robot tests, ...
- [ ] Includes documentation?
  - **Documentation** - Does the change contain explicit documentation additions
    outside direct code modifications (and comments)?
  - Examples: Update readme file, add feature readme file, link to documentation
    on an a separate Web page, ...

How This Was Tested

Tested by running the paging audit on SBSA and Q35 and by comparing
the output against the Memory Attribute Protocol

Integration Instructions

N/A
  • Loading branch information
TaylorBeebe committed Sep 29, 2023
1 parent 05f1d88 commit 8914491
Show file tree
Hide file tree
Showing 7 changed files with 795 additions and 0 deletions.
166 changes: 166 additions & 0 deletions UefiTestingPkg/Include/Library/FlatPageTableLib.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,166 @@
/** @file
Library to parse page/translation table entries.
Copyright (c) Microsoft Corporation.
SPDX-License-Identifier: BSD-2-Clause-Patent
**/

#ifndef PAGE_TABLE_ATTRIBUTE_LIB_H_
#define PAGE_TABLE_ATTRIBUTE_LIB_H_

typedef struct {
UINT64 LinearAddress;
UINT64 Length;
UINT64 PageEntry;
} PAGE_MAP_ENTRY;

typedef struct {
UINT32 ArchSignature;
PAGE_MAP_ENTRY *Entries;
UINTN EntryCount;
UINTN EntryPagesAllocated;
} PAGE_MAP;

// The signature of the PAGE_MAP struct is used to determine the architecture of the page/translation table
// entries.
#define AARCH64_PAGE_MAP_SIGNATURE SIGNATURE_32 ('A','A','6','4')
#define X64_PAGE_MAP_SIGNATURE SIGNATURE_32 ('X','6','4',' ')

// This union can be used to interpret each PAGE_MAP_ENTRY on an AARCH64 system.
typedef union {
struct {
UINT64 Valid : 1; // BIT0
UINT64 BlockOrTable : 1; // BIT1
UINT64 AttributeIndex : 3; // BIT2-4
UINT64 NonSecure : 1; // BIT5
UINT64 AccessPermissions : 2; // BIT6-7
UINT64 Shareability : 2; // BIT8-9
UINT64 AccessFlag : 1; // BIT10
UINT64 NonGlobal : 1; // BIT11
UINT64 Oa : 4; // BIT12-15
UINT64 Nt : 1; // BIT16
UINT64 OutputAddress : 33; // BIT17-49
UINT64 Guarded : 1; // BIT50
UINT64 Dirty : 1; // BIT51
UINT64 Contiguous : 1; // BIT52
UINT64 Pxn : 1; // BIT53
UINT64 Xn : 1; // BIT54
UINT64 Ignored : 4; // BIT55-58
UINT64 PageBasedHardwareAttribute : 4; // BIT59-62
UINT64 Reserved : 1; // BIT63
} Bits;
UINT64 Uint64;
} AARCH64_PAGE_MAP_ENTRY;

// This union can be used to interpret each PAGE_MAP_ENTRY on an X64 system.
typedef union {
struct {
UINT64 Present : 1; // BIT0
UINT64 ReadWrite : 1; // BIT1
UINT64 UserSupervisor : 1; // BIT2
UINT64 WriteThrough : 1; // BIT3
UINT64 CacheDisabled : 1; // BIT4
UINT64 Accessed : 1; // BIT5
UINT64 Dirty : 1; // BIT6
UINT64 Pat : 1; // BIT7
UINT64 Global : 1; // BIT8
UINT64 Reserved1 : 3; // BIT9-11
UINT64 PageTableBaseAddress : 40; // BIT12-51
UINT64 Reserved2 : 7; // BIT52-58
UINT64 ProtectionKey : 4; // BIT59-62
UINT64 Nx : 1; // BIT63
} Bits;
UINT64 Uint64;
} X64_PAGE_MAP_ENTRY;

// When the page/translation table is parsed to create an array of PAGE_MAP_ENTRY, the following bitmasks
// are used to determine the attributes of one page/translation table entry are the same as another
// page/translation table entry. If contiguous leaf/block entries have the same attributes, then they
// will be represented in a single PAGE_MAP_ENTRY.
#define AARCH64_ATTRIBUTES_MASK ((0xFFFULL << 52) | (0x3FFULL << 2))
#define X64_ATTRIBUTES_MASK ((0xFFFULL << 52) | 0xFFFULL)

/**
Populate the input page/translation table map.
@param[in, out] Map Pointer to the PAGE_MAP struct to be populated.
@retval RETURN_SUCCESS The translation table is parsed successfully.
@retval RETURN_INVALID_PARAMETER MapCount is NULL, or Map is NULL but *MapCount is nonzero.
@retval RETURN_BUFFER_TOO_SMALL *MapCount is too small.
MapCount is updated to indicate the expected number of entries.
Caller may still get RETURN_BUFFER_TOO_SMALL with the new MapCount.
**/
EFI_STATUS
EFIAPI
CreateFlatPageTable (
IN OUT PAGE_MAP *Map
);

/**
Checks the input flat page/translation table for the input region and converts the associated
table entries to EFI access attributes.
@param[in] Map Pointer to the PAGE_MAP struct to be parsed
@param[in] Address Start address of the region
@param[in] Length Length of the region
@param[out] Attributes EFI Attributes of the region (EFI_MEMORY_XP, EFI_MEMORY_RO, EFI_MEMORY_RP)
@retval EFI_SUCCESS The output Attributes is valid
@retval EFI_INVALID_PARAMETER The flat translation table has not been built or
Attributes was NULL or Length was 0
@retval EFI_NOT_FOUND The input region could not be found.
**/
EFI_STATUS
EFIAPI
GetRegionAccessAttributes (
IN PAGE_MAP *Map,
IN UINT64 Address,
IN UINT64 Length,
OUT UINT64 *Attributes
);

/**
Parses the input page to determine if it is writable.
@param[in] Page The page entry to parse.
@retval TRUE The page is writable.
@retval FALSE The page is not writable.
**/
BOOLEAN
EFIAPI
IsPageWritable (
IN UINT64 Page
);

/**
Parses the input page to determine if it is executable.
@param[in] Page The page entry to parse.
@retval TRUE The page is executable.
@retval FALSE The page is not executable.
**/
BOOLEAN
EFIAPI
IsPageExecutable (
IN UINT64 Page
);

/**
Parses the input page to determine if it is readable.
@param[in] Page The page entry to parse.
@retval TRUE The page is readable.
@retval FALSE The page is not readable.
**/
BOOLEAN
EFIAPI
IsPageReadable (
IN UINT64 Page
);

#endif
Loading

0 comments on commit 8914491

Please sign in to comment.