mirror of
https://github.com/alliedmodders/metamod-source.git
synced 2024-12-01 13:24:25 +01:00
e19413dd5b
--HG-- rename : sourcemm/IPluginManager.h => core/IPluginManager.h rename : sourcemm/ISmmAPI.h => core/ISmmAPI.h rename : sourcemm/ISmmPlugin.h => core/ISmmPlugin.h rename : sourcemm/ISmmPluginExt.h => core/ISmmPluginExt.h rename : sourcemm/LICENSE.txt => core/LICENSE.txt rename : sourcemm/Makefile => core/Makefile rename : sourcemm/changelog.txt => core/changelog.txt rename : sourcemm/episode1/console.cpp => core/episode1/console.cpp rename : sourcemm/episode1/console.h => core/episode1/console.h rename : sourcemm/episode1/convar_smm.h => core/episode1/convar_smm.h rename : sourcemm/episode1/msvc8/sourcemm.sln => core/episode1/msvc8/sourcemm.sln rename : sourcemm/episode1/msvc8/sourcemm.vcproj => core/episode1/msvc8/sourcemm.vcproj rename : sourcemm/episode1/provider_ep1.cpp => core/episode1/provider_ep1.cpp rename : sourcemm/episode1/provider_ep1.h => core/episode1/provider_ep1.h rename : sourcemm/episode1/vsp_listener.cpp => core/episode1/vsp_listener.cpp rename : sourcemm/episode1/vsp_listener.h => core/episode1/vsp_listener.h rename : sourcemm/metamod.cpp => core/metamod.cpp rename : sourcemm/metamod.h => core/metamod.h rename : sourcemm/metamod_console.cpp => core/metamod_console.cpp rename : sourcemm/metamod_console.h => core/metamod_console.h rename : sourcemm/metamod_oslink.cpp => core/metamod_oslink.cpp rename : sourcemm/metamod_oslink.h => core/metamod_oslink.h rename : sourcemm/metamod_plugins.cpp => core/metamod_plugins.cpp rename : sourcemm/metamod_plugins.h => core/metamod_plugins.h rename : sourcemm/metamod_provider.h => core/metamod_provider.h rename : sourcemm/metamod_util.cpp => core/metamod_util.cpp rename : sourcemm/metamod_util.h => core/metamod_util.h rename : sourcemm/episode2/console.cpp => core/provider/console.cpp rename : sourcemm/episode2/console.h => core/provider/console.h rename : sourcemm/episode2/msvc8/sourcemm.sln => core/provider/msvc8/sourcemm.sln rename : sourcemm/episode2/msvc8/sourcemm.vcproj => core/provider/msvc8/sourcemm.vcproj rename : sourcemm/episode2/msvc9/sourcemm.sln => core/provider/msvc9/sourcemm.sln rename : sourcemm/episode2/msvc9/sourcemm.vcproj => core/provider/msvc9/sourcemm.vcproj rename : sourcemm/episode2/provider_ep2.cpp => core/provider/provider_ep2.cpp rename : sourcemm/episode2/provider_ep2.h => core/provider/provider_ep2.h rename : sourcemm/episode2/vsp_listener.cpp => core/provider/vsp_listener.cpp rename : sourcemm/episode2/vsp_listener.h => core/provider/vsp_listener.h rename : sourcehook/FastDelegate.h => core/sourcehook/FastDelegate.h rename : sourcehook/generate/FastDelegate.h => core/sourcehook/generate/FastDelegate.h rename : sourcehook/generate/FastDelegate.hxx => core/sourcehook/generate/FastDelegate.hxx rename : sourcehook/generate/generate => core/sourcehook/generate/generate rename : sourcehook/generate/generate.bat => core/sourcehook/generate/generate.bat rename : sourcehook/generate/sh_memfuncinfo.h => core/sourcehook/generate/sh_memfuncinfo.h rename : sourcehook/generate/sh_memfuncinfo.hxx => core/sourcehook/generate/sh_memfuncinfo.hxx rename : sourcehook/generate/shworker.bin => core/sourcehook/generate/shworker.bin rename : sourcehook/generate/shworker.exe => core/sourcehook/generate/shworker.exe rename : sourcehook/generate/shworker/Makefile => core/sourcehook/generate/shworker/Makefile rename : sourcehook/generate/shworker/fd_hopter.cpp => core/sourcehook/generate/shworker/fd_hopter.cpp rename : sourcehook/generate/shworker/msvc7/shworker.vcproj => core/sourcehook/generate/shworker/msvc7/shworker.vcproj rename : sourcehook/generate/shworker/msvc8/shworker.vcproj => core/sourcehook/generate/shworker/msvc8/shworker.vcproj rename : sourcehook/generate/shworker/shworker.cpp => core/sourcehook/generate/shworker/shworker.cpp rename : sourcehook/generate/sourcehook.h => core/sourcehook/generate/sourcehook.h rename : sourcehook/generate/sourcehook.hxx => core/sourcehook/generate/sourcehook.hxx rename : sourcehook/sh_list.h => core/sourcehook/sh_list.h rename : sourcehook/sh_memfuncinfo.h => core/sourcehook/sh_memfuncinfo.h rename : sourcehook/sh_memory.h => core/sourcehook/sh_memory.h rename : sourcehook/sh_pagealloc.h => core/sourcehook/sh_pagealloc.h rename : sourcehook/sh_stack.h => core/sourcehook/sh_stack.h rename : sourcehook/sh_string.h => core/sourcehook/sh_string.h rename : sourcehook/sh_tinyhash.h => core/sourcehook/sh_tinyhash.h rename : sourcehook/sh_vector.h => core/sourcehook/sh_vector.h rename : sourcehook/sourcehook.cpp => core/sourcehook/sourcehook.cpp rename : sourcehook/sourcehook.h => core/sourcehook/sourcehook.h rename : sourcehook/sourcehook_hookmangen.cpp => core/sourcehook/sourcehook_hookmangen.cpp rename : sourcehook/sourcehook_hookmangen.h => core/sourcehook/sourcehook_hookmangen.h rename : sourcehook/sourcehook_hookmangen_x86.h => core/sourcehook/sourcehook_hookmangen_x86.h rename : sourcehook/sourcehook_impl.h => core/sourcehook/sourcehook_impl.h rename : sourcehook/sourcehook_impl_chook.h => core/sourcehook/sourcehook_impl_chook.h rename : sourcehook/sourcehook_impl_chookidman.h => core/sourcehook/sourcehook_impl_chookidman.h rename : sourcehook/sourcehook_impl_chookmaninfo.h => core/sourcehook/sourcehook_impl_chookmaninfo.h rename : sourcehook/sourcehook_impl_ciface.h => core/sourcehook/sourcehook_impl_ciface.h rename : sourcehook/sourcehook_impl_cproto.h => core/sourcehook/sourcehook_impl_cproto.h rename : sourcehook/sourcehook_impl_cvfnptr.h => core/sourcehook/sourcehook_impl_cvfnptr.h rename : sourcehook/sourcehook_pibuilder.h => core/sourcehook/sourcehook_pibuilder.h rename : sourcehook/test/Makefile => core/sourcehook/test/Makefile rename : sourcehook/test/generate.bat => core/sourcehook/test/generate.bat rename : sourcehook/test/main.cpp => core/sourcehook/test/main.cpp rename : sourcehook/test/msvc7/test.vcproj => core/sourcehook/test/msvc7/test.vcproj rename : sourcehook/test/msvc8/test.vcproj => core/sourcehook/test/msvc8/test.vcproj rename : sourcehook/test/sourcehook_test.h => core/sourcehook/test/sourcehook_test.h rename : sourcehook/test/test1.cpp => core/sourcehook/test/test1.cpp rename : sourcehook/test/test2.cpp => core/sourcehook/test/test2.cpp rename : sourcehook/test/test3.cpp => core/sourcehook/test/test3.cpp rename : sourcehook/test/test4.cpp => core/sourcehook/test/test4.cpp rename : sourcehook/test/testbail.cpp => core/sourcehook/test/testbail.cpp rename : sourcehook/test/testbail.h => core/sourcehook/test/testbail.h rename : sourcehook/test/testbail2.cpp => core/sourcehook/test/testbail2.cpp rename : sourcehook/test/testevents.h => core/sourcehook/test/testevents.h rename : sourcehook/test/testhookmangen.cpp => core/sourcehook/test/testhookmangen.cpp rename : sourcehook/test/testhookmangen.h => core/sourcehook/test/testhookmangen.h rename : sourcehook/test/testhookmangen.hxx => core/sourcehook/test/testhookmangen.hxx rename : sourcehook/test/testlist.cpp => core/sourcehook/test/testlist.cpp rename : sourcehook/test/testmanual.cpp => core/sourcehook/test/testmanual.cpp rename : sourcehook/test/testmulti.cpp => core/sourcehook/test/testmulti.cpp rename : sourcehook/test/testrecall.cpp => core/sourcehook/test/testrecall.cpp rename : sourcehook/test/testreentr.cpp => core/sourcehook/test/testreentr.cpp rename : sourcehook/test/testref.cpp => core/sourcehook/test/testref.cpp rename : sourcehook/test/testrefret.cpp => core/sourcehook/test/testrefret.cpp rename : sourcehook/test/testvphooks.cpp => core/sourcehook/test/testvphooks.cpp rename : sourcemm/svn_version.h => core/svn_version.h rename : sourcemm/svn_version.tpl => core/svn_version.tpl rename : sourcemm/version.rc => core/version.rc
289 lines
6.9 KiB
C++
289 lines
6.9 KiB
C++
#ifndef __SH_PAGEALLOC_H__
|
||
#define __SH_PAGEALLOC_H__
|
||
|
||
# if /********/ defined _WIN32
|
||
# include <windows.h>
|
||
# elif /******/ defined __linux__
|
||
# include <sys/mman.h>
|
||
# include <unistd.h>
|
||
# else
|
||
# error Unsupported OS/Compiler
|
||
# endif
|
||
|
||
|
||
namespace SourceHook
|
||
{
|
||
|
||
/*
|
||
Class which lets us allocate memory regions in special pages only meant for on the fly code generation.
|
||
|
||
If we alloc with malloc and then set the page access type to read/exec only, other regions returned by
|
||
malloc that are in the same page would lose their write access as well and the process could crash.
|
||
|
||
Allocating one page per code generation session is usually a waste of memory and on some platforms also
|
||
a waste of virtual address space (Windows’ VirtualAlloc has a granularity of 64K).
|
||
|
||
|
||
IMPORTANT: the memory that Alloc() returns is not a in a defined state!
|
||
It could be in read+exec OR read+write mode.
|
||
-> call SetRE() or SetRW() before using allocated memory!
|
||
*/
|
||
class CPageAlloc
|
||
{
|
||
struct AllocationUnit
|
||
{
|
||
size_t begin_offset;
|
||
size_t size;
|
||
|
||
AllocationUnit(size_t p_offs, size_t p_size) : begin_offset(p_offs), size(p_size)
|
||
{
|
||
}
|
||
|
||
bool operator < (const AllocationUnit &other) const
|
||
{
|
||
return begin_offset < other.begin_offset;
|
||
}
|
||
};
|
||
|
||
typedef List<AllocationUnit> AUList;
|
||
struct AllocatedRegion
|
||
{
|
||
void *startPtr;
|
||
size_t size;
|
||
bool isolated; // may contain only one AU
|
||
size_t minAlignment;
|
||
AUList allocUnits;
|
||
|
||
void CheckGap(size_t gap_begin, size_t gap_end, size_t reqsize,
|
||
size_t &smallestgap_pos, size_t &smallestgap_size, size_t &outAlignBytes)
|
||
{
|
||
size_t gapsize = gap_end - gap_begin;
|
||
// How many bytes do we actually need here?
|
||
// = requested size + alignment bytes
|
||
size_t neededSize = reqsize;
|
||
size_t alignBytes = minAlignment - ((reinterpret_cast<intptr_t>(startPtr) + gap_begin) % minAlignment);
|
||
|
||
alignBytes %= minAlignment;
|
||
neededSize += alignBytes;
|
||
|
||
if (gapsize >= neededSize)
|
||
{
|
||
if (gapsize < smallestgap_size)
|
||
{
|
||
smallestgap_size = gapsize;
|
||
smallestgap_pos = gap_begin;
|
||
outAlignBytes = alignBytes;
|
||
}
|
||
}
|
||
}
|
||
|
||
bool TryAlloc(size_t reqsize, void * &outAddr)
|
||
{
|
||
// Check for isolated
|
||
if (isolated && !allocUnits.empty())
|
||
return false;
|
||
|
||
// Find the smallest gap where req fits
|
||
size_t lastend = 0;
|
||
size_t smallestgap_pos = size + 1;
|
||
size_t smallestgap_size = size + 1;
|
||
size_t alignmentbytes = 0;
|
||
|
||
for (AUList::iterator iter = allocUnits.begin(); iter != allocUnits.end(); ++iter)
|
||
{
|
||
CheckGap(lastend, iter->begin_offset, reqsize, smallestgap_pos, smallestgap_size, alignmentbytes);
|
||
lastend = iter->begin_offset + iter->size;
|
||
}
|
||
|
||
CheckGap(lastend, size, reqsize, smallestgap_pos, smallestgap_size, alignmentbytes);
|
||
|
||
if (smallestgap_pos < size)
|
||
{
|
||
outAddr = reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + smallestgap_pos + alignmentbytes);
|
||
allocUnits.push_sorted( AllocationUnit(smallestgap_pos, reqsize + alignmentbytes) );
|
||
return true;
|
||
}
|
||
else
|
||
{
|
||
return false;
|
||
}
|
||
}
|
||
|
||
bool TryFree(void *addr)
|
||
{
|
||
if (addr < startPtr || addr >= reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + size))
|
||
return false;
|
||
|
||
intptr_t start = reinterpret_cast<intptr_t>(startPtr);
|
||
|
||
for (AUList::iterator iter = allocUnits.begin(); iter != allocUnits.end(); ++iter)
|
||
{
|
||
size_t AUBegin = start + iter->begin_offset;
|
||
void *alignedAUBegin = reinterpret_cast<void*>(
|
||
AUBegin + ((minAlignment - AUBegin % minAlignment) % minAlignment)
|
||
);
|
||
|
||
if (addr == alignedAUBegin)
|
||
{
|
||
allocUnits.erase(iter);
|
||
return true;
|
||
}
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
bool Contains(void *addr)
|
||
{
|
||
return addr >= startPtr && addr < reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + size);
|
||
}
|
||
|
||
void FreeRegion()
|
||
{
|
||
#ifdef __linux__
|
||
munmap(startPtr, size);
|
||
#else
|
||
VirtualFree(startPtr, 0, MEM_RELEASE);
|
||
#endif
|
||
}
|
||
};
|
||
|
||
typedef List<AllocatedRegion> ARList;
|
||
|
||
size_t m_MinAlignment;
|
||
size_t m_PageSize;
|
||
ARList m_Regions;
|
||
|
||
bool AddRegion(size_t minSize, bool isolated)
|
||
{
|
||
AllocatedRegion newRegion;
|
||
newRegion.startPtr = 0;
|
||
newRegion.isolated = isolated;
|
||
newRegion.minAlignment = m_MinAlignment;
|
||
|
||
// Compute real size -> align up to m_PageSize boundary
|
||
|
||
newRegion.size = minSize - (minSize % m_PageSize);
|
||
if (newRegion.size < minSize)
|
||
newRegion.size += m_PageSize;
|
||
|
||
#ifdef __linux__
|
||
newRegion.startPtr = mmap(0, newRegion.size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
|
||
#else
|
||
newRegion.startPtr = VirtualAlloc(NULL, newRegion.size, MEM_COMMIT, PAGE_READWRITE);
|
||
#endif
|
||
|
||
if (newRegion.startPtr)
|
||
{
|
||
m_Regions.push_back(newRegion);
|
||
return true;
|
||
}
|
||
else
|
||
{
|
||
return false;
|
||
}
|
||
|
||
}
|
||
|
||
void *AllocPriv(size_t size, bool isolated)
|
||
{
|
||
void *addr;
|
||
|
||
if (!isolated)
|
||
{
|
||
for (ARList::iterator iter = m_Regions.begin(); iter != m_Regions.end(); ++iter)
|
||
{
|
||
if (iter->TryAlloc(size, addr))
|
||
return addr;
|
||
}
|
||
}
|
||
|
||
if (!AddRegion(size, isolated))
|
||
return NULL;
|
||
|
||
bool tmp = m_Regions.back().TryAlloc(size, addr);
|
||
SH_ASSERT(tmp, ("TryAlloc fails after AddRegion"));
|
||
return tmp ? addr : NULL;
|
||
}
|
||
|
||
public:
|
||
CPageAlloc(size_t minAlignment = 1 /* power of 2 */ ) : m_MinAlignment(minAlignment)
|
||
{
|
||
#ifdef __linux__
|
||
m_PageSize = sysconf(_SC_PAGESIZE);
|
||
#else
|
||
SYSTEM_INFO sysInfo;
|
||
GetSystemInfo(&sysInfo);
|
||
m_PageSize = sysInfo.dwPageSize;
|
||
#endif
|
||
}
|
||
|
||
~CPageAlloc()
|
||
{
|
||
// Free all regions
|
||
for (ARList::iterator iter = m_Regions.begin(); iter != m_Regions.end(); ++iter)
|
||
{
|
||
iter->FreeRegion();
|
||
}
|
||
}
|
||
|
||
void *Alloc(size_t size)
|
||
{
|
||
return AllocPriv(size, false);
|
||
}
|
||
|
||
void *AllocIsolated(size_t size)
|
||
{
|
||
return AllocPriv(size, true);
|
||
}
|
||
|
||
void Free(void *ptr)
|
||
{
|
||
for (ARList::iterator iter = m_Regions.begin(); iter != m_Regions.end(); ++iter)
|
||
{
|
||
if (iter->TryFree(ptr))
|
||
{
|
||
if (iter->allocUnits.empty())
|
||
{
|
||
iter->FreeRegion();
|
||
m_Regions.erase(iter);
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
void SetRE(void *ptr)
|
||
{
|
||
for (ARList::iterator iter = m_Regions.begin(); iter != m_Regions.end(); ++iter)
|
||
{
|
||
if (iter->Contains(ptr))
|
||
{
|
||
SetMemAccess(iter->startPtr, iter->size, SH_MEM_READ | SH_MEM_EXEC);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
void SetRW(void *ptr)
|
||
{
|
||
for (ARList::iterator iter = m_Regions.begin(); iter != m_Regions.end(); ++iter)
|
||
{
|
||
if (iter->Contains(ptr))
|
||
{
|
||
SetMemAccess(iter->startPtr, iter->size, SH_MEM_READ | SH_MEM_WRITE);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
size_t GetPageSize()
|
||
{
|
||
return m_PageSize;
|
||
}
|
||
};
|
||
}
|
||
|
||
#endif
|
||
|