Browse Source
Merge pull request #2533 from DarkLordZach/memory-frozen
Merge pull request #2533 from DarkLordZach/memory-frozen
memory: Add class to manage and enforce memory freezingpull/15/merge
committed by
GitHub
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 274 additions and 0 deletions
-
2src/core/CMakeLists.txt
-
2src/core/core.cpp
-
188src/core/tools/freezer.cpp
-
82src/core/tools/freezer.h
@ -0,0 +1,188 @@ |
|||||
|
// Copyright 2019 yuzu Emulator Project
|
||||
|
// Licensed under GPLv2 or any later version
|
||||
|
// Refer to the license.txt file included.
|
||||
|
|
||||
|
#include "common/assert.h"
|
||||
|
#include "common/logging/log.h"
|
||||
|
#include "core/core.h"
|
||||
|
#include "core/core_timing.h"
|
||||
|
#include "core/core_timing_util.h"
|
||||
|
#include "core/memory.h"
|
||||
|
#include "core/tools/freezer.h"
|
||||
|
|
||||
|
namespace Tools { |
||||
|
|
||||
|
namespace { |
||||
|
|
||||
|
constexpr s64 MEMORY_FREEZER_TICKS = static_cast<s64>(Core::Timing::BASE_CLOCK_RATE / 60); |
||||
|
|
||||
|
u64 MemoryReadWidth(u32 width, VAddr addr) { |
||||
|
switch (width) { |
||||
|
case 1: |
||||
|
return Memory::Read8(addr); |
||||
|
case 2: |
||||
|
return Memory::Read16(addr); |
||||
|
case 4: |
||||
|
return Memory::Read32(addr); |
||||
|
case 8: |
||||
|
return Memory::Read64(addr); |
||||
|
default: |
||||
|
UNREACHABLE(); |
||||
|
return 0; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void MemoryWriteWidth(u32 width, VAddr addr, u64 value) { |
||||
|
switch (width) { |
||||
|
case 1: |
||||
|
Memory::Write8(addr, static_cast<u8>(value)); |
||||
|
break; |
||||
|
case 2: |
||||
|
Memory::Write16(addr, static_cast<u16>(value)); |
||||
|
break; |
||||
|
case 4: |
||||
|
Memory::Write32(addr, static_cast<u32>(value)); |
||||
|
break; |
||||
|
case 8: |
||||
|
Memory::Write64(addr, value); |
||||
|
break; |
||||
|
default: |
||||
|
UNREACHABLE(); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
} // Anonymous namespace
|
||||
|
|
||||
|
Freezer::Freezer(Core::Timing::CoreTiming& core_timing) : core_timing(core_timing) { |
||||
|
event = core_timing.RegisterEvent( |
||||
|
"MemoryFreezer::FrameCallback", |
||||
|
[this](u64 userdata, s64 cycles_late) { FrameCallback(userdata, cycles_late); }); |
||||
|
core_timing.ScheduleEvent(MEMORY_FREEZER_TICKS, event); |
||||
|
} |
||||
|
|
||||
|
Freezer::~Freezer() { |
||||
|
core_timing.UnscheduleEvent(event, 0); |
||||
|
} |
||||
|
|
||||
|
void Freezer::SetActive(bool active) { |
||||
|
if (!this->active.exchange(active)) { |
||||
|
FillEntryReads(); |
||||
|
core_timing.ScheduleEvent(MEMORY_FREEZER_TICKS, event); |
||||
|
LOG_DEBUG(Common_Memory, "Memory freezer activated!"); |
||||
|
} else { |
||||
|
LOG_DEBUG(Common_Memory, "Memory freezer deactivated!"); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
bool Freezer::IsActive() const { |
||||
|
return active.load(std::memory_order_relaxed); |
||||
|
} |
||||
|
|
||||
|
void Freezer::Clear() { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
LOG_DEBUG(Common_Memory, "Clearing all frozen memory values."); |
||||
|
|
||||
|
entries.clear(); |
||||
|
} |
||||
|
|
||||
|
u64 Freezer::Freeze(VAddr address, u32 width) { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
const auto current_value = MemoryReadWidth(width, address); |
||||
|
entries.push_back({address, width, current_value}); |
||||
|
|
||||
|
LOG_DEBUG(Common_Memory, |
||||
|
"Freezing memory for address={:016X}, width={:02X}, current_value={:016X}", address, |
||||
|
width, current_value); |
||||
|
|
||||
|
return current_value; |
||||
|
} |
||||
|
|
||||
|
void Freezer::Unfreeze(VAddr address) { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
LOG_DEBUG(Common_Memory, "Unfreezing memory for address={:016X}", address); |
||||
|
|
||||
|
entries.erase( |
||||
|
std::remove_if(entries.begin(), entries.end(), |
||||
|
[&address](const Entry& entry) { return entry.address == address; }), |
||||
|
entries.end()); |
||||
|
} |
||||
|
|
||||
|
bool Freezer::IsFrozen(VAddr address) const { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
return std::find_if(entries.begin(), entries.end(), [&address](const Entry& entry) { |
||||
|
return entry.address == address; |
||||
|
}) != entries.end(); |
||||
|
} |
||||
|
|
||||
|
void Freezer::SetFrozenValue(VAddr address, u64 value) { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
const auto iter = std::find_if(entries.begin(), entries.end(), [&address](const Entry& entry) { |
||||
|
return entry.address == address; |
||||
|
}); |
||||
|
|
||||
|
if (iter == entries.end()) { |
||||
|
LOG_ERROR(Common_Memory, |
||||
|
"Tried to set freeze value for address={:016X} that is not frozen!", address); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
LOG_DEBUG(Common_Memory, |
||||
|
"Manually overridden freeze value for address={:016X}, width={:02X} to value={:016X}", |
||||
|
iter->address, iter->width, value); |
||||
|
iter->value = value; |
||||
|
} |
||||
|
|
||||
|
std::optional<Freezer::Entry> Freezer::GetEntry(VAddr address) const { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
const auto iter = std::find_if(entries.begin(), entries.end(), [&address](const Entry& entry) { |
||||
|
return entry.address == address; |
||||
|
}); |
||||
|
|
||||
|
if (iter == entries.end()) { |
||||
|
return std::nullopt; |
||||
|
} |
||||
|
|
||||
|
return *iter; |
||||
|
} |
||||
|
|
||||
|
std::vector<Freezer::Entry> Freezer::GetEntries() const { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
return entries; |
||||
|
} |
||||
|
|
||||
|
void Freezer::FrameCallback(u64 userdata, s64 cycles_late) { |
||||
|
if (!IsActive()) { |
||||
|
LOG_DEBUG(Common_Memory, "Memory freezer has been deactivated, ending callback events."); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
for (const auto& entry : entries) { |
||||
|
LOG_DEBUG(Common_Memory, |
||||
|
"Enforcing memory freeze at address={:016X}, value={:016X}, width={:02X}", |
||||
|
entry.address, entry.value, entry.width); |
||||
|
MemoryWriteWidth(entry.width, entry.address, entry.value); |
||||
|
} |
||||
|
|
||||
|
core_timing.ScheduleEvent(MEMORY_FREEZER_TICKS - cycles_late, event); |
||||
|
} |
||||
|
|
||||
|
void Freezer::FillEntryReads() { |
||||
|
std::lock_guard lock{entries_mutex}; |
||||
|
|
||||
|
LOG_DEBUG(Common_Memory, "Updating memory freeze entries to current values."); |
||||
|
|
||||
|
for (auto& entry : entries) { |
||||
|
entry.value = MemoryReadWidth(entry.width, entry.address); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
} // namespace Tools
|
||||
@ -0,0 +1,82 @@ |
|||||
|
// Copyright 2019 yuzu Emulator Project |
||||
|
// Licensed under GPLv2 or any later version |
||||
|
// Refer to the license.txt file included. |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include <atomic> |
||||
|
#include <mutex> |
||||
|
#include <optional> |
||||
|
#include <vector> |
||||
|
#include "common/common_types.h" |
||||
|
|
||||
|
namespace Core::Timing { |
||||
|
class CoreTiming; |
||||
|
struct EventType; |
||||
|
} // namespace Core::Timing |
||||
|
|
||||
|
namespace Tools { |
||||
|
|
||||
|
/** |
||||
|
* This class allows the user to prevent an application from writing new values to certain memory |
||||
|
* locations. This has a variety of uses when attempting to reverse a game. |
||||
|
* |
||||
|
* One example could be a cheat to prevent Mario from taking damage in SMO. One could freeze the |
||||
|
* memory address that the game uses to store Mario's health so when he takes damage (and the game |
||||
|
* tries to write the new health value to memory), the value won't change. |
||||
|
*/ |
||||
|
class Freezer { |
||||
|
public: |
||||
|
struct Entry { |
||||
|
VAddr address; |
||||
|
u32 width; |
||||
|
u64 value; |
||||
|
}; |
||||
|
|
||||
|
explicit Freezer(Core::Timing::CoreTiming& core_timing); |
||||
|
~Freezer(); |
||||
|
|
||||
|
// Enables or disables the entire memory freezer. |
||||
|
void SetActive(bool active); |
||||
|
|
||||
|
// Returns whether or not the freezer is active. |
||||
|
bool IsActive() const; |
||||
|
|
||||
|
// Removes all entries from the freezer. |
||||
|
void Clear(); |
||||
|
|
||||
|
// Freezes a value to its current memory address. The value the memory is kept at will be the |
||||
|
// value that is read during this function. Width can be 1, 2, 4, or 8 (in bytes). |
||||
|
u64 Freeze(VAddr address, u32 width); |
||||
|
|
||||
|
// Unfreezes the memory value at address. If the address isn't frozen, this is a no-op. |
||||
|
void Unfreeze(VAddr address); |
||||
|
|
||||
|
// Returns whether or not the address is frozen. |
||||
|
bool IsFrozen(VAddr address) const; |
||||
|
|
||||
|
// Sets the value that address should be frozen to. This doesn't change the width set by using |
||||
|
// Freeze(). If the value isn't frozen, this will not freeze it and is thus a no-op. |
||||
|
void SetFrozenValue(VAddr address, u64 value); |
||||
|
|
||||
|
// Returns the entry corresponding to the address if the address is frozen, otherwise |
||||
|
// std::nullopt. |
||||
|
std::optional<Entry> GetEntry(VAddr address) const; |
||||
|
|
||||
|
// Returns all the entries in the freezer, an empty vector means nothing is frozen. |
||||
|
std::vector<Entry> GetEntries() const; |
||||
|
|
||||
|
private: |
||||
|
void FrameCallback(u64 userdata, s64 cycles_late); |
||||
|
void FillEntryReads(); |
||||
|
|
||||
|
std::atomic_bool active{false}; |
||||
|
|
||||
|
mutable std::mutex entries_mutex; |
||||
|
std::vector<Entry> entries; |
||||
|
|
||||
|
Core::Timing::EventType* event; |
||||
|
Core::Timing::CoreTiming& core_timing; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Tools |
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue