2008-07-06 00:28:02 +02:00
|
|
|
/**
|
|
|
|
* vim: set ts=4 :
|
|
|
|
* =============================================================================
|
|
|
|
* SourceMod
|
|
|
|
* Copyright (C) 2004-2008 AlliedModders LLC. All rights reserved.
|
|
|
|
* =============================================================================
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it under
|
|
|
|
* the terms of the GNU General Public License, version 3.0, as published by the
|
|
|
|
* Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program 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
|
|
|
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* As a special exception, AlliedModders LLC gives you permission to link the
|
|
|
|
* code of this program (as well as its derivative works) to "Half-Life 2," the
|
|
|
|
* "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
|
|
|
|
* by the Valve Corporation. You must obey the GNU General Public License in
|
|
|
|
* all respects for all other code used. Additionally, AlliedModders LLC grants
|
|
|
|
* this exception to all derivative works. AlliedModders LLC defines further
|
|
|
|
* exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
|
|
|
|
* or <http://www.sourcemod.net/license.php>.
|
|
|
|
*
|
2009-02-26 00:42:13 +01:00
|
|
|
* Version: $Id: detours.cpp 248 2008-08-27 00:56:22Z pred $
|
2008-07-06 00:28:02 +02:00
|
|
|
*/
|
|
|
|
|
2008-07-01 11:04:00 +02:00
|
|
|
#include "detours.h"
|
|
|
|
#include <asm/asm.h>
|
|
|
|
|
|
|
|
ISourcePawnEngine *CDetourManager::spengine = NULL;
|
|
|
|
IGameConfig *CDetourManager::gameconf = NULL;
|
|
|
|
|
|
|
|
void CDetourManager::Init(ISourcePawnEngine *spengine, IGameConfig *gameconf)
|
|
|
|
{
|
|
|
|
CDetourManager::spengine = spengine;
|
|
|
|
CDetourManager::gameconf = gameconf;
|
|
|
|
}
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
CDetour *CDetourManager::CreateDetour(void *callbackfunction, void **trampoline, const char *signame)
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
2009-02-26 00:42:13 +01:00
|
|
|
CDetour *detour = new CDetour(callbackfunction, trampoline, signame);
|
2008-07-01 11:04:00 +02:00
|
|
|
if (detour)
|
|
|
|
{
|
|
|
|
if (!detour->Init(spengine, gameconf))
|
|
|
|
{
|
|
|
|
delete detour;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return detour;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
CDetour::CDetour(void *callbackfunction, void **trampoline, const char *signame)
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
|
|
|
enabled = false;
|
|
|
|
detoured = false;
|
|
|
|
detour_address = NULL;
|
2009-02-26 00:42:13 +01:00
|
|
|
detour_trampoline = NULL;
|
2008-07-01 11:04:00 +02:00
|
|
|
this->signame = signame;
|
2009-02-26 00:42:13 +01:00
|
|
|
this->detour_callback = callbackfunction;
|
2008-07-01 11:04:00 +02:00
|
|
|
spengine = NULL;
|
|
|
|
gameconf = NULL;
|
2009-02-26 00:42:13 +01:00
|
|
|
this->trampoline = trampoline;
|
2008-07-01 11:04:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CDetour::Init(ISourcePawnEngine *spengine, IGameConfig *gameconf)
|
|
|
|
{
|
|
|
|
this->spengine = spengine;
|
|
|
|
this->gameconf = gameconf;
|
|
|
|
|
|
|
|
if (!CreateDetour())
|
|
|
|
{
|
|
|
|
enabled = false;
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
enabled = true;
|
|
|
|
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
void CDetour::Destroy()
|
|
|
|
{
|
|
|
|
DeleteDetour();
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
2008-07-01 11:04:00 +02:00
|
|
|
bool CDetour::IsEnabled()
|
|
|
|
{
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDetour::CreateDetour()
|
|
|
|
{
|
|
|
|
if (!gameconf->GetMemSig(signame, &detour_address))
|
|
|
|
{
|
|
|
|
g_pSM->LogError(myself, "Could not locate %s - Disabling detour", signame);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!detour_address)
|
|
|
|
{
|
2008-07-06 00:28:02 +02:00
|
|
|
g_pSM->LogError(myself, "Sigscan for %s failed - Disabling detour to prevent crashes", signame);
|
2008-07-01 11:04:00 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-07-06 00:28:02 +02:00
|
|
|
detour_restore.bytes = copy_bytes((unsigned char *)detour_address, NULL, OP_JMP_SIZE+1);
|
2008-07-01 11:04:00 +02:00
|
|
|
|
|
|
|
/* First, save restore bits */
|
|
|
|
for (size_t i=0; i<detour_restore.bytes; i++)
|
|
|
|
{
|
|
|
|
detour_restore.patch[i] = ((unsigned char *)detour_address)[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
JitWriter wr;
|
|
|
|
JitWriter *jit = ≀
|
|
|
|
jit_uint32_t CodeSize = 0;
|
|
|
|
|
|
|
|
wr.outbase = NULL;
|
|
|
|
wr.outptr = NULL;
|
|
|
|
|
|
|
|
jit_rewind:
|
|
|
|
|
|
|
|
/* Patch old bytes in */
|
|
|
|
if (wr.outbase != NULL)
|
|
|
|
{
|
|
|
|
copy_bytes((unsigned char *)detour_address, (unsigned char*)wr.outptr, detour_restore.bytes);
|
|
|
|
}
|
|
|
|
wr.outptr += detour_restore.bytes;
|
|
|
|
|
|
|
|
/* Return to the original function */
|
2009-02-26 00:42:13 +01:00
|
|
|
jitoffs_t call = IA32_Jump_Imm32(jit, 0);
|
2008-07-01 11:04:00 +02:00
|
|
|
IA32_Write_Jump32_Abs(jit, call, (unsigned char *)detour_address + detour_restore.bytes);
|
|
|
|
|
|
|
|
if (wr.outbase == NULL)
|
|
|
|
{
|
|
|
|
CodeSize = wr.get_outputpos();
|
|
|
|
wr.outbase = (jitcode_t)spengine->AllocatePageMemory(CodeSize);
|
|
|
|
spengine->SetReadWrite(wr.outbase);
|
|
|
|
wr.outptr = wr.outbase;
|
2009-02-26 00:42:13 +01:00
|
|
|
detour_trampoline = wr.outbase;
|
2008-07-01 11:04:00 +02:00
|
|
|
goto jit_rewind;
|
|
|
|
}
|
|
|
|
|
|
|
|
spengine->SetReadExecute(wr.outbase);
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
*trampoline = detour_trampoline;
|
2008-07-01 11:04:00 +02:00
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
return true;
|
2008-07-01 11:04:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CDetour::DeleteDetour()
|
|
|
|
{
|
|
|
|
if (detoured)
|
|
|
|
{
|
|
|
|
DisableDetour();
|
|
|
|
}
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
if (detour_trampoline)
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
2009-02-26 00:42:13 +01:00
|
|
|
/* Free the allocated trampoline memory */
|
|
|
|
spengine->FreePageMemory(detour_trampoline);
|
|
|
|
detour_trampoline = NULL;
|
2008-07-01 11:04:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
void CDetour::EnableDetour()
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
2009-02-26 00:42:13 +01:00
|
|
|
if (!detoured)
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
2009-02-26 00:42:13 +01:00
|
|
|
DoGatePatch((unsigned char *)detour_address, &detour_callback);
|
|
|
|
detoured = true;
|
2008-07-01 11:04:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-26 00:42:13 +01:00
|
|
|
void CDetour::DisableDetour()
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
2009-02-26 00:42:13 +01:00
|
|
|
if (detoured)
|
2008-07-01 11:04:00 +02:00
|
|
|
{
|
2009-02-26 00:42:13 +01:00
|
|
|
/* Remove the patch */
|
|
|
|
ApplyPatch(detour_address, 0, &detour_restore, NULL);
|
|
|
|
detoured = false;
|
2008-07-01 11:04:00 +02:00
|
|
|
}
|
|
|
|
}
|