xenia-canary/src/xenia/memory.h

544 lines
21 KiB
C
Raw Normal View History

/**
******************************************************************************
* Xenia : Xbox 360 Emulator Research Project *
******************************************************************************
* Copyright 2020 Ben Vanik. All rights reserved. *
* Released under the BSD license - see LICENSE in the root for more details. *
******************************************************************************
*/
#ifndef XENIA_MEMORY_H_
#define XENIA_MEMORY_H_
#include <cstdint>
#include <memory>
#include <mutex>
2015-05-24 07:02:47 +00:00
#include <string>
#include <utility>
#include <vector>
#include "xenia/base/memory.h"
2015-05-25 06:16:43 +00:00
#include "xenia/base/mutex.h"
2015-02-01 06:49:47 +00:00
#include "xenia/cpu/mmio_handler.h"
2015-12-01 23:26:55 +00:00
namespace xe {
class ByteStream;
} // namespace xe
namespace xe {
class Memory;
enum SystemHeapFlag : uint32_t {
kSystemHeapVirtual = 1 << 0,
kSystemHeapPhysical = 1 << 1,
kSystemHeapDefault = kSystemHeapVirtual,
};
enum class HeapType : uint8_t {
kGuestVirtual,
kGuestXex,
kGuestPhysical,
kHostPhysical,
};
2015-05-16 07:23:13 +00:00
enum MemoryAllocationFlag : uint32_t {
kMemoryAllocationReserve = 1 << 0,
kMemoryAllocationCommit = 1 << 1,
};
enum MemoryProtectFlag : uint32_t {
kMemoryProtectRead = 1 << 0,
kMemoryProtectWrite = 1 << 1,
kMemoryProtectNoCache = 1 << 2,
kMemoryProtectWriteCombine = 1 << 3,
kMemoryProtectNoAccess = 0,
};
// Equivalent to the Win32 MEMORY_BASIC_INFORMATION struct.
2015-05-16 07:23:13 +00:00
struct HeapAllocationInfo {
// A pointer to the base address of the region of pages.
uint32_t base_address;
2015-05-16 07:23:13 +00:00
// A pointer to the base address of a range of pages allocated by the
// VirtualAlloc function. The page pointed to by the BaseAddress member is
// contained within this allocation range.
uint32_t allocation_base;
2015-05-16 07:23:13 +00:00
// The memory protection option when the region was initially allocated.
uint32_t allocation_protect;
// The size specified when the region was initially allocated, in bytes.
uint32_t allocation_size;
2015-05-16 07:23:13 +00:00
// The size of the region beginning at the base address in which all pages
// have identical attributes, in bytes.
uint32_t region_size;
// The state of the pages in the region (commit/free/reserve).
uint32_t state;
// The access protection of the pages in the region.
uint32_t protect;
};
2015-12-03 01:37:48 +00:00
// Describes a single page in the page table.
2015-05-16 07:23:13 +00:00
union PageEntry {
struct {
2015-12-03 01:37:48 +00:00
// Base address of the allocated region in 4k pages.
uint32_t base_address : 20;
// Total number of pages in the allocated region in 4k pages.
uint32_t region_page_count : 20;
// Protection bits specified during region allocation.
// Composed of bits from MemoryProtectFlag.
2015-05-16 07:23:13 +00:00
uint32_t allocation_protect : 4;
2015-12-03 01:37:48 +00:00
// Current protection bits as of the last Protect.
// Composed of bits from MemoryProtectFlag.
2015-05-16 07:23:13 +00:00
uint32_t current_protect : 4;
2015-12-03 01:37:48 +00:00
// Allocation state of the page as a MemoryAllocationFlag bit mask.
2015-05-16 07:23:13 +00:00
uint32_t state : 2;
uint32_t reserved : 14;
};
uint64_t qword;
};
2015-12-03 01:37:48 +00:00
// Heap abstraction for page-based allocation.
2015-05-16 07:23:13 +00:00
class BaseHeap {
public:
virtual ~BaseHeap();
// Offset of the heap in relative to membase, without host_address_offset
// adjustment.
uint32_t heap_base() const { return heap_base_; }
// Length of the heap range.
uint32_t heap_size() const { return heap_size_; }
2015-12-03 01:37:48 +00:00
// Size of each page within the heap range in bytes.
2015-05-16 07:23:13 +00:00
uint32_t page_size() const { return page_size_; }
// Type of specified heap
HeapType heap_type() const { return heap_type_; }
2019-08-04 20:55:54 +00:00
// Offset added to the virtual addresses to convert them to host addresses
// (not including membase).
uint32_t host_address_offset() const { return host_address_offset_; }
template <typename T = uint8_t*>
inline T TranslateRelative(size_t relative_address) const {
return reinterpret_cast<T>(membase_ + heap_base_ + host_address_offset_ +
relative_address);
}
2015-12-03 01:37:48 +00:00
// Disposes and decommits all memory and clears the page table.
2015-05-16 07:23:13 +00:00
virtual void Dispose();
2015-12-03 01:37:48 +00:00
// Dumps information about all allocations within the heap to the log.
2015-05-16 07:23:13 +00:00
void DumpMap();
2016-06-21 15:10:08 +00:00
uint32_t GetTotalPageCount();
uint32_t GetUnreservedPageCount();
2015-12-03 01:37:48 +00:00
// Allocates pages with the given properties and allocation strategy.
// This can reserve and commit the pages as well as set protection modes.
// This will fail if not enough contiguous pages can be found.
2015-05-16 07:23:13 +00:00
virtual bool Alloc(uint32_t size, uint32_t alignment,
uint32_t allocation_type, uint32_t protect, bool top_down,
uint32_t* out_address);
2015-12-03 01:37:48 +00:00
// Allocates pages at the given address.
// This can reserve and commit the pages as well as set protection modes.
// This will fail if the pages are already allocated.
2015-05-16 07:23:13 +00:00
virtual bool AllocFixed(uint32_t base_address, uint32_t size,
uint32_t alignment, uint32_t allocation_type,
uint32_t protect);
2015-12-03 01:37:48 +00:00
// Allocates pages at an address within the given address range.
// This can reserve and commit the pages as well as set protection modes.
// This will fail if not enough contiguous pages can be found.
2015-05-16 07:23:13 +00:00
virtual bool AllocRange(uint32_t low_address, uint32_t high_address,
uint32_t size, uint32_t alignment,
uint32_t allocation_type, uint32_t protect,
bool top_down, uint32_t* out_address);
2015-12-03 01:37:48 +00:00
// Decommits pages in the given range.
// Partial overlapping pages will also be decommitted.
2015-05-16 07:23:13 +00:00
virtual bool Decommit(uint32_t address, uint32_t size);
2015-12-03 01:37:48 +00:00
// Decommits and releases pages in the given range.
// Partial overlapping pages will also be released.
2015-05-16 07:23:13 +00:00
virtual bool Release(uint32_t address, uint32_t* out_region_size = nullptr);
2015-12-03 01:37:48 +00:00
// Modifies the protection mode of pages within the given range.
virtual bool Protect(uint32_t address, uint32_t size, uint32_t protect,
uint32_t* old_protect = nullptr);
2015-05-16 07:23:13 +00:00
2015-12-03 01:37:48 +00:00
// Queries information about the given region of pages.
2015-05-16 07:23:13 +00:00
bool QueryRegionInfo(uint32_t base_address, HeapAllocationInfo* out_info);
2015-12-03 01:37:48 +00:00
// Queries the size of the region containing the given address.
2015-05-16 07:23:13 +00:00
bool QuerySize(uint32_t address, uint32_t* out_size);
2018-02-11 03:58:44 +00:00
// Queries the base and size of a region containing the given address.
bool QueryBaseAndSize(uint32_t* in_out_address, uint32_t* out_size);
2015-12-03 01:37:48 +00:00
// Queries the current protection mode of the region containing the given
// address.
2015-05-16 07:23:13 +00:00
bool QueryProtect(uint32_t address, uint32_t* out_protect);
// Queries the currently strictest readability and writability for the entire
// range.
xe::memory::PageAccess QueryRangeAccess(uint32_t low_address,
uint32_t high_address);
2015-12-01 23:26:55 +00:00
bool Save(ByteStream* stream);
bool Restore(ByteStream* stream);
void Reset();
2015-05-16 07:23:13 +00:00
protected:
BaseHeap();
void Initialize(Memory* memory, uint8_t* membase, HeapType heap_type,
uint32_t heap_base, uint32_t heap_size, uint32_t page_size,
2019-08-04 20:55:54 +00:00
uint32_t host_address_offset = 0);
2015-05-16 07:23:13 +00:00
Memory* memory_;
2015-05-16 07:23:13 +00:00
uint8_t* membase_;
HeapType heap_type_;
2015-05-16 07:23:13 +00:00
uint32_t heap_base_;
uint32_t heap_size_;
uint32_t page_size_;
2019-08-04 20:55:54 +00:00
uint32_t host_address_offset_;
xe::global_critical_region global_critical_region_;
2015-05-16 07:23:13 +00:00
std::vector<PageEntry> page_table_;
};
2015-12-03 01:37:48 +00:00
// Normal heap allowing allocations from guest virtual address ranges.
2015-05-16 07:23:13 +00:00
class VirtualHeap : public BaseHeap {
public:
VirtualHeap();
~VirtualHeap() override;
2015-12-03 01:37:48 +00:00
// Initializes the heap properties and allocates the page table.
void Initialize(Memory* memory, uint8_t* membase, HeapType heap_type,
uint32_t heap_base, uint32_t heap_size, uint32_t page_size);
2015-05-16 07:23:13 +00:00
};
2015-12-03 01:37:48 +00:00
// A heap for ranges of memory that are mapped to physical ranges.
// Physical ranges are used by the audio and graphics subsystems representing
// hardware wired directly to memory in the console.
//
// The physical heap and the behavior of sharing pages with virtual pages is
// implemented by having a 'parent' heap that is used to perform allocation in
// the guest virtual address space 1:1 with the physical address space.
2015-05-16 07:23:13 +00:00
class PhysicalHeap : public BaseHeap {
public:
PhysicalHeap();
~PhysicalHeap() override;
2015-12-03 01:37:48 +00:00
// Initializes the heap properties and allocates the page table.
void Initialize(Memory* memory, uint8_t* membase, HeapType heap_type,
uint32_t heap_base, uint32_t heap_size, uint32_t page_size,
VirtualHeap* parent_heap);
2015-05-16 07:23:13 +00:00
bool Alloc(uint32_t size, uint32_t alignment, uint32_t allocation_type,
uint32_t protect, bool top_down, uint32_t* out_address) override;
bool AllocFixed(uint32_t base_address, uint32_t size, uint32_t alignment,
uint32_t allocation_type, uint32_t protect) override;
bool AllocRange(uint32_t low_address, uint32_t high_address, uint32_t size,
uint32_t alignment, uint32_t allocation_type,
uint32_t protect, bool top_down,
uint32_t* out_address) override;
bool Decommit(uint32_t address, uint32_t size) override;
bool Release(uint32_t base_address,
uint32_t* out_region_size = nullptr) override;
bool Protect(uint32_t address, uint32_t size, uint32_t protect,
uint32_t* old_protect = nullptr) override;
2015-05-16 07:23:13 +00:00
void EnableAccessCallbacks(uint32_t physical_address, uint32_t length,
bool enable_invalidation_notifications,
bool enable_data_providers);
// Returns true if any page in the range was watched.
bool TriggerCallbacks(
std::unique_lock<std::recursive_mutex> global_lock_locked_once,
uint32_t virtual_address, uint32_t length, bool is_write,
bool unwatch_exact_range, bool unprotect = true);
uint32_t GetPhysicalAddress(uint32_t address) const;
2015-05-16 07:23:13 +00:00
protected:
VirtualHeap* parent_heap_;
uint32_t system_page_size_;
uint32_t system_page_count_;
struct SystemPageFlagsBlock {
// Whether writing to each page should result trigger invalidation
// callbacks.
uint64_t notify_on_invalidation;
};
// Protected by global_critical_region. Flags for each 64 system pages,
// interleaved as blocks, so bit scan can be used to quickly extract ranges.
std::vector<SystemPageFlagsBlock> system_page_flags_;
};
2015-12-03 01:37:48 +00:00
// Models the entire guest memory system on the console.
// This exposes interfaces to both virtual and physical memory and a TLB and
// page table for allocation, mapping, and protection.
//
// The memory is backed by a memory mapped file and is placed at a stable
// fixed address in the host address space (like 0x100000000). This allows
// efficient guest<->host address translations as well as easy sharing of the
// memory across various subsystems.
//
// The guest memory address space is split into several ranges that have varying
// properties such as page sizes, caching strategies, protections, and
// overlap with other ranges. Each range is represented by a BaseHeap of either
// VirtualHeap or PhysicalHeap depending on type. Heaps model the page tables
// and can handle reservation and committing of requested pages.
class Memory {
public:
Memory();
~Memory();
2015-12-03 01:37:48 +00:00
// Initializes the memory system.
// This may fail if the host address space could not be reserved or the
// mapping to the file system fails.
bool Initialize();
2015-12-27 20:03:30 +00:00
// Resets all memory to zero and resets all allocations.
void Reset();
2015-12-03 01:37:48 +00:00
// Full file name and path of the memory-mapped file backing all memory.
const std::filesystem::path& file_name() const { return file_name_; }
2015-05-24 07:02:47 +00:00
2015-12-03 01:37:48 +00:00
// Base address of virtual memory in the host address space.
// This is often something like 0x100000000.
inline uint8_t* virtual_membase() const { return virtual_membase_; }
2015-12-03 01:37:48 +00:00
// Translates a guest virtual address to a host address that can be accessed
// as a normal pointer.
// Note that the contents at the specified host address are big-endian.
template <typename T = uint8_t*>
inline T TranslateVirtual(uint32_t guest_address) const {
uint8_t* host_address = virtual_membase_ + guest_address;
const auto heap = LookupHeap(guest_address);
if (heap) {
host_address += heap->host_address_offset();
}
return reinterpret_cast<T>(host_address);
2015-08-07 03:17:01 +00:00
}
2015-12-03 01:37:48 +00:00
// Base address of physical memory in the host address space.
// This is often something like 0x200000000.
inline uint8_t* physical_membase() const { return physical_membase_; }
2015-12-03 01:37:48 +00:00
// Translates a guest physical address to a host address that can be accessed
// as a normal pointer.
// Note that the contents at the specified host address are big-endian.
template <typename T = uint8_t*>
inline T TranslatePhysical(uint32_t guest_address) const {
return reinterpret_cast<T>(physical_membase_ +
(guest_address & 0x1FFFFFFF));
}
// Translates a host address to a guest virtual address.
// Note that the contents at the returned host address are big-endian.
uint32_t HostToGuestVirtual(const void* host_address) const;
// Returns the guest physical address for the guest virtual address, or
// UINT32_MAX if it can't be obtained.
uint32_t GetPhysicalAddress(uint32_t address) const;
2015-12-03 01:37:48 +00:00
// Zeros out a range of memory at the given guest address.
void Zero(uint32_t address, uint32_t size);
2015-12-03 01:37:48 +00:00
// Fills a range of guest memory with the given byte value.
void Fill(uint32_t address, uint32_t size, uint8_t value);
2015-12-03 01:37:48 +00:00
// Copies a non-overlapping range of guest memory (like a memcpy).
void Copy(uint32_t dest, uint32_t src, uint32_t size);
2015-12-03 01:37:48 +00:00
// Searches the given range of guest memory for a run of dword values in
// big-endian order.
uint32_t SearchAligned(uint32_t start, uint32_t end, const uint32_t* values,
size_t value_count);
2015-12-03 01:37:48 +00:00
// Defines a memory-mapped IO (MMIO) virtual address range that when accessed
// will trigger the specified read and write callbacks for dword read/writes.
2015-05-16 07:23:13 +00:00
bool AddVirtualMappedRange(uint32_t virtual_address, uint32_t mask,
uint32_t size, void* context,
cpu::MMIOReadCallback read_callback,
cpu::MMIOWriteCallback write_callback);
2015-12-03 01:37:48 +00:00
// Gets the defined MMIO range for the given virtual address, if any.
cpu::MMIORange* LookupVirtualMappedRange(uint32_t virtual_address);
// Physical memory access callbacks, two types of them.
//
// This is simple per-system-page protection without reference counting or
// stored ranges. Whenever a watched page is accessed, all callbacks for it
// are triggered. Also the only way to remove callbacks is to trigger them
// somehow. Since there are no references from pages to individual callbacks,
// there's no way to disable only a specific callback for a page. Also
// callbacks may be triggered spuriously, and handlers should properly ignore
// pages they don't care about.
//
// Once callbacks are triggered for a page, the page is not watched anymore
// until requested again later. It is, however, unwatched only in one guest
// view of physical memory (because different views may have different
// protection for the same memory) - but it's rare when the same memory is
// used with different guest page sizes, and it's okay to fire a callback more
// than once.
//
// Only accessing the guest virtual memory views of physical memory triggers
// callbacks - data providers, for instance, must write to the host physical
// heap directly, otherwise their threads may infinitely await themselves.
//
// - Invalidation notifications:
//
// Protecting from writing. One-shot callbacks for invalidation of various
// kinds of physical memory caches (such as the GPU copy of the memory).
//
// May be triggered for a single page (in case of a write access violation or
// when need to synchronize data given by data providers) or for multiple
// pages (like when memory is released, or explicitly to trigger callbacks
// when host-side code can't rely on regular access violations, like when
// accessing a file).
//
// Since granularity of callbacks is one single page, an invalidation
// notification handler must invalidate the all the data stored in the touched
// pages.
//
// Because large ranges (like whole framebuffers) may be written to and
// exceptions are expensive, it's better to unprotect multiple pages as a
// result of a write access violation, so the shortest common range returned
// by all the invalidation callbacks (clamped to a sane range and also not to
// touch pages with provider callbacks) is unprotected.
//
// - Data providers:
//
// TODO(Triang3l): Implement data providers - more complicated because they
// will need to be able to release the global lock.
// Returns start and length of the smallest physical memory region surrounding
// the watched region that can be safely unwatched, if it doesn't matter,
// return (0, UINT32_MAX).
typedef std::pair<uint32_t, uint32_t> (*PhysicalMemoryInvalidationCallback)(
void* context_ptr, uint32_t physical_address_start, uint32_t length,
bool exact_range);
// Returns a handle for unregistering or for skipping one notification handler
// while triggering data providers.
void* RegisterPhysicalMemoryInvalidationCallback(
PhysicalMemoryInvalidationCallback callback, void* callback_context);
// Unregisters a physical memory invalidation callback previously added with
// RegisterPhysicalMemoryInvalidationCallback.
void UnregisterPhysicalMemoryInvalidationCallback(void* callback_handle);
// Enables physical memory access callbacks for the specified memory range,
// snapped to system page boundaries.
void EnablePhysicalMemoryAccessCallbacks(
uint32_t physical_address, uint32_t length,
bool enable_invalidation_notifications, bool enable_data_providers);
// Forces triggering of watch callbacks for a virtual address range if pages
// are watched there and unwatching them. Returns whether any page was
// watched. Must be called with global critical region locking depth of 1.
// TODO(Triang3l): Implement data providers - this is why locking depth of 1
// will be required in the future.
bool TriggerPhysicalMemoryCallbacks(
std::unique_lock<std::recursive_mutex> global_lock_locked_once,
uint32_t virtual_address, uint32_t length, bool is_write,
bool unwatch_exact_range, bool unprotect = true);
2015-12-03 01:37:48 +00:00
// Allocates virtual memory from the 'system' heap.
// System memory is kept separate from game memory but is still accessible
// using normal guest virtual addresses. Kernel structures and other internal
// 'system' allocations should come from this heap when possible.
uint32_t SystemHeapAlloc(uint32_t size, uint32_t alignment = 0x20,
uint32_t system_heap_flags = kSystemHeapDefault);
2015-12-03 01:37:48 +00:00
// Frees memory allocated with SystemHeapAlloc.
void SystemHeapFree(uint32_t address);
2015-12-03 01:37:48 +00:00
// Gets the heap for the address space containing the given address.
const BaseHeap* LookupHeap(uint32_t address) const;
inline BaseHeap* LookupHeap(uint32_t address) {
return const_cast<BaseHeap*>(
const_cast<const Memory*>(this)->LookupHeap(address));
}
2015-12-03 01:37:48 +00:00
// Gets the heap with the given properties.
2015-05-16 07:23:13 +00:00
BaseHeap* LookupHeapByType(bool physical, uint32_t page_size);
2014-01-05 19:19:02 +00:00
2018-02-10 22:45:06 +00:00
// Gets the physical base heap.
VirtualHeap* GetPhysicalHeap();
2015-12-03 01:37:48 +00:00
// Dumps a map of all allocated memory to the log.
2015-05-16 07:23:13 +00:00
void DumpMap();
2015-12-01 23:26:55 +00:00
bool Save(ByteStream* stream);
bool Restore(ByteStream* stream);
private:
int MapViews(uint8_t* mapping_base);
void UnmapViews();
static uint32_t HostToGuestVirtualThunk(const void* context,
const void* host_address);
bool AccessViolationCallback(
std::unique_lock<std::recursive_mutex> global_lock_locked_once,
void* host_address, bool is_write);
static bool AccessViolationCallbackThunk(
std::unique_lock<std::recursive_mutex> global_lock_locked_once,
void* context, void* host_address, bool is_write);
std::filesystem::path file_name_;
uint32_t system_page_size_ = 0;
uint32_t system_allocation_granularity_ = 0;
uint8_t* virtual_membase_ = nullptr;
uint8_t* physical_membase_ = nullptr;
xe::memory::FileMappingHandle mapping_ =
xe::memory::kFileMappingHandleInvalid;
uint8_t* mapping_base_ = nullptr;
union {
struct {
uint8_t* v00000000;
uint8_t* v40000000;
uint8_t* v7F000000;
uint8_t* v80000000;
2014-09-14 02:32:37 +00:00
uint8_t* v90000000;
uint8_t* vA0000000;
uint8_t* vC0000000;
uint8_t* vE0000000;
2015-05-16 07:23:13 +00:00
uint8_t* physical;
};
uint8_t* all_views[9];
2015-07-16 06:26:58 +00:00
} views_ = {{0}};
std::unique_ptr<cpu::MMIOHandler> mmio_handler_;
2015-05-16 07:23:13 +00:00
struct {
VirtualHeap v00000000;
VirtualHeap v40000000;
VirtualHeap v80000000;
VirtualHeap v90000000;
VirtualHeap physical;
PhysicalHeap vA0000000;
PhysicalHeap vC0000000;
PhysicalHeap vE0000000;
} heaps_;
2015-05-16 07:23:13 +00:00
friend class BaseHeap;
friend class PhysicalHeap;
xe::global_critical_region global_critical_region_;
std::vector<std::pair<PhysicalMemoryInvalidationCallback, void*>*>
physical_memory_invalidation_callbacks_;
};
} // namespace xe
#endif // XENIA_MEMORY_H_