mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-22 06:36:10 +00:00
169 lines
6.3 KiB
C++
169 lines
6.3 KiB
C++
/*
|
|
* Copyright (c) Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
#include "dmnt2_hardware_watchpoint.hpp"
|
|
#include "dmnt2_hardware_breakpoint.hpp"
|
|
#include "dmnt2_debug_process.hpp"
|
|
#include "dmnt2_debug_log.hpp"
|
|
|
|
namespace ams::dmnt {
|
|
|
|
namespace {
|
|
|
|
Result SetDataBreakPoint(svc::HardwareBreakPointRegisterName reg, svc::HardwareBreakPointRegisterName ctx, u64 address, u64 size, bool read, bool write) {
|
|
/* Determine lsc. */
|
|
const u8 lsc = (read ? 1 : 0) | (write ? 2 : 0);
|
|
|
|
/* Check that the watchpoint is valid. */
|
|
if (lsc != 0) {
|
|
R_UNLESS(HardwareWatchPointManager::IsValidWatchPoint(address, size), svc::ResultInvalidArgument());
|
|
}
|
|
|
|
/* Determine bas/mask. */
|
|
u8 bas = 0, mask = 0;
|
|
if (size <= 8) {
|
|
bas = ((1 << size) - 1) << (address & 7);
|
|
address = util::AlignDown(address, 8);
|
|
} else {
|
|
bas = 0xFF;
|
|
mask = util::PopCount(size - 1);
|
|
}
|
|
|
|
/* Build dbgbcr value. */
|
|
const u64 dbgbcr = (mask << 24) | (ctx << 16) | (bas << 5) | (lsc << 3) | ((lsc != 0) ? 1 : 0);
|
|
|
|
/* Set the breakpoint. */
|
|
const Result result = HardwareBreakPointManager::SetHardwareBreakPoint(reg, dbgbcr, address);
|
|
if (R_FAILED(result)) {
|
|
AMS_DMNT2_GDB_LOG_ERROR("SetDataBreakPoint FAIL 0x%08x, reg=%d, address=0x%lx\n", result.GetValue(), reg, address);
|
|
}
|
|
R_RETURN(result);
|
|
}
|
|
|
|
}
|
|
|
|
bool HardwareWatchPointManager::IsValidWatchPoint(u64 address, u64 size) {
|
|
/* Check size. */
|
|
if (size == 0) {
|
|
AMS_DMNT2_GDB_LOG_ERROR("HardwareWatchPointManager::IsValidWatchPoint(%lx, %lx) FAIL size == 0\n", address, size);
|
|
return false;
|
|
}
|
|
|
|
/* Validate. */
|
|
if (size <= 8) {
|
|
/* Check that address is aligned. */
|
|
if (util::AlignDown(address, 8) != util::AlignDown(address + size - 1, 8)) {
|
|
AMS_DMNT2_GDB_LOG_ERROR("HardwareWatchPointManager::IsValidWatchPoint(%lx, %lx) FAIL range crosses qword boundary\n", address, size);
|
|
return false;
|
|
}
|
|
} else {
|
|
/* Check size is small enough. */
|
|
if (size > 0x80000000) {
|
|
AMS_DMNT2_GDB_LOG_ERROR("HardwareWatchPointManager::IsValidWatchPoint(%lx, %lx) FAIL size too big\n", address, size);
|
|
return false;
|
|
}
|
|
|
|
/* Check size is power of two. */
|
|
if (!util::IsPowerOfTwo(size)) {
|
|
AMS_DMNT2_GDB_LOG_ERROR("HardwareWatchPointManager::IsValidWatchPoint(%lx, %lx) FAIL size not power of two\n", address, size);
|
|
return false;
|
|
}
|
|
|
|
/* Check alignment. */
|
|
if (!util::IsAligned(address, size)) {
|
|
AMS_DMNT2_GDB_LOG_ERROR("HardwareWatchPointManager::IsValidWatchPoint(%lx, %lx) FAIL address not size-aligned\n", address, size);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
Result WatchPoint::Clear(DebugProcess *debug_process) {
|
|
AMS_UNUSED(debug_process);
|
|
|
|
Result result = svc::ResultInvalidArgument();
|
|
if (m_in_use) {
|
|
AMS_DMNT2_GDB_LOG_DEBUG("WatchPoint::Clear %p 0x%lx\n", this, m_address);
|
|
result = SetDataBreakPoint(m_reg, m_ctx, 0, 0, false, false);
|
|
this->Reset();
|
|
}
|
|
R_RETURN(result);
|
|
}
|
|
|
|
Result WatchPoint::Set(DebugProcess *debug_process, uintptr_t address, size_t size, bool read, bool write) {
|
|
/* Set fields. */
|
|
m_address = address;
|
|
m_size = size;
|
|
m_read = read;
|
|
m_write = write;
|
|
|
|
/* Set context breakpoint. */
|
|
R_TRY(HardwareBreakPointManager::SetContextBreakPoint(m_ctx, debug_process));
|
|
|
|
/* Set watchpoint. */
|
|
R_TRY(SetDataBreakPoint(m_reg, m_ctx, address, size, read, write));
|
|
|
|
/* Set as in-use. */
|
|
m_in_use = true;
|
|
R_SUCCEED();
|
|
}
|
|
|
|
HardwareWatchPointManager::HardwareWatchPointManager(DebugProcess *debug_process) : BreakPointManagerBase(debug_process) {
|
|
const svc::HardwareBreakPointRegisterName ctx = HardwareBreakPointManager::GetWatchPointContextRegister();
|
|
for (size_t i = 0; i < util::size(m_breakpoints); ++i) {
|
|
m_breakpoints[i].Initialize(static_cast<svc::HardwareBreakPointRegisterName>(svc::HardwareBreakPointRegisterName_D0 + i), ctx);
|
|
}
|
|
}
|
|
|
|
BreakPointBase *HardwareWatchPointManager::GetBreakPoint(size_t index) {
|
|
if (index < util::size(m_breakpoints)) {
|
|
return m_breakpoints + index;
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
Result HardwareWatchPointManager::SetWatchPoint(u64 address, u64 size, bool read, bool write) {
|
|
/* Get a free watchpoint. */
|
|
auto *bp = static_cast<WatchPoint *>(this->GetFreeBreakPoint());
|
|
R_UNLESS(bp != nullptr, svc::ResultOutOfHandles());
|
|
|
|
/* Set the watchpoint. */
|
|
R_RETURN(bp->Set(m_debug_process, address, size, read, write));
|
|
}
|
|
|
|
Result HardwareWatchPointManager::GetWatchPointInfo(u64 address, bool &read, bool &write) {
|
|
/* Find a matching watchpoint. */
|
|
for (const auto &bp : m_breakpoints) {
|
|
if (bp.m_in_use) {
|
|
if (bp.m_address <= address && address < bp.m_address + bp.m_size) {
|
|
read = bp.m_read;
|
|
write = bp.m_write;
|
|
R_SUCCEED();
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Otherwise, we failed. */
|
|
AMS_DMNT2_GDB_LOG_ERROR("HardwareWatchPointManager::GetWatchPointInfo FAIL 0x%lx\n", address);
|
|
read = false;
|
|
write = false;
|
|
|
|
R_THROW(svc::ResultInvalidArgument());
|
|
}
|
|
|
|
}
|