2009-09-26 23:14:50 +02:00
|
|
|
/**
|
|
|
|
* vim: set ts=4 sw=4 tw=99 noet :
|
|
|
|
* =============================================================================
|
|
|
|
* SourceMod
|
|
|
|
* Copyright (C) 2004-2009 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>.
|
|
|
|
*
|
|
|
|
* Version: $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* On SourceHook v4.3 or lower, there are no DVP hooks. Very sad, right?
|
2009-10-28 08:37:34 +01:00
|
|
|
* Only do this on newer versions. For the older code, we'll do an incredibly
|
2009-09-26 23:14:50 +02:00
|
|
|
* hacky detour.
|
|
|
|
*
|
|
|
|
* The idea of the "non-hacky" (yeah... no) code is that every unique
|
|
|
|
* ConCommand vtable gets its own DVP hook. We watch for unloading and
|
|
|
|
* loading commands to remove stale hooks from SH.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "sourcemod.h"
|
|
|
|
#include "sourcemm_api.h"
|
|
|
|
#include "Logger.h"
|
|
|
|
#include "compat_wrappers.h"
|
|
|
|
#include "ConsoleDetours.h"
|
2010-05-15 08:35:42 +02:00
|
|
|
#include <IGameConfigs.h>
|
2009-09-26 23:14:50 +02:00
|
|
|
#include "sm_stringutil.h"
|
|
|
|
#include "ConCmdManager.h"
|
|
|
|
#include "HalfLife2.h"
|
|
|
|
#include "ConCommandBaseIterator.h"
|
2013-03-29 19:37:29 +01:00
|
|
|
#include "logic_bridge.h"
|
2013-08-25 21:15:26 +02:00
|
|
|
#include <am-utility.h>
|
2009-09-26 23:14:50 +02:00
|
|
|
|
2010-05-30 00:11:47 +02:00
|
|
|
#if defined PLATFORM_POSIX
|
2009-09-26 23:14:50 +02:00
|
|
|
# include <dlfcn.h>
|
|
|
|
# include <sys/mman.h>
|
|
|
|
# include <stdint.h>
|
|
|
|
# include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if SH_IMPL_VERSION >= 5
|
2013-04-23 13:11:49 +02:00
|
|
|
# if SOURCE_ENGINE == SE_DOTA
|
2013-07-12 08:31:41 +02:00
|
|
|
SH_DECL_EXTERN2_void(ConCommand, Dispatch, SH_NOATTRIB, false, const CCommandContext &, const CCommand &);
|
2013-04-23 13:11:49 +02:00
|
|
|
# elif SOURCE_ENGINE >= SE_ORANGEBOX
|
2009-09-26 23:14:50 +02:00
|
|
|
SH_DECL_EXTERN1_void(ConCommand, Dispatch, SH_NOATTRIB, false, const CCommand &);
|
|
|
|
# else
|
|
|
|
SH_DECL_EXTERN0_void(ConCommand, Dispatch, SH_NOATTRIB, false);
|
|
|
|
# endif
|
2009-10-29 07:30:52 +01:00
|
|
|
#else
|
|
|
|
# if SH_IMPL_VERSION >= 4
|
|
|
|
extern int __SourceHook_FHVPAddConCommandDispatch(void *,bool,class fastdelegate::FastDelegate0<void>,bool);
|
|
|
|
extern int __SourceHook_FHAddConCommandDispatch(void *, bool, class fastdelegate::FastDelegate0<void>);
|
|
|
|
# else
|
|
|
|
extern bool __SourceHook_FHAddConCommandDispatch(void *, bool, class fastdelegate::FastDelegate0<void>);
|
|
|
|
# endif
|
2009-09-26 23:14:50 +02:00
|
|
|
extern bool __SourceHook_FHRemoveConCommandDispatch(void *, bool, class fastdelegate::FastDelegate0<void>);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if SH_IMPL_VERSION >= 4
|
|
|
|
|
|
|
|
class GenericCommandHooker : public IConCommandLinkListener
|
|
|
|
{
|
|
|
|
struct HackInfo
|
|
|
|
{
|
|
|
|
void **vtable;
|
|
|
|
int hook;
|
|
|
|
unsigned int refcount;
|
|
|
|
};
|
|
|
|
CVector<HackInfo> vtables;
|
|
|
|
bool enabled;
|
|
|
|
SourceHook::MemFuncInfo dispatch;
|
|
|
|
|
|
|
|
inline void **GetVirtualTable(ConCommandBase *pBase)
|
|
|
|
{
|
|
|
|
return *reinterpret_cast<void***>(reinterpret_cast<char*>(pBase) +
|
|
|
|
dispatch.thisptroffs +
|
|
|
|
dispatch.vtbloffs);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool FindVtable(void **ptr, size_t& index)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < vtables.size(); i++)
|
|
|
|
{
|
|
|
|
if (vtables[i].vtable == ptr)
|
|
|
|
{
|
|
|
|
index = i;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MakeHookable(ConCommandBase *pBase)
|
|
|
|
{
|
|
|
|
if (!pBase->IsCommand())
|
|
|
|
return;
|
|
|
|
|
|
|
|
ConCommand *cmd = (ConCommand*)pBase;
|
|
|
|
void **vtable = GetVirtualTable(cmd);
|
|
|
|
|
|
|
|
size_t index;
|
|
|
|
if (!FindVtable(vtable, index))
|
|
|
|
{
|
|
|
|
HackInfo hack;
|
|
|
|
hack.vtable = vtable;
|
|
|
|
hack.hook = SH_ADD_VPHOOK(ConCommand, Dispatch, cmd, SH_MEMBER(this, &GenericCommandHooker::Dispatch), false);
|
|
|
|
hack.refcount = 1;
|
|
|
|
vtables.push_back(hack);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
vtables[index].refcount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-23 13:11:49 +02:00
|
|
|
# if SOURCE_ENGINE == SE_DOTA
|
2013-07-12 08:31:41 +02:00
|
|
|
void Dispatch(const CCommandContext &context, const CCommand& args)
|
2013-04-23 13:11:49 +02:00
|
|
|
# elif SOURCE_ENGINE >= SE_ORANGEBOX
|
2009-09-26 23:14:50 +02:00
|
|
|
void Dispatch(const CCommand& args)
|
|
|
|
# else
|
|
|
|
void Dispatch()
|
|
|
|
# endif
|
|
|
|
{
|
|
|
|
cell_t res = ConsoleDetours::Dispatch(META_IFACEPTR(ConCommand)
|
|
|
|
# if SOURCE_ENGINE >= SE_ORANGEBOX
|
|
|
|
, args
|
|
|
|
# endif
|
|
|
|
);
|
|
|
|
if (res >= Pl_Handled)
|
|
|
|
RETURN_META(MRES_SUPERCEDE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ReparseCommandList()
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < vtables.size(); i++)
|
|
|
|
vtables[i].refcount = 0;
|
|
|
|
for (ConCommandBaseIterator iter; iter.IsValid(); iter.Next())
|
|
|
|
MakeHookable(iter.Get());
|
|
|
|
CVector<HackInfo>::iterator iter = vtables.begin();
|
|
|
|
while (iter != vtables.end())
|
|
|
|
{
|
|
|
|
if ((*iter).refcount)
|
2009-10-29 08:06:12 +01:00
|
|
|
{
|
|
|
|
iter++;
|
2009-09-26 23:14:50 +02:00
|
|
|
continue;
|
2009-10-29 08:06:12 +01:00
|
|
|
}
|
2009-09-26 23:14:50 +02:00
|
|
|
/* Damn it. This event happens AFTER the plugin has unloaded!
|
|
|
|
* There's two options. Remove the hook now and hope SH's memory
|
|
|
|
* protection will prevent a crash. Otherwise, we can wait until
|
|
|
|
* the server shuts down and more likely crash then.
|
|
|
|
*
|
|
|
|
* This situation only arises if:
|
|
|
|
* 1) Someone has used AddCommandFilter()
|
|
|
|
* 2) ... on a Dark Messiah server (mm:s new api)
|
|
|
|
* 3) ... and another MM:S plugin that uses ConCommands has unloaded.
|
|
|
|
*
|
|
|
|
* Even though the impact is really small, we'll wait until the
|
|
|
|
* server shuts down, so normal operation isn't interrupted.
|
|
|
|
*
|
|
|
|
* See bug 4018.
|
|
|
|
*/
|
|
|
|
iter = vtables.erase(iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UnhookCommand(ConCommandBase *pBase)
|
|
|
|
{
|
|
|
|
if (!pBase->IsCommand())
|
|
|
|
return;
|
|
|
|
|
|
|
|
ConCommand *cmd = (ConCommand*)pBase;
|
|
|
|
void **vtable = GetVirtualTable(cmd);
|
|
|
|
|
|
|
|
size_t index;
|
|
|
|
if (!FindVtable(vtable, index))
|
|
|
|
{
|
2014-08-13 23:24:35 +02:00
|
|
|
logger->LogError("Console detour tried to unhook command \"%s\" but it wasn't found", pBase->GetName());
|
2009-09-26 23:14:50 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(vtables[index].refcount > 0);
|
|
|
|
vtables[index].refcount--;
|
|
|
|
if (vtables[index].refcount == 0)
|
|
|
|
{
|
|
|
|
SH_REMOVE_HOOK_ID(vtables[index].hook);
|
|
|
|
vtables.erase(vtables.iterAt(index));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
GenericCommandHooker() : enabled(false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Enable()
|
|
|
|
{
|
|
|
|
SourceHook::GetFuncInfo(&ConCommand::Dispatch, dispatch);
|
|
|
|
|
|
|
|
if (dispatch.thisptroffs < 0)
|
|
|
|
{
|
2014-08-13 23:24:35 +02:00
|
|
|
logger->LogError("Command filter could not determine ConCommand layout");
|
2009-09-26 23:14:50 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ConCommandBaseIterator iter; iter.IsValid(); iter.Next())
|
|
|
|
MakeHookable(iter.Get());
|
|
|
|
|
|
|
|
if (!vtables.size())
|
|
|
|
{
|
2014-08-13 23:24:35 +02:00
|
|
|
logger->LogError("Command filter could not find any cvars!");
|
2009-09-26 23:14:50 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
enabled = true;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Disable()
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < vtables.size(); i++)
|
|
|
|
SH_REMOVE_HOOK_ID(vtables[i].hook);
|
|
|
|
vtables.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void OnLinkConCommand(ConCommandBase *pBase)
|
|
|
|
{
|
|
|
|
if (!enabled)
|
|
|
|
return;
|
|
|
|
MakeHookable(pBase);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void OnUnlinkConCommand(ConCommandBase *pBase)
|
|
|
|
{
|
|
|
|
if (!enabled)
|
|
|
|
return;
|
|
|
|
if (pBase == NULL)
|
|
|
|
ReparseCommandList();
|
|
|
|
else
|
|
|
|
UnhookCommand(pBase);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* END DVP HOOK VERSION
|
|
|
|
*/
|
|
|
|
|
|
|
|
#else /* SH_IMPL_VERSION >= 4 */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* BEGIN ENGINE DETOUR VERSION
|
|
|
|
*/
|
|
|
|
|
|
|
|
# include <jit/jit_helpers.h>
|
|
|
|
# include <jit/x86/x86_macros.h>
|
|
|
|
|
|
|
|
class GenericCommandHooker
|
|
|
|
{
|
|
|
|
struct Patch
|
|
|
|
{
|
|
|
|
Patch() : applied(false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool applied;
|
|
|
|
void *base;
|
|
|
|
unsigned char search[16];
|
|
|
|
size_t search_len;
|
|
|
|
size_t offset;
|
|
|
|
size_t saveat;
|
|
|
|
int regparam;
|
|
|
|
void *detour;
|
|
|
|
};
|
|
|
|
|
|
|
|
Patch ces;
|
|
|
|
Patch cgc;
|
|
|
|
|
|
|
|
public:
|
2009-10-28 08:50:48 +01:00
|
|
|
static void DelayedActivation(void *inparam)
|
|
|
|
{
|
|
|
|
GenericCommandHooker *cdtrs = reinterpret_cast<GenericCommandHooker*>(inparam);
|
|
|
|
/* Safe to re-enter because the frame queue is lock+swapped. */
|
|
|
|
if ((!cdtrs->ces.applied || !cdtrs->cgc.applied) &&
|
|
|
|
g_HL2.PeekCommandStack() != NULL)
|
|
|
|
{
|
|
|
|
g_SourceMod.AddFrameAction(GenericCommandHooker::DelayedActivation, cdtrs);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cdtrs->ces.applied)
|
|
|
|
cdtrs->ApplyPatch(&cdtrs->ces);
|
|
|
|
if (!cdtrs->cgc.applied)
|
|
|
|
cdtrs->ApplyPatch(&cdtrs->cgc);
|
|
|
|
}
|
|
|
|
|
2009-09-26 23:14:50 +02:00
|
|
|
bool Enable()
|
|
|
|
{
|
|
|
|
const char *platform = NULL;
|
|
|
|
# if defined(PLATFORM_WINDOWS)
|
|
|
|
platform = "Windows";
|
|
|
|
# else
|
|
|
|
void *addrInBase = (void *)g_SMAPI->GetEngineFactory(false);
|
|
|
|
Dl_info info;
|
|
|
|
if (!dladdr(addrInBase, &info))
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter could not find engine name");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (strstr(info.dli_fname, "engine_i486"))
|
|
|
|
{
|
|
|
|
platform = "Linux_486";
|
|
|
|
}
|
|
|
|
else if (strstr(info.dli_fname, "engine_i686"))
|
|
|
|
{
|
|
|
|
platform = "Linux_686";
|
|
|
|
}
|
|
|
|
else if (strstr(info.dli_fname, "engine_amd"))
|
|
|
|
{
|
|
|
|
platform = "Linux_AMD";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter could not determine engine (%s)", info.dli_fname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
# endif
|
|
|
|
|
|
|
|
if (!PrepPatch("Cmd_ExecuteString", "CES", platform, &ces))
|
|
|
|
return false;
|
|
|
|
if (!PrepPatch("CGameClient::ExecuteString", "CGC", platform, &cgc))
|
|
|
|
return false;
|
|
|
|
|
2009-10-28 08:50:48 +01:00
|
|
|
if (g_HL2.PeekCommandStack() != NULL)
|
|
|
|
{
|
|
|
|
g_SourceMod.AddFrameAction(GenericCommandHooker::DelayedActivation, this);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-09-26 23:14:50 +02:00
|
|
|
ApplyPatch(&ces);
|
|
|
|
ApplyPatch(&cgc);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Disable()
|
|
|
|
{
|
|
|
|
UndoPatch(&ces);
|
|
|
|
UndoPatch(&cgc);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
# if !defined PLATFORM_WINDOWS
|
|
|
|
# define PAGE_READWRITE PROT_READ|PROT_WRITE
|
|
|
|
# define PAGE_EXECUTE_READ PROT_READ|PROT_EXEC
|
|
|
|
|
|
|
|
static inline uintptr_t AddrToPage(uintptr_t address)
|
|
|
|
{
|
|
|
|
return (address & ~(uintptr_t(sysconf(_SC_PAGE_SIZE) - 1)));
|
|
|
|
}
|
|
|
|
|
|
|
|
# endif
|
|
|
|
|
|
|
|
void Protect(void *addr, size_t length, int prot)
|
|
|
|
{
|
|
|
|
# if defined PLATFORM_WINDOWS
|
|
|
|
DWORD ignore;
|
|
|
|
VirtualProtect(addr, length, prot, &ignore);
|
|
|
|
# else
|
|
|
|
uintptr_t startPage = AddrToPage(uintptr_t(addr));
|
|
|
|
length += (uintptr_t(addr) - startPage);
|
|
|
|
mprotect((void*)startPage, length, prot);
|
|
|
|
# endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void UndoPatch(Patch *patch)
|
|
|
|
{
|
|
|
|
if (!patch->applied || patch->detour == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_pSourcePawn->FreePageMemory(patch->detour);
|
|
|
|
|
|
|
|
unsigned char *source = (unsigned char *)patch->base + patch->offset;
|
|
|
|
Protect(source, patch->search_len, PAGE_READWRITE);
|
|
|
|
for (size_t i = 0; i < patch->search_len; i++)
|
|
|
|
source[i] = patch->search[i];
|
|
|
|
Protect(source, patch->search_len, PAGE_EXECUTE_READ);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ApplyPatch(Patch *patch)
|
|
|
|
{
|
|
|
|
assert(!patch->applied);
|
|
|
|
|
|
|
|
size_t length = 0;
|
|
|
|
void *callback = (void*)&ConsoleDetours::Dispatch;
|
|
|
|
|
|
|
|
/* Bogus assignment to make compiler is doing the right thing. */
|
|
|
|
patch->detour = callback;
|
|
|
|
|
|
|
|
/* Assemgle the detour. */
|
|
|
|
JitWriter writer;
|
|
|
|
writer.outbase = NULL;
|
|
|
|
writer.outptr = NULL;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
/* Need a specific register, or value on stack? */
|
|
|
|
if (patch->regparam != -1)
|
|
|
|
IA32_Push_Reg(&writer, patch->regparam);
|
|
|
|
/* Call real function. */
|
|
|
|
IA32_Write_Jump32_Abs(&writer, IA32_Call_Imm32(&writer, 0), callback);
|
|
|
|
/* Restore stack. */
|
|
|
|
if (patch->regparam != -1)
|
|
|
|
IA32_Pop_Reg(&writer, patch->regparam);
|
|
|
|
/* Copy any saved bytes */
|
|
|
|
if (patch->saveat)
|
|
|
|
{
|
|
|
|
for (size_t i = patch->saveat; i < patch->search_len; i++)
|
|
|
|
{
|
|
|
|
writer.write_ubyte(patch->search[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Jump back to the caller. */
|
|
|
|
unsigned char *target = (unsigned char *)patch->base + patch->offset + patch->search_len;
|
|
|
|
IA32_Jump_Imm32_Abs(&writer, target);
|
|
|
|
/* Assemble, if we can. */
|
|
|
|
if (writer.outbase == NULL)
|
|
|
|
{
|
|
|
|
length = writer.outptr - writer.outbase;
|
|
|
|
patch->detour = g_pSourcePawn->AllocatePageMemory(length);
|
|
|
|
if (patch->detour == NULL)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Ran out of memory!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
g_pSourcePawn->SetReadWrite(patch->detour);
|
|
|
|
writer.outbase = (jitcode_t)patch->detour;
|
|
|
|
writer.outptr = writer.outbase;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (true);
|
|
|
|
|
|
|
|
g_pSourcePawn->SetReadExecute(patch->detour);
|
|
|
|
|
|
|
|
unsigned char *source = (unsigned char *)patch->base + patch->offset;
|
|
|
|
Protect(source, 6, PAGE_READWRITE);
|
|
|
|
source[0] = 0xFF;
|
|
|
|
source[1] = 0x25;
|
|
|
|
*(void **)&source[2] = &patch->detour;
|
|
|
|
Protect(source, 6, PAGE_EXECUTE_READ);
|
|
|
|
|
|
|
|
patch->applied = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PrepPatch(const char *signature, const char *name, const char *platform, Patch *patch)
|
|
|
|
{
|
|
|
|
/* Get the base address of the function. */
|
|
|
|
if (!g_pGameConf->GetMemSig(signature, &patch->base) || patch->base == NULL)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter could not find signature: %s", signature);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *value;
|
|
|
|
char keyname[255];
|
|
|
|
|
|
|
|
/* Get the verification bytes that will be written over. */
|
|
|
|
UTIL_Format(keyname, sizeof(keyname), "%s_Patch_%s", name, platform);
|
|
|
|
if ((value = g_pGameConf->GetKeyValue(keyname)) == NULL)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter could not find key: %s", keyname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
patch->search_len = UTIL_DecodeHexString(patch->search, sizeof(patch->search), value);
|
|
|
|
if (patch->search_len < 6)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Error decoding %s value, or not long enough", keyname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the offset into the function. */
|
|
|
|
UTIL_Format(keyname, sizeof(keyname), "%s_Offset_%s", name, platform);
|
|
|
|
if ((value = g_pGameConf->GetKeyValue(keyname)) == NULL)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter could not find key: %s", keyname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
patch->offset = atoi(value);
|
|
|
|
if (patch->offset > 20000)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter %s value is bogus", keyname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the number of bytes to save from what was written over. */
|
|
|
|
patch->saveat = 0;
|
|
|
|
UTIL_Format(keyname, sizeof(keyname), "%s_Save_%s", name, platform);
|
|
|
|
if ((value = g_pGameConf->GetKeyValue(keyname)) != NULL)
|
|
|
|
{
|
|
|
|
patch->saveat = atoi(value);
|
|
|
|
if (patch->saveat >= patch->search_len)
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter %s value is too large", keyname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get register for parameter, if any. */
|
|
|
|
patch->regparam = -1;
|
|
|
|
UTIL_Format(keyname, sizeof(keyname), "%s_Reg_%s", name, platform);
|
|
|
|
if ((value = g_pGameConf->GetKeyValue(keyname)) != NULL)
|
|
|
|
{
|
|
|
|
patch->regparam = atoi(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Everything loaded from gamedata, make sure the patch will succeed. */
|
|
|
|
unsigned char *address = (unsigned char *)patch->base + patch->offset;
|
|
|
|
for (size_t i = 0; i < patch->search_len; i++)
|
|
|
|
{
|
|
|
|
if (address[i] != patch->search[i])
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Command filter %s has changed (byte %x is not %x sub-offset %d)",
|
|
|
|
name, address[i], patch->search[i], i);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool dummy_hook_set = false;
|
|
|
|
void DummyHook()
|
|
|
|
{
|
|
|
|
if (dummy_hook_set)
|
|
|
|
{
|
|
|
|
dummy_hook_set = false;
|
|
|
|
RETURN_META(MRES_SUPERCEDE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* BEGIN THE ACTUALLY GENERIC CODE.
|
|
|
|
*/
|
|
|
|
|
2009-10-28 08:37:34 +01:00
|
|
|
#define FEATURECAP_COMMANDLISTENER "command listener"
|
|
|
|
|
2009-09-26 23:14:50 +02:00
|
|
|
static GenericCommandHooker s_GenericHooker;
|
|
|
|
ConsoleDetours g_ConsoleDetours;
|
|
|
|
|
2009-10-28 08:37:34 +01:00
|
|
|
ConsoleDetours::ConsoleDetours() : status(FeatureStatus_Unknown)
|
2009-09-26 23:14:50 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConsoleDetours::OnSourceModAllInitialized()
|
|
|
|
{
|
2013-10-09 14:43:08 +02:00
|
|
|
m_pForward = forwardsys->CreateForwardEx("OnAnyCommand", ET_Hook, 3, NULL, Param_Cell,
|
2009-09-26 23:14:50 +02:00
|
|
|
Param_String, Param_Cell);
|
2013-03-29 19:37:29 +01:00
|
|
|
sharesys->AddCapabilityProvider(NULL, this, FEATURECAP_COMMANDLISTENER);
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConsoleDetours::OnSourceModShutdown()
|
|
|
|
{
|
2013-08-25 21:15:26 +02:00
|
|
|
for (StringHashMap<IChangeableForward *>::iterator iter = m_Listeners.iter();
|
|
|
|
!iter.empty();
|
|
|
|
iter.next())
|
2009-09-26 23:14:50 +02:00
|
|
|
{
|
2013-10-09 14:43:08 +02:00
|
|
|
forwardsys->ReleaseForward(iter->value);
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
|
|
|
|
2013-10-09 14:43:08 +02:00
|
|
|
forwardsys->ReleaseForward(m_pForward);
|
2009-09-26 23:14:50 +02:00
|
|
|
s_GenericHooker.Disable();
|
|
|
|
}
|
|
|
|
|
2009-10-28 08:37:34 +01:00
|
|
|
FeatureStatus ConsoleDetours::GetFeatureStatus(FeatureType type, const char *name)
|
|
|
|
{
|
|
|
|
return GetStatus();
|
|
|
|
}
|
|
|
|
|
|
|
|
FeatureStatus ConsoleDetours::GetStatus()
|
2009-09-26 23:14:50 +02:00
|
|
|
{
|
2009-10-28 08:37:34 +01:00
|
|
|
if (status == FeatureStatus_Unknown)
|
|
|
|
{
|
|
|
|
status = s_GenericHooker.Enable() ? FeatureStatus_Available : FeatureStatus_Unavailable;
|
|
|
|
}
|
|
|
|
return status;
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ConsoleDetours::AddListener(IPluginFunction *fun, const char *command)
|
|
|
|
{
|
2009-10-28 08:37:34 +01:00
|
|
|
if (GetStatus() != FeatureStatus_Available)
|
2009-09-26 23:14:50 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if (command == NULL)
|
|
|
|
{
|
|
|
|
m_pForward->AddFunction(fun);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-08-25 21:15:26 +02:00
|
|
|
ke::AutoArray<char> str(UTIL_ToLowerCase(command));
|
|
|
|
IChangeableForward *forward;
|
|
|
|
if (!m_Listeners.retrieve(str, &forward))
|
2009-09-26 23:14:50 +02:00
|
|
|
{
|
2013-10-09 14:43:08 +02:00
|
|
|
forward = forwardsys->CreateForwardEx(NULL, ET_Hook, 3, NULL, Param_Cell,
|
2013-08-25 21:15:26 +02:00
|
|
|
Param_String, Param_Cell);
|
|
|
|
m_Listeners.insert(str, forward);
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
2013-08-25 21:15:26 +02:00
|
|
|
forward->AddFunction(fun);
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ConsoleDetours::RemoveListener(IPluginFunction *fun, const char *command)
|
|
|
|
{
|
|
|
|
if (command == NULL)
|
|
|
|
{
|
|
|
|
return m_pForward->RemoveFunction(fun);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-08-25 21:15:26 +02:00
|
|
|
ke::AutoArray<char> str(UTIL_ToLowerCase(command));
|
|
|
|
IChangeableForward *forward;
|
|
|
|
if (!m_Listeners.retrieve(str, &forward))
|
2009-09-26 23:14:50 +02:00
|
|
|
return false;
|
2013-08-25 21:15:26 +02:00
|
|
|
return forward->RemoveFunction(fun);
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cell_t ConsoleDetours::InternalDispatch(int client, const CCommand& args)
|
|
|
|
{
|
|
|
|
char name[255];
|
|
|
|
const char *realname = args.Arg(0);
|
|
|
|
size_t len = strlen(realname);
|
2012-04-14 01:50:44 +02:00
|
|
|
|
|
|
|
// Disallow command strings that are too long, for now.
|
|
|
|
if (len >= sizeof(name) - 1)
|
|
|
|
return Pl_Continue;
|
|
|
|
|
2009-09-26 23:14:50 +02:00
|
|
|
for (size_t i = 0; i < len; i++)
|
|
|
|
{
|
|
|
|
if (realname[i] >= 'A' && realname[i] <= 'Z')
|
|
|
|
name[i] = tolower(realname[i]);
|
|
|
|
else
|
|
|
|
name[i] = realname[i];
|
|
|
|
}
|
|
|
|
name[len] = '\0';
|
|
|
|
|
|
|
|
cell_t result = Pl_Continue;
|
|
|
|
m_pForward->PushCell(client);
|
|
|
|
m_pForward->PushString(name);
|
|
|
|
m_pForward->PushCell(args.ArgC() - 1);
|
|
|
|
m_pForward->Execute(&result, NULL);
|
|
|
|
|
|
|
|
/* Don't let plugins block this. */
|
|
|
|
if (strcmp(name, "sm") == 0)
|
|
|
|
result = Pl_Continue;
|
|
|
|
|
2013-08-25 05:56:39 +02:00
|
|
|
if (result >= Pl_Handled)
|
2009-09-26 23:14:50 +02:00
|
|
|
return result;
|
|
|
|
|
2013-08-25 21:15:26 +02:00
|
|
|
IChangeableForward *forward;
|
|
|
|
if (!m_Listeners.retrieve(name, &forward))
|
2009-09-26 23:14:50 +02:00
|
|
|
return result;
|
2013-08-25 21:15:26 +02:00
|
|
|
if (forward->GetFunctionCount() == 0)
|
2009-09-26 23:14:50 +02:00
|
|
|
return result;
|
|
|
|
|
|
|
|
cell_t result2 = Pl_Continue;
|
2013-08-25 21:15:26 +02:00
|
|
|
forward->PushCell(client);
|
|
|
|
forward->PushString(name);
|
|
|
|
forward->PushCell(args.ArgC() - 1);
|
|
|
|
forward->Execute(&result2, NULL);
|
2009-09-26 23:14:50 +02:00
|
|
|
|
|
|
|
if (result2 > result)
|
|
|
|
result = result2;
|
|
|
|
|
|
|
|
/* "sm" should not have flown through the above. */
|
|
|
|
assert(strcmp(name, "sm") != 0 || result == Pl_Continue);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SOURCE_ENGINE >= SE_ORANGEBOX
|
|
|
|
cell_t ConsoleDetours::Dispatch(ConCommand *pBase, const CCommand& args)
|
|
|
|
{
|
|
|
|
#else
|
|
|
|
cell_t ConsoleDetours::Dispatch(ConCommand *pBase)
|
|
|
|
{
|
|
|
|
CCommand args;
|
|
|
|
#endif
|
|
|
|
g_HL2.PushCommandStack(&args);
|
|
|
|
cell_t res = g_ConsoleDetours.InternalDispatch(g_ConCmds.GetCommandClient(), args);
|
|
|
|
g_HL2.PopCommandStack();
|
|
|
|
|
|
|
|
#if SH_IMPL_VERSION < 4
|
|
|
|
if (res >= Pl_Handled)
|
|
|
|
{
|
|
|
|
/* See bug 4020 - we can't optimize this because looking at the vtable
|
|
|
|
* is probably more expensive, since there's no SH_GET_ORIG_VFNPTR_ENTRY.
|
|
|
|
*/
|
2013-03-17 03:15:25 +01:00
|
|
|
SH_ADD_HOOK(ConCommand, Dispatch, pBase, SH_STATIC(DummyHook), false);
|
2009-09-26 23:14:50 +02:00
|
|
|
dummy_hook_set = true;
|
|
|
|
pBase->Dispatch();
|
2013-03-17 03:15:25 +01:00
|
|
|
SH_REMOVE_HOOK(ConCommand, Dispatch, pBase, SH_STATIC(DummyHook), false);
|
2009-09-26 23:14:50 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Make sure the command gets invoked. See bug 4019 on making this better. */
|
|
|
|
pBase->Dispatch();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|