2009-10-23 20:24:59 +00:00
|
|
|
/* PCSX2 - PS2 Emulator for PCs
|
2010-05-03 14:08:02 +00:00
|
|
|
* Copyright (C) 2002-2010 PCSX2 Dev Team
|
2009-10-23 20:24:59 +00:00
|
|
|
*
|
|
|
|
* PCSX2 is free software: you can redistribute it and/or modify it under the terms
|
|
|
|
* of the GNU Lesser General Public License as published by the Free Software Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* PCSX2 is distributed in the hope that 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 PCSX2.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2010-10-27 19:18:46 +00:00
|
|
|
// [TODO] Rename this file to VirtualMemory.h !!
|
|
|
|
|
2009-10-23 20:24:59 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
// =====================================================================================================
|
|
|
|
// Cross-Platform Memory Protection (Used by VTLB, Recompilers and Texture caches)
|
|
|
|
// =====================================================================================================
|
|
|
|
// Win32 platforms use the SEH model: __try {} __except {}
|
|
|
|
// Linux platforms use the POSIX Signals model: sigaction()
|
2010-10-27 19:18:46 +00:00
|
|
|
// [TODO] OS-X (Darwin) platforms should use the Mach exception model (not implemented)
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2010-10-27 19:18:46 +00:00
|
|
|
#include "EventSource.h"
|
2009-10-23 20:24:59 +00:00
|
|
|
|
|
|
|
struct PageFaultInfo
|
|
|
|
{
|
2016-11-12 15:28:37 +00:00
|
|
|
uptr addr;
|
2009-10-23 21:05:32 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
PageFaultInfo(uptr address)
|
|
|
|
{
|
|
|
|
addr = address;
|
|
|
|
}
|
2009-10-23 20:24:59 +00:00
|
|
|
};
|
|
|
|
|
2010-01-22 15:22:01 +00:00
|
|
|
// --------------------------------------------------------------------------------------
|
|
|
|
// IEventListener_PageFault
|
|
|
|
// --------------------------------------------------------------------------------------
|
|
|
|
class IEventListener_PageFault : public IEventDispatcher<PageFaultInfo>
|
2009-10-23 20:24:59 +00:00
|
|
|
{
|
|
|
|
public:
|
2016-11-12 15:28:37 +00:00
|
|
|
typedef PageFaultInfo EvtParams;
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2010-01-22 15:22:01 +00:00
|
|
|
public:
|
2017-05-06 12:53:02 +00:00
|
|
|
virtual ~IEventListener_PageFault() = default;
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
virtual void DispatchEvent(const PageFaultInfo &evtinfo, bool &handled)
|
|
|
|
{
|
|
|
|
OnPageFaultEvent(evtinfo, handled);
|
|
|
|
}
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
virtual void DispatchEvent(const PageFaultInfo &evtinfo)
|
|
|
|
{
|
|
|
|
pxFailRel("Don't call me, damnit. Use DispatchException instead.");
|
|
|
|
}
|
2010-01-22 15:22:01 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
virtual void OnPageFaultEvent(const PageFaultInfo &evtinfo, bool &handled) {}
|
2009-10-23 20:24:59 +00:00
|
|
|
};
|
|
|
|
|
2010-10-22 16:23:52 +00:00
|
|
|
// --------------------------------------------------------------------------------------
|
2010-11-15 14:05:02 +00:00
|
|
|
// EventListener_PageFault / EventListenerHelper_PageFault
|
2010-10-22 16:23:52 +00:00
|
|
|
// --------------------------------------------------------------------------------------
|
2010-01-23 17:13:03 +00:00
|
|
|
class EventListener_PageFault : public IEventListener_PageFault
|
|
|
|
{
|
|
|
|
public:
|
2016-11-12 15:28:37 +00:00
|
|
|
EventListener_PageFault();
|
2017-05-06 12:22:00 +00:00
|
|
|
virtual ~EventListener_PageFault();
|
2010-01-23 17:13:03 +00:00
|
|
|
};
|
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
template <typename TypeToDispatchTo>
|
2010-11-15 14:05:02 +00:00
|
|
|
class EventListenerHelper_PageFault : public EventListener_PageFault
|
|
|
|
{
|
|
|
|
public:
|
2016-11-12 15:28:37 +00:00
|
|
|
TypeToDispatchTo *Owner;
|
2010-11-15 14:05:02 +00:00
|
|
|
|
|
|
|
public:
|
2016-11-12 15:28:37 +00:00
|
|
|
EventListenerHelper_PageFault(TypeToDispatchTo &dispatchTo)
|
|
|
|
{
|
|
|
|
Owner = &dispatchTo;
|
|
|
|
}
|
2010-11-15 14:05:02 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
EventListenerHelper_PageFault(TypeToDispatchTo *dispatchTo)
|
|
|
|
{
|
|
|
|
Owner = dispatchTo;
|
|
|
|
}
|
2010-11-15 14:05:02 +00:00
|
|
|
|
2017-05-06 12:53:02 +00:00
|
|
|
virtual ~EventListenerHelper_PageFault() = default;
|
2010-11-15 14:05:02 +00:00
|
|
|
|
|
|
|
protected:
|
2016-11-12 15:28:37 +00:00
|
|
|
virtual void OnPageFaultEvent(const PageFaultInfo &info, bool &handled)
|
|
|
|
{
|
|
|
|
Owner->OnPageFaultEvent(info, handled);
|
|
|
|
}
|
2010-11-15 14:05:02 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// --------------------------------------------------------------------------------------
|
|
|
|
// SrcType_PageFault
|
|
|
|
// --------------------------------------------------------------------------------------
|
2010-01-22 15:22:01 +00:00
|
|
|
class SrcType_PageFault : public EventSource<IEventListener_PageFault>
|
|
|
|
{
|
|
|
|
protected:
|
2016-11-12 15:28:37 +00:00
|
|
|
typedef EventSource<IEventListener_PageFault> _parent;
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2010-01-22 15:22:01 +00:00
|
|
|
protected:
|
2016-11-12 15:28:37 +00:00
|
|
|
bool m_handled;
|
2010-01-22 15:22:01 +00:00
|
|
|
|
|
|
|
public:
|
2016-11-12 15:28:37 +00:00
|
|
|
SrcType_PageFault()
|
|
|
|
: m_handled(false)
|
|
|
|
{
|
|
|
|
}
|
2017-05-06 12:53:02 +00:00
|
|
|
virtual ~SrcType_PageFault() = default;
|
2010-01-22 15:22:01 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
bool WasHandled() const { return m_handled; }
|
|
|
|
virtual void Dispatch(const PageFaultInfo ¶ms);
|
2010-01-22 15:22:01 +00:00
|
|
|
|
|
|
|
protected:
|
2016-11-12 15:28:37 +00:00
|
|
|
virtual void _DispatchRaw(ListenerIterator iter, const ListenerIterator &iend, const PageFaultInfo &evt);
|
2010-01-22 15:22:01 +00:00
|
|
|
};
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2010-11-15 14:05:02 +00:00
|
|
|
|
2010-10-22 16:23:52 +00:00
|
|
|
// --------------------------------------------------------------------------------------
|
2010-11-15 14:05:02 +00:00
|
|
|
// VirtualMemoryReserve
|
2010-10-22 16:23:52 +00:00
|
|
|
// --------------------------------------------------------------------------------------
|
2010-11-15 14:05:02 +00:00
|
|
|
class VirtualMemoryReserve
|
2010-10-22 16:23:52 +00:00
|
|
|
{
|
2016-11-12 15:28:37 +00:00
|
|
|
DeclareNoncopyableObject(VirtualMemoryReserve);
|
2010-10-22 16:23:52 +00:00
|
|
|
|
|
|
|
protected:
|
2016-11-12 15:28:37 +00:00
|
|
|
wxString m_name;
|
|
|
|
|
|
|
|
// Default size of the reserve, in bytes. Can be specified when the object is constructed.
|
|
|
|
// Is used as the reserve size when Reserve() is called, unless an override is specified
|
|
|
|
// in the Reserve parameters.
|
|
|
|
size_t m_defsize;
|
2010-11-16 15:58:42 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
void *m_baseptr;
|
2010-11-15 14:05:02 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
// reserved memory (in pages).
|
|
|
|
uptr m_pages_reserved;
|
2010-10-22 16:23:52 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
// Records the number of pages committed to memory.
|
|
|
|
// (metric for analysis of buffer usage)
|
|
|
|
uptr m_pages_commited;
|
2010-10-22 16:23:52 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
// Protection mode to be applied to committed blocks.
|
|
|
|
PageProtectionMode m_prot_mode;
|
2010-10-22 16:23:52 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
// Controls write access to the entire reserve. When true (the default), the reserve
|
|
|
|
// operates normally. When set to false, all committed blocks are re-protected with
|
|
|
|
// write disabled, and accesses to uncommitted blocks (read or write) will cause a GPF
|
|
|
|
// as well.
|
|
|
|
bool m_allow_writes;
|
2010-12-30 06:21:07 +00:00
|
|
|
|
2010-10-22 16:23:52 +00:00
|
|
|
public:
|
2016-11-12 15:28:37 +00:00
|
|
|
VirtualMemoryReserve(const wxString &name = wxEmptyString, size_t size = 0);
|
2017-05-06 12:22:00 +00:00
|
|
|
virtual ~VirtualMemoryReserve()
|
2016-11-12 15:28:37 +00:00
|
|
|
{
|
|
|
|
Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void *Reserve(size_t size = 0, uptr base = 0, uptr upper_bounds = 0);
|
|
|
|
virtual void *ReserveAt(uptr base = 0, uptr upper_bounds = 0)
|
|
|
|
{
|
|
|
|
return Reserve(m_defsize, base, upper_bounds);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void Reset();
|
|
|
|
virtual void Release();
|
|
|
|
virtual bool TryResize(uint newsize);
|
|
|
|
virtual bool Commit();
|
|
|
|
|
|
|
|
virtual void ForbidModification();
|
|
|
|
virtual void AllowModification();
|
|
|
|
|
|
|
|
bool IsOk() const { return m_baseptr != NULL; }
|
|
|
|
wxString GetName() const { return m_name; }
|
|
|
|
|
|
|
|
uptr GetReserveSizeInBytes() const { return m_pages_reserved * __pagesize; }
|
|
|
|
uptr GetReserveSizeInPages() const { return m_pages_reserved; }
|
|
|
|
uint GetCommittedPageCount() const { return m_pages_commited; }
|
|
|
|
uint GetCommittedBytes() const { return m_pages_commited * __pagesize; }
|
|
|
|
|
|
|
|
u8 *GetPtr() { return (u8 *)m_baseptr; }
|
|
|
|
const u8 *GetPtr() const { return (u8 *)m_baseptr; }
|
|
|
|
u8 *GetPtrEnd() { return (u8 *)m_baseptr + (m_pages_reserved * __pagesize); }
|
|
|
|
const u8 *GetPtrEnd() const { return (u8 *)m_baseptr + (m_pages_reserved * __pagesize); }
|
|
|
|
|
|
|
|
VirtualMemoryReserve &SetName(const wxString &newname);
|
|
|
|
VirtualMemoryReserve &SetBaseAddr(uptr newaddr);
|
|
|
|
VirtualMemoryReserve &SetPageAccessOnCommit(const PageProtectionMode &mode);
|
|
|
|
|
|
|
|
operator void *() { return m_baseptr; }
|
|
|
|
operator const void *() const { return m_baseptr; }
|
|
|
|
|
|
|
|
operator u8 *() { return (u8 *)m_baseptr; }
|
|
|
|
operator const u8 *() const { return (u8 *)m_baseptr; }
|
|
|
|
|
|
|
|
u8 &operator[](uint idx)
|
|
|
|
{
|
|
|
|
pxAssert(idx < (m_pages_reserved * __pagesize));
|
|
|
|
return *((u8 *)m_baseptr + idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
const u8 &operator[](uint idx) const
|
|
|
|
{
|
|
|
|
pxAssert(idx < (m_pages_reserved * __pagesize));
|
|
|
|
return *((u8 *)m_baseptr + idx);
|
|
|
|
}
|
2010-11-06 16:25:40 +00:00
|
|
|
|
2010-12-30 06:21:07 +00:00
|
|
|
protected:
|
2016-11-12 15:28:37 +00:00
|
|
|
virtual void ReprotectCommittedBlocks(const PageProtectionMode &newmode);
|
2010-11-15 14:05:02 +00:00
|
|
|
};
|
|
|
|
|
2015-11-17 17:35:02 +00:00
|
|
|
#ifdef __POSIX__
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
#define PCSX2_PAGEFAULT_PROTECT
|
|
|
|
#define PCSX2_PAGEFAULT_EXCEPT
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
#elif defined(_WIN32)
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2010-01-22 15:22:01 +00:00
|
|
|
struct _EXCEPTION_POINTERS;
|
2016-11-12 15:28:37 +00:00
|
|
|
extern int SysPageFaultExceptionFilter(struct _EXCEPTION_POINTERS *eps);
|
2009-10-23 20:24:59 +00:00
|
|
|
|
2016-11-12 15:28:37 +00:00
|
|
|
#define PCSX2_PAGEFAULT_PROTECT __try
|
|
|
|
#define PCSX2_PAGEFAULT_EXCEPT \
|
|
|
|
__except (SysPageFaultExceptionFilter(GetExceptionInformation())) {}
|
2009-10-23 20:24:59 +00:00
|
|
|
|
|
|
|
#else
|
2016-11-12 15:28:37 +00:00
|
|
|
#error PCSX2 - Unsupported operating system platform.
|
2009-10-23 20:24:59 +00:00
|
|
|
#endif
|
2010-01-22 15:22:01 +00:00
|
|
|
|
2010-11-15 14:05:02 +00:00
|
|
|
extern void pxInstallSignalHandler();
|
|
|
|
extern void _platform_InstallSignalHandler();
|
2010-01-22 15:22:01 +00:00
|
|
|
|
2011-08-12 02:31:49 +00:00
|
|
|
#include "Threading.h"
|
2016-11-12 15:28:37 +00:00
|
|
|
extern SrcType_PageFault *Source_PageFault;
|
|
|
|
extern Threading::Mutex PageFault_Mutex;
|