mirror of
https://github.com/momo5502/emulator.git
synced 2026-01-19 03:33:56 +00:00
193 lines
5.9 KiB
C++
193 lines
5.9 KiB
C++
#include "std_include.hpp"
|
|
#include "kusd_mmio.hpp"
|
|
#include <utils/time.hpp>
|
|
#include "windows_emulator.hpp"
|
|
|
|
#include <address_utils.hpp>
|
|
|
|
constexpr auto KUSD_ADDRESS = 0x7ffe0000ULL;
|
|
constexpr auto KUSD_SIZE = sizeof(KUSER_SHARED_DATA64);
|
|
constexpr auto KUSD_BUFFER_SIZE = page_align_up(KUSD_SIZE);
|
|
|
|
namespace
|
|
{
|
|
void setup_kusd(KUSER_SHARED_DATA64& kusd)
|
|
{
|
|
memset(reinterpret_cast<void*>(&kusd), 0, sizeof(kusd));
|
|
|
|
kusd.TickCountMultiplier = 0x0fa00000;
|
|
kusd.InterruptTime.LowPart = 0x17bd9547;
|
|
kusd.InterruptTime.High1Time = 0x0000004b;
|
|
kusd.InterruptTime.High2Time = 0x0000004b;
|
|
kusd.SystemTime.LowPart = 0x7af9da99;
|
|
kusd.SystemTime.High1Time = 0x01db27b9;
|
|
kusd.SystemTime.High2Time = 0x01db27b9;
|
|
kusd.TimeZoneBias.LowPart = 0x3c773000;
|
|
kusd.TimeZoneBias.High1Time = -17;
|
|
kusd.TimeZoneBias.High2Time = -17;
|
|
kusd.TimeZoneId = 0x00000002;
|
|
kusd.LargePageMinimum = 0x00200000;
|
|
kusd.RNGSeedVersion = 0;
|
|
kusd.TimeZoneBiasStamp = 0x00000004;
|
|
kusd.NtBuildNumber = 19045;
|
|
kusd.NtProductType = NtProductWinNt;
|
|
kusd.ProductTypeIsValid = 0x01;
|
|
kusd.NativeProcessorArchitecture = 0x0009;
|
|
kusd.NtMajorVersion = 0x0000000a;
|
|
kusd.BootId = 0;
|
|
kusd.SystemExpirationDate.QuadPart = 0;
|
|
kusd.SuiteMask = 0;
|
|
kusd.MitigationPolicies.MitigationPolicies = 0x0a;
|
|
kusd.MitigationPolicies.NXSupportPolicy = 0x02;
|
|
kusd.MitigationPolicies.SEHValidationPolicy = 0x02;
|
|
kusd.CyclesPerYield = 0x0064;
|
|
kusd.DismountCount = 0x00000006;
|
|
kusd.ComPlusPackage = 0x00000001;
|
|
kusd.LastSystemRITEventTickCount = 0x01ec1fd3;
|
|
kusd.NumberOfPhysicalPages = 0x00bf0958;
|
|
kusd.FullNumberOfPhysicalPages = 0x0000000000bf0958;
|
|
kusd.TickCount.TickCount.LowPart = 0x001f7f05;
|
|
kusd.TickCount.TickCountQuad = 0x00000000001f7f05;
|
|
kusd.Cookie = 0;
|
|
kusd.ConsoleSessionForegroundProcessId = 0x00000000000028f4;
|
|
kusd.TimeUpdateLock = 0x0000000002b28586;
|
|
kusd.BaselineSystemTimeQpc = 0x0000004b17cd596c;
|
|
kusd.BaselineInterruptTimeQpc = 0x0000004b17cd596c;
|
|
kusd.QpcSystemTimeIncrement = 0x8000000000000000;
|
|
kusd.QpcInterruptTimeIncrement = 0x8000000000000000;
|
|
kusd.QpcSystemTimeIncrementShift = 0x01;
|
|
kusd.QpcInterruptTimeIncrementShift = 0x01;
|
|
kusd.UnparkedProcessorCount = 0x000c;
|
|
kusd.TelemetryCoverageRound = 0x00000001;
|
|
kusd.LangGenerationCount = 0x00000003;
|
|
kusd.InterruptTimeBias = 0x00000015a5d56406;
|
|
kusd.ActiveProcessorCount = 0x00000004;
|
|
kusd.ActiveGroupCount = 0x01;
|
|
kusd.TimeZoneBiasEffectiveStart.QuadPart = 0x01db276e654cb2ff;
|
|
kusd.TimeZoneBiasEffectiveEnd.QuadPart = 0x01db280b8c3b2800;
|
|
kusd.XState.EnabledFeatures = 0x000000000000001f;
|
|
kusd.XState.EnabledVolatileFeatures = 0x000000000000000f;
|
|
kusd.XState.Size = 0x000003c0;
|
|
kusd.QpcData.QpcData = 0x0083;
|
|
kusd.QpcData.QpcBypassEnabled = 0x83;
|
|
kusd.QpcBias = 0x000000159530c4af;
|
|
kusd.QpcFrequency = utils::clock::steady_duration::period::den;
|
|
kusd.Reserved1 = 0x7ffeffff;
|
|
kusd.Reserved3 = 0x80000000;
|
|
kusd.ProcessorFeatures.arr[PF_RDTSC_INSTRUCTION_AVAILABLE] = 1;
|
|
kusd.ProcessorFeatures.arr[PF_RDTSCP_INSTRUCTION_AVAILABLE] = 1;
|
|
kusd.ProcessorFeatures.arr[PF_RDPID_INSTRUCTION_AVAILABLE] = 0;
|
|
|
|
constexpr std::u16string_view root_dir{u"C:\\Windows"};
|
|
memcpy(&kusd.NtSystemRoot.arr[0], root_dir.data(), root_dir.size() * 2);
|
|
|
|
kusd.ImageNumberLow = IMAGE_FILE_MACHINE_AMD64;
|
|
kusd.ImageNumberHigh = IMAGE_FILE_MACHINE_AMD64;
|
|
}
|
|
}
|
|
|
|
namespace utils
|
|
{
|
|
static void serialize(buffer_serializer& buffer, const KUSER_SHARED_DATA64& kusd)
|
|
{
|
|
static_assert(KUSD_SIZE == sizeof(kusd));
|
|
buffer.write(&kusd, KUSD_SIZE);
|
|
}
|
|
|
|
static void deserialize(buffer_deserializer& buffer, KUSER_SHARED_DATA64& kusd)
|
|
{
|
|
buffer.read(&kusd, KUSD_SIZE);
|
|
}
|
|
}
|
|
|
|
kusd_mmio::kusd_mmio(memory_manager& memory, utils::clock& clock)
|
|
: memory_(&memory),
|
|
clock_(&clock)
|
|
{
|
|
}
|
|
|
|
kusd_mmio::~kusd_mmio()
|
|
{
|
|
this->deregister_mmio();
|
|
}
|
|
|
|
kusd_mmio::kusd_mmio(utils::buffer_deserializer& buffer)
|
|
: kusd_mmio(buffer.read<memory_manager_wrapper>(), buffer.read<clock_wrapper>())
|
|
{
|
|
}
|
|
|
|
void kusd_mmio::setup()
|
|
{
|
|
setup_kusd(this->kusd_);
|
|
this->register_mmio();
|
|
}
|
|
|
|
void kusd_mmio::serialize(utils::buffer_serializer& buffer) const
|
|
{
|
|
buffer.write(this->kusd_);
|
|
}
|
|
|
|
void kusd_mmio::deserialize(utils::buffer_deserializer& buffer)
|
|
{
|
|
buffer.read(this->kusd_);
|
|
|
|
this->deregister_mmio();
|
|
this->register_mmio();
|
|
}
|
|
|
|
void kusd_mmio::read(const uint64_t addr, void* data, const size_t size)
|
|
{
|
|
this->update();
|
|
|
|
if (addr >= KUSD_SIZE)
|
|
{
|
|
return;
|
|
}
|
|
|
|
const auto end = addr + size;
|
|
const auto valid_end = std::min(end, static_cast<uint64_t>(KUSD_SIZE));
|
|
const auto real_size = valid_end - addr;
|
|
|
|
const auto* kusd_buffer = reinterpret_cast<uint8_t*>(&this->kusd_);
|
|
memcpy(data, kusd_buffer + addr, static_cast<size_t>(real_size));
|
|
}
|
|
|
|
uint64_t kusd_mmio::address()
|
|
{
|
|
return KUSD_ADDRESS;
|
|
}
|
|
|
|
void kusd_mmio::update()
|
|
{
|
|
const auto time = this->clock_->system_now();
|
|
utils::convert_to_ksystem_time(&this->kusd_.SystemTime, time);
|
|
}
|
|
|
|
void kusd_mmio::register_mmio()
|
|
{
|
|
if (this->registered_)
|
|
{
|
|
return;
|
|
}
|
|
|
|
this->registered_ = true;
|
|
|
|
this->memory_->allocate_mmio(
|
|
KUSD_ADDRESS, KUSD_BUFFER_SIZE,
|
|
[this](const uint64_t addr, void* data, const size_t size) {
|
|
this->read(addr, data, size); //
|
|
},
|
|
[](const uint64_t, const void*, const size_t) {
|
|
// Writing not supported!
|
|
});
|
|
}
|
|
|
|
void kusd_mmio::deregister_mmio()
|
|
{
|
|
if (this->registered_)
|
|
{
|
|
this->registered_ = false;
|
|
this->memory_->release_memory(KUSD_ADDRESS, KUSD_BUFFER_SIZE);
|
|
}
|
|
}
|