mirror of
https://github.com/alliedmodders/metamod-source.git
synced 2025-02-20 13:54:14 +01:00
Moved CPageAlloc from sh_memory into sh_pagealloc.h
--HG-- branch : hookman_autogen extra : convert_revision : svn%3Ac2935e3e-5518-0410-8daf-afa5dab7d4e3/branches/hookman_autogen%40545
This commit is contained in:
parent
b6f0b74f5b
commit
c5595bd4e5
@ -185,258 +185,6 @@ namespace SourceHook
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
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 plattforms 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
|
|
||||||
AUList allocUnits;
|
|
||||||
|
|
||||||
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;
|
|
||||||
|
|
||||||
for (AUList::iterator iter = allocUnits.begin(); iter != allocUnits.end(); ++iter)
|
|
||||||
{
|
|
||||||
if (iter->begin_offset - lastend >= reqsize)
|
|
||||||
{
|
|
||||||
if (iter->begin_offset - lastend < smallestgap_size)
|
|
||||||
{
|
|
||||||
smallestgap_size = iter->begin_offset - lastend;
|
|
||||||
smallestgap_pos = lastend;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
lastend = iter->begin_offset + iter->size;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (size - lastend >= reqsize)
|
|
||||||
{
|
|
||||||
if (size - lastend < smallestgap_size)
|
|
||||||
{
|
|
||||||
smallestgap_size = size - lastend;
|
|
||||||
smallestgap_pos = lastend;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (smallestgap_pos < size)
|
|
||||||
{
|
|
||||||
outAddr = reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + smallestgap_pos);
|
|
||||||
allocUnits.push_sorted( AllocationUnit(smallestgap_pos, reqsize) );
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryFree(void *addr)
|
|
||||||
{
|
|
||||||
if (addr < startPtr || addr >= reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + size))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
size_t offs = reinterpret_cast<char*>(addr) - reinterpret_cast<char*>(startPtr);
|
|
||||||
|
|
||||||
for (AUList::iterator iter = allocUnits.begin(); iter != allocUnits.end(); ++iter)
|
|
||||||
{
|
|
||||||
if (iter->begin_offset == offs)
|
|
||||||
{
|
|
||||||
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_PageSize;
|
|
||||||
ARList m_Regions;
|
|
||||||
|
|
||||||
bool AddRegion(size_t minSize, bool isolated)
|
|
||||||
{
|
|
||||||
AllocatedRegion newRegion;
|
|
||||||
newRegion.startPtr = 0;
|
|
||||||
newRegion.isolated = isolated;
|
|
||||||
|
|
||||||
// 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 addr;
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
|
||||||
CPageAlloc()
|
|
||||||
{
|
|
||||||
#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
|
#endif
|
||||||
|
273
sourcehook/sh_pagealloc.h
Normal file
273
sourcehook/sh_pagealloc.h
Normal file
@ -0,0 +1,273 @@
|
|||||||
|
#ifndef __SH_PAGEALLOC_H__
|
||||||
|
#define __SH_PAGEALLOC_H__
|
||||||
|
|
||||||
|
# if /********/ defined _WIN32
|
||||||
|
# include <windows.h>
|
||||||
|
# define SH_MEM_READ 1
|
||||||
|
# define SH_MEM_WRITE 2
|
||||||
|
# define SH_MEM_EXEC 4
|
||||||
|
# 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 plattforms 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
|
||||||
|
AUList allocUnits;
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
for (AUList::iterator iter = allocUnits.begin(); iter != allocUnits.end(); ++iter)
|
||||||
|
{
|
||||||
|
if (iter->begin_offset - lastend >= reqsize)
|
||||||
|
{
|
||||||
|
if (iter->begin_offset - lastend < smallestgap_size)
|
||||||
|
{
|
||||||
|
smallestgap_size = iter->begin_offset - lastend;
|
||||||
|
smallestgap_pos = lastend;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
lastend = iter->begin_offset + iter->size;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size - lastend >= reqsize)
|
||||||
|
{
|
||||||
|
if (size - lastend < smallestgap_size)
|
||||||
|
{
|
||||||
|
smallestgap_size = size - lastend;
|
||||||
|
smallestgap_pos = lastend;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (smallestgap_pos < size)
|
||||||
|
{
|
||||||
|
outAddr = reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + smallestgap_pos);
|
||||||
|
allocUnits.push_sorted( AllocationUnit(smallestgap_pos, reqsize) );
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryFree(void *addr)
|
||||||
|
{
|
||||||
|
if (addr < startPtr || addr >= reinterpret_cast<void*>(reinterpret_cast<char*>(startPtr) + size))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
size_t offs = reinterpret_cast<char*>(addr) - reinterpret_cast<char*>(startPtr);
|
||||||
|
|
||||||
|
for (AUList::iterator iter = allocUnits.begin(); iter != allocUnits.end(); ++iter)
|
||||||
|
{
|
||||||
|
if (iter->begin_offset == offs)
|
||||||
|
{
|
||||||
|
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_PageSize;
|
||||||
|
ARList m_Regions;
|
||||||
|
|
||||||
|
bool AddRegion(size_t minSize, bool isolated)
|
||||||
|
{
|
||||||
|
AllocatedRegion newRegion;
|
||||||
|
newRegion.startPtr = 0;
|
||||||
|
newRegion.isolated = isolated;
|
||||||
|
|
||||||
|
// 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 addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
CPageAlloc()
|
||||||
|
{
|
||||||
|
#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
|
@ -12,6 +12,7 @@
|
|||||||
#define __SOURCEHOOK_HOOKMANGEN_H__
|
#define __SOURCEHOOK_HOOKMANGEN_H__
|
||||||
|
|
||||||
#include "sourcehook_impl.h"
|
#include "sourcehook_impl.h"
|
||||||
|
#include "sh_pagealloc.h"
|
||||||
|
|
||||||
namespace SourceHook
|
namespace SourceHook
|
||||||
{
|
{
|
||||||
|
@ -312,6 +312,12 @@
|
|||||||
<File
|
<File
|
||||||
RelativePath="..\..\sh_list.h">
|
RelativePath="..\..\sh_list.h">
|
||||||
</File>
|
</File>
|
||||||
|
<File
|
||||||
|
RelativePath="..\..\sh_memory.h">
|
||||||
|
</File>
|
||||||
|
<File
|
||||||
|
RelativePath="..\..\sh_pagealloc.h">
|
||||||
|
</File>
|
||||||
<File
|
<File
|
||||||
RelativePath="..\..\sh_stack.h">
|
RelativePath="..\..\sh_stack.h">
|
||||||
</File>
|
</File>
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
#include "testevents.h"
|
#include "testevents.h"
|
||||||
#include "sourcehook_hookmangen.h"
|
#include "sourcehook_hookmangen.h"
|
||||||
#include "sh_memory.h"
|
#include "sh_memory.h"
|
||||||
|
#include "sh_pagealloc.h"
|
||||||
|
|
||||||
// TESTHOOKMANGEN
|
// TESTHOOKMANGEN
|
||||||
// Test automatic hookman generation
|
// Test automatic hookman generation
|
||||||
|
Loading…
x
Reference in New Issue
Block a user