2008-03-30 09:00:22 +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>.
|
|
|
|
*
|
|
|
|
* Version: $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <time.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "sm_stringutil.h"
|
|
|
|
#include "sm_globals.h"
|
|
|
|
#include "sourcemod.h"
|
|
|
|
#include "LibrarySys.h"
|
|
|
|
#include "TimerSys.h"
|
|
|
|
#include "ForwardSys.h"
|
|
|
|
#include "Logger.h"
|
2012-04-14 01:31:07 +02:00
|
|
|
#include <sh_memory.h>
|
2013-03-29 19:37:29 +01:00
|
|
|
#include "logic_bridge.h"
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
#if defined PLATFORM_WINDOWS
|
|
|
|
#include <windows.h>
|
2010-05-30 00:11:47 +02:00
|
|
|
#elif defined PLATFORM_POSIX
|
2008-03-30 09:00:22 +02:00
|
|
|
#include <limits.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/times.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
HandleType_t g_PlIter;
|
|
|
|
ConVar sm_datetime_format("sm_datetime_format", "%m/%d/%Y - %H:%M:%S", 0, "Default formatting time rules");
|
|
|
|
IForward *g_OnLogAction = NULL;
|
|
|
|
|
|
|
|
class CoreNativeHelpers :
|
|
|
|
public SMGlobalClass,
|
|
|
|
public IHandleTypeDispatch
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void OnSourceModAllInitialized()
|
|
|
|
{
|
|
|
|
HandleAccess hacc;
|
2013-03-29 19:37:29 +01:00
|
|
|
handlesys->InitAccessDefaults(NULL, &hacc);
|
2008-03-30 09:00:22 +02:00
|
|
|
hacc.access[HandleAccess_Clone] = HANDLE_RESTRICT_IDENTITY|HANDLE_RESTRICT_OWNER;
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
g_PlIter = handlesys->CreateType("PluginIterator", this, 0, NULL, NULL, g_pCoreIdent, NULL);
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
g_OnLogAction = g_Forwards.CreateForward("OnLogAction",
|
|
|
|
ET_Hook,
|
|
|
|
5,
|
|
|
|
NULL,
|
|
|
|
Param_Cell,
|
|
|
|
Param_Cell,
|
|
|
|
Param_Cell,
|
|
|
|
Param_Cell,
|
|
|
|
Param_String);
|
|
|
|
}
|
|
|
|
void OnHandleDestroy(HandleType_t type, void *object)
|
|
|
|
{
|
|
|
|
IPluginIterator *iter = (IPluginIterator *)object;
|
|
|
|
iter->Release();
|
|
|
|
}
|
|
|
|
void OnSourceModShutdown()
|
|
|
|
{
|
|
|
|
g_Forwards.ReleaseForward(g_OnLogAction);
|
2013-03-29 19:37:29 +01:00
|
|
|
handlesys->RemoveType(g_PlIter, g_pCoreIdent);
|
2008-03-30 09:00:22 +02:00
|
|
|
}
|
|
|
|
} g_CoreNativeHelpers;
|
|
|
|
|
|
|
|
void LogAction(Handle_t hndl, int type, int client, int target, const char *message)
|
|
|
|
{
|
|
|
|
if (g_OnLogAction->GetFunctionCount())
|
|
|
|
{
|
|
|
|
cell_t result = 0;
|
|
|
|
g_OnLogAction->PushCell(hndl);
|
|
|
|
g_OnLogAction->PushCell(type);
|
|
|
|
g_OnLogAction->PushCell(client);
|
|
|
|
g_OnLogAction->PushCell(target);
|
|
|
|
g_OnLogAction->PushString(message);
|
|
|
|
g_OnLogAction->Execute(&result);
|
|
|
|
|
|
|
|
if (result >= (ResultType)Pl_Handled)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *logtag = "SM";
|
|
|
|
if (type == 2)
|
|
|
|
{
|
|
|
|
HandleError err;
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = scripts->FindPluginByHandle(hndl, &err);
|
2008-03-30 09:00:22 +02:00
|
|
|
if (pPlugin)
|
|
|
|
{
|
|
|
|
logtag = pPlugin->GetFilename();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_Logger.LogMessage("[%s] %s", logtag, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t ThrowError(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char buffer[512];
|
|
|
|
|
2008-05-10 11:23:55 +02:00
|
|
|
g_SourceMod.SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE);
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
g_SourceMod.FormatString(buffer, sizeof(buffer), pContext, params, 1);
|
|
|
|
|
2008-07-11 10:18:43 +02:00
|
|
|
if (pContext->GetLastNativeError() == SP_ERROR_NONE)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
pContext->ThrowNativeErrorEx(SP_ERROR_ABORTED, "%s", buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t GetTime(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
time_t t = GetAdjustedTime();
|
|
|
|
cell_t *addr;
|
|
|
|
pContext->LocalToPhysAddr(params[1], &addr);
|
|
|
|
|
|
|
|
*(time_t *)addr = t;
|
|
|
|
|
|
|
|
return static_cast<cell_t>(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined SUBPLATFORM_SECURECRT
|
|
|
|
void _ignore_invalid_parameter(
|
|
|
|
const wchar_t * expression,
|
|
|
|
const wchar_t * function,
|
|
|
|
const wchar_t * file,
|
|
|
|
unsigned int line,
|
|
|
|
uintptr_t pReserved
|
|
|
|
)
|
|
|
|
{
|
|
|
|
/* Wow we don't care, thanks Microsoft. */
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static cell_t FormatTime(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *format, *buffer;
|
|
|
|
pContext->LocalToString(params[1], &buffer);
|
|
|
|
pContext->LocalToStringNULL(params[3], &format);
|
|
|
|
|
|
|
|
if (format == NULL)
|
|
|
|
{
|
|
|
|
format = const_cast<char *>(sm_datetime_format.GetString());
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined SUBPLATFORM_SECURECRT
|
|
|
|
_invalid_parameter_handler handler = _set_invalid_parameter_handler(_ignore_invalid_parameter);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
time_t t = (params[4] == -1) ? GetAdjustedTime() : (time_t)params[4];
|
|
|
|
size_t written = strftime(buffer, params[2], format, localtime(&t));
|
|
|
|
|
|
|
|
#if defined SUBPLATFORM_SECURECRT
|
|
|
|
_set_invalid_parameter_handler(handler);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (params[2] && format[0] != '\0' && !written)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Invalid time format or buffer too small");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t GetPluginIterator(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
IPluginIterator *iter = scripts->GetPluginIterator();
|
2008-03-30 09:00:22 +02:00
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
Handle_t hndl = handlesys->CreateHandle(g_PlIter, iter, pContext->GetIdentity(), g_pCoreIdent, NULL);
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
if (hndl == BAD_HANDLE)
|
|
|
|
{
|
|
|
|
iter->Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return hndl;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t MorePlugins(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
Handle_t hndl = (Handle_t)params[1];
|
|
|
|
HandleError err;
|
|
|
|
IPluginIterator *pIter;
|
|
|
|
|
|
|
|
HandleSecurity sec;
|
|
|
|
sec.pIdentity = g_pCoreIdent;
|
|
|
|
sec.pOwner = pContext->GetIdentity();
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
if ((err=handlesys->ReadHandle(hndl, g_PlIter, &sec, (void **)&pIter)) != HandleError_None)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pIter->MorePlugins() ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t ReadPlugin(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
Handle_t hndl = (Handle_t)params[1];
|
|
|
|
HandleError err;
|
|
|
|
IPluginIterator *pIter;
|
|
|
|
|
|
|
|
HandleSecurity sec;
|
|
|
|
sec.pIdentity = g_pCoreIdent;
|
|
|
|
sec.pOwner = pContext->GetIdentity();
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
if ((err=handlesys->ReadHandle(hndl, g_PlIter, &sec, (void **)&pIter)) != HandleError_None)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
return pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err);
|
|
|
|
}
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = pIter->GetPlugin();
|
2008-03-30 09:00:22 +02:00
|
|
|
if (!pPlugin)
|
|
|
|
{
|
|
|
|
return BAD_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
pIter->NextPlugin();
|
|
|
|
|
|
|
|
return pPlugin->GetMyHandle();
|
|
|
|
}
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *GetPluginFromHandle(IPluginContext *pContext, Handle_t hndl)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
if (hndl == BAD_HANDLE)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
return scripts->FindPluginByContext(pContext->GetContext());
|
2008-03-30 09:00:22 +02:00
|
|
|
} else {
|
|
|
|
HandleError err;
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = scripts->FindPluginByHandle(hndl, &err);
|
2008-03-30 09:00:22 +02:00
|
|
|
if (!pPlugin)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Could not read Handle %x (error %d)", hndl, err);
|
|
|
|
}
|
|
|
|
return pPlugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t GetPluginStatus(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]);
|
2008-03-30 09:00:22 +02:00
|
|
|
if (!pPlugin)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pPlugin->GetStatus();
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t GetPluginFilename(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]);
|
2008-03-30 09:00:22 +02:00
|
|
|
if (!pPlugin)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
pContext->StringToLocalUTF8(params[2], params[3], pPlugin->GetFilename(), NULL);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t IsPluginDebugging(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]);
|
2008-03-30 09:00:22 +02:00
|
|
|
if (!pPlugin)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pPlugin->IsDebugging() ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Local to plugins only */
|
|
|
|
enum PluginInfo
|
|
|
|
{
|
|
|
|
PlInfo_Name, /**< Plugin name */
|
|
|
|
PlInfo_Author, /**< Plugin author */
|
|
|
|
PlInfo_Description, /**< Plugin description */
|
|
|
|
PlInfo_Version, /**< Plugin verison */
|
|
|
|
PlInfo_URL, /**< Plugin URL */
|
|
|
|
};
|
|
|
|
|
|
|
|
static cell_t GetPluginInfo(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = GetPluginFromHandle(pContext, params[1]);
|
2008-03-30 09:00:22 +02:00
|
|
|
if (!pPlugin)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const sm_plugininfo_t *info = pPlugin->GetPublicInfo();
|
|
|
|
|
|
|
|
if (!info)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *str = NULL;
|
|
|
|
|
|
|
|
switch ((PluginInfo)params[2])
|
|
|
|
{
|
|
|
|
case PlInfo_Name:
|
|
|
|
{
|
|
|
|
str = info->name;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case PlInfo_Author:
|
|
|
|
{
|
|
|
|
str = info->author;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case PlInfo_Description:
|
|
|
|
{
|
|
|
|
str = info->description;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case PlInfo_Version:
|
|
|
|
{
|
|
|
|
str = info->version;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case PlInfo_URL:
|
|
|
|
{
|
|
|
|
str = info->url;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!str || str[0] == '\0')
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
pContext->StringToLocalUTF8(params[3], params[4], str, NULL);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t SetFailState(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *str;
|
2013-03-29 19:37:29 +01:00
|
|
|
SMPlugin *pPlugin;
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
pContext->LocalToString(params[1], &str);
|
2013-03-29 19:37:29 +01:00
|
|
|
pPlugin = scripts->FindPluginByContext(pContext->GetContext());
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
if (params[0] == 1)
|
|
|
|
{
|
|
|
|
pPlugin->SetErrorState(Plugin_Error, "%s", str);
|
|
|
|
|
|
|
|
return pContext->ThrowNativeErrorEx(SP_ERROR_ABORTED, "%s", str);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
char buffer[2048];
|
|
|
|
|
|
|
|
g_SourceMod.FormatString(buffer, sizeof(buffer), pContext, params, 1);
|
2008-07-11 10:18:43 +02:00
|
|
|
if (pContext->GetLastNativeError() != SP_ERROR_NONE)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
pPlugin->SetErrorState(Plugin_Error, "%s", str);
|
|
|
|
return pContext->ThrowNativeErrorEx(SP_ERROR_ABORTED, "Formatting error (%s)", str);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pPlugin->SetErrorState(Plugin_Error, "%s", buffer);
|
|
|
|
return pContext->ThrowNativeErrorEx(SP_ERROR_ABORTED, "%s", buffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t GetSysTickCount(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
#if defined PLATFORM_WINDOWS
|
|
|
|
return (cell_t)GetTickCount();
|
2010-05-30 00:11:47 +02:00
|
|
|
#elif defined PLATFORM_POSIX
|
2008-03-30 09:00:22 +02:00
|
|
|
tms tm;
|
|
|
|
clock_t ticks = times(&tm);
|
|
|
|
long ticks_per_sec = sysconf(_SC_CLK_TCK);
|
|
|
|
double fticks = (double)ticks / (double)ticks_per_sec;
|
|
|
|
fticks *= 1000.0f;
|
|
|
|
if (fticks > INT_MAX)
|
|
|
|
{
|
|
|
|
double r = (int)(fticks / INT_MAX) * (double)INT_MAX;
|
|
|
|
fticks -= r;
|
|
|
|
}
|
|
|
|
return (cell_t)fticks;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t AutoExecConfig(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
SMPlugin *plugin = scripts->FindPluginByContext(pContext->GetContext());
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
char *cfg, *folder;
|
|
|
|
pContext->LocalToString(params[2], &cfg);
|
|
|
|
pContext->LocalToString(params[3], &folder);
|
|
|
|
|
|
|
|
if (cfg[0] == '\0')
|
|
|
|
{
|
|
|
|
static char temp_str[255];
|
|
|
|
static char temp_file[PLATFORM_MAX_PATH];
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
g_LibSys.GetFileFromPath(temp_str, sizeof(temp_str), plugin->GetFilename());
|
|
|
|
if ((ptr = strstr(temp_str, ".smx")) != NULL)
|
|
|
|
{
|
|
|
|
*ptr = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have the raw filename! */
|
|
|
|
UTIL_Format(temp_file, sizeof(temp_file), "plugin.%s", temp_str);
|
|
|
|
cfg = temp_file;
|
|
|
|
}
|
|
|
|
|
|
|
|
plugin->AddConfig(params[1] ? true : false, cfg, folder);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t MarkNativeAsOptional(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
uint32_t idx;
|
2008-07-11 10:18:43 +02:00
|
|
|
sp_native_t *native;
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
pContext->LocalToString(params[1], &name);
|
|
|
|
if (pContext->FindNativeByName(name, &idx) != SP_ERROR_NONE)
|
|
|
|
{
|
|
|
|
/* Oops! This HAS to silently fail! */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-07-11 10:18:43 +02:00
|
|
|
pContext->GetNativeByIndex(idx, &native);
|
|
|
|
|
|
|
|
native->flags |= SP_NTVFLAG_OPTIONAL;
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t RegPluginLibrary(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *name;
|
2013-03-29 19:37:29 +01:00
|
|
|
SMPlugin *pl = scripts->FindPluginByContext(pContext->GetContext());
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
pContext->LocalToString(params[1], &name);
|
|
|
|
|
|
|
|
pl->AddLibrary(name);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t LibraryExists(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
pContext->LocalToString(params[1], &str);
|
|
|
|
|
2009-10-28 08:37:34 +01:00
|
|
|
if (strcmp(str, "__CanTestFeatures__") == 0)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
if (scripts->LibraryExists(str))
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
if (extsys->LibraryExists(str))
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t sm_LogAction(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char buffer[2048];
|
2008-05-10 11:23:55 +02:00
|
|
|
g_SourceMod.SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE);
|
2008-03-30 09:00:22 +02:00
|
|
|
g_SourceMod.FormatString(buffer, sizeof(buffer), pContext, params, 3);
|
|
|
|
|
2008-07-11 10:18:43 +02:00
|
|
|
if (pContext->GetLastNativeError() != SP_ERROR_NONE)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = scripts->FindPluginByContext(pContext->GetContext());
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
LogAction(pPlugin->GetMyHandle(), 2, params[1], params[2], buffer);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t LogToFile(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *file;
|
|
|
|
pContext->LocalToString(params[1], &file);
|
|
|
|
|
|
|
|
char path[PLATFORM_MAX_PATH];
|
|
|
|
g_SourceMod.BuildPath(Path_Game, path, sizeof(path), "%s", file);
|
|
|
|
|
|
|
|
FILE *fp = fopen(path, "at");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
return pContext->ThrowNativeError("Could not open file \"%s\"", path);
|
|
|
|
}
|
|
|
|
|
|
|
|
char buffer[2048];
|
2008-05-10 11:23:55 +02:00
|
|
|
g_SourceMod.SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE);
|
2008-03-30 09:00:22 +02:00
|
|
|
g_SourceMod.FormatString(buffer, sizeof(buffer), pContext, params, 2);
|
|
|
|
|
2008-07-11 10:18:43 +02:00
|
|
|
if (pContext->GetLastNativeError() != SP_ERROR_NONE)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = scripts->FindPluginByContext(pContext->GetContext());
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
g_Logger.LogToOpenFile(fp, "[%s] %s", pPlugin->GetFilename(), buffer);
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t LogToFileEx(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *file;
|
|
|
|
pContext->LocalToString(params[1], &file);
|
|
|
|
|
|
|
|
char path[PLATFORM_MAX_PATH];
|
|
|
|
g_SourceMod.BuildPath(Path_Game, path, sizeof(path), "%s", file);
|
|
|
|
|
|
|
|
FILE *fp = fopen(path, "at");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
return pContext->ThrowNativeError("Could not open file \"%s\"", path);
|
|
|
|
}
|
|
|
|
|
|
|
|
char buffer[2048];
|
2008-05-10 11:23:55 +02:00
|
|
|
g_SourceMod.SetGlobalTarget(SOURCEMOD_SERVER_LANGUAGE);
|
2008-03-30 09:00:22 +02:00
|
|
|
g_SourceMod.FormatString(buffer, sizeof(buffer), pContext, params, 2);
|
|
|
|
|
2008-07-11 10:18:43 +02:00
|
|
|
if (pContext->GetLastNativeError() != SP_ERROR_NONE)
|
2008-03-30 09:00:22 +02:00
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_Logger.LogToOpenFile(fp, "%s", buffer);
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t GetExtensionFileStatus(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
pContext->LocalToString(params[1], &str);
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
IExtension *pExtension = extsys->FindExtensionByFile(str);
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
if (!pExtension)
|
|
|
|
{
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pExtension->IsLoaded())
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *error;
|
|
|
|
pContext->LocalToString(params[2], &error);
|
|
|
|
if (!pExtension->IsRunning(error, params[3]))
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t FindPluginByNumber(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
2013-03-29 19:37:29 +01:00
|
|
|
IPlugin *pPlugin = scripts->FindPluginByOrder(params[1]);
|
2008-03-30 09:00:22 +02:00
|
|
|
|
|
|
|
if (pPlugin == NULL)
|
|
|
|
{
|
|
|
|
return BAD_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pPlugin->GetMyHandle();
|
|
|
|
}
|
|
|
|
|
2008-10-06 00:20:17 +02:00
|
|
|
static cell_t VerifyCoreVersion(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
|
2009-10-28 08:37:34 +01:00
|
|
|
static cell_t GetFeatureStatus(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
FeatureType type = (FeatureType)params[1];
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
pContext->LocalToString(params[2], &name);
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
return sharesys->TestFeature(pContext->GetRuntime(), type, name);
|
2009-10-28 08:37:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static cell_t RequireFeature(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
FeatureType type = (FeatureType)params[1];
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
pContext->LocalToString(params[2], &name);
|
|
|
|
|
2013-03-29 19:37:29 +01:00
|
|
|
if (sharesys->TestFeature(pContext->GetRuntime(), type, name) != FeatureStatus_Available)
|
2009-10-28 08:37:34 +01:00
|
|
|
{
|
|
|
|
char buffer[255];
|
|
|
|
char *msg = buffer;
|
|
|
|
char default_message[255];
|
2013-03-29 19:37:29 +01:00
|
|
|
SMPlugin *pPlugin = scripts->FindPluginByContext(pContext->GetContext());
|
2009-10-28 08:37:34 +01:00
|
|
|
|
|
|
|
g_SourceMod.FormatString(buffer, sizeof(buffer), pContext, params, 3);
|
|
|
|
if (pContext->GetLastNativeError() != SP_ERROR_NONE || buffer[0] == '\0')
|
|
|
|
{
|
|
|
|
UTIL_Format(default_message, sizeof(default_message), "Feature \"%s\" not available", name);
|
|
|
|
msg = default_message;
|
|
|
|
}
|
|
|
|
pPlugin->SetErrorState(Plugin_Error, "%s", msg);
|
|
|
|
return pContext->ThrowNativeErrorEx(SP_ERROR_ABORTED, "%s", msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2010-01-12 07:46:44 +01:00
|
|
|
enum NumberType
|
|
|
|
{
|
|
|
|
NumberType_Int8,
|
|
|
|
NumberType_Int16,
|
|
|
|
NumberType_Int32
|
|
|
|
};
|
|
|
|
|
|
|
|
//memory addresses below 0x10000 are automatically considered invalid for dereferencing
|
|
|
|
#define VALID_MINIMUM_MEMORY_ADDRESS 0x10000
|
|
|
|
|
|
|
|
static cell_t LoadFromAddress(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
void *addr = reinterpret_cast<void*>(params[1]);
|
|
|
|
|
|
|
|
if (addr == NULL)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Address cannot be null");
|
|
|
|
}
|
|
|
|
else if (reinterpret_cast<uintptr_t>(addr) < VALID_MINIMUM_MEMORY_ADDRESS)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Invalid address 0x%x is pointing to reserved memory.", addr);
|
|
|
|
}
|
|
|
|
NumberType size = static_cast<NumberType>(params[2]);
|
|
|
|
|
|
|
|
switch(size)
|
|
|
|
{
|
|
|
|
case NumberType_Int8:
|
|
|
|
return *reinterpret_cast<uint8_t*>(addr);
|
|
|
|
case NumberType_Int16:
|
|
|
|
return *reinterpret_cast<uint16_t*>(addr);
|
|
|
|
case NumberType_Int32:
|
|
|
|
return *reinterpret_cast<uint32_t*>(addr);
|
|
|
|
default:
|
|
|
|
pContext->ThrowNativeError("Invalid number types %d", size);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static cell_t StoreToAddress(IPluginContext *pContext, const cell_t *params)
|
|
|
|
{
|
|
|
|
void *addr = reinterpret_cast<void*>(params[1]);
|
|
|
|
|
|
|
|
if (addr == NULL)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Address cannot be null");
|
|
|
|
}
|
|
|
|
else if (reinterpret_cast<uintptr_t>(addr) < VALID_MINIMUM_MEMORY_ADDRESS)
|
|
|
|
{
|
|
|
|
pContext->ThrowNativeError("Invalid address 0x%x is pointing to reserved memory.", addr);
|
|
|
|
}
|
|
|
|
cell_t data = params[2];
|
|
|
|
|
|
|
|
NumberType size = static_cast<NumberType>(params[3]);
|
|
|
|
|
|
|
|
switch(size)
|
|
|
|
{
|
|
|
|
case NumberType_Int8:
|
2012-04-14 01:31:07 +02:00
|
|
|
SourceHook::SetMemAccess(addr, sizeof(uint8_t), SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC);
|
2010-01-12 07:46:44 +01:00
|
|
|
*reinterpret_cast<uint8_t*>(addr) = data;
|
2012-03-12 00:07:24 +01:00
|
|
|
break;
|
2010-01-12 07:46:44 +01:00
|
|
|
case NumberType_Int16:
|
2012-04-14 01:31:07 +02:00
|
|
|
SourceHook::SetMemAccess(addr, sizeof(uint16_t), SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC);
|
2010-01-12 07:46:44 +01:00
|
|
|
*reinterpret_cast<uint16_t*>(addr) = data;
|
2012-03-12 00:07:24 +01:00
|
|
|
break;
|
2010-01-12 07:46:44 +01:00
|
|
|
case NumberType_Int32:
|
2012-04-14 01:31:07 +02:00
|
|
|
SourceHook::SetMemAccess(addr, sizeof(uint32_t), SH_MEM_READ|SH_MEM_WRITE|SH_MEM_EXEC);
|
2010-01-12 07:46:44 +01:00
|
|
|
*reinterpret_cast<uint32_t*>(addr) = data;
|
2012-03-12 00:07:24 +01:00
|
|
|
break;
|
2010-01-12 07:46:44 +01:00
|
|
|
default:
|
|
|
|
pContext->ThrowNativeError("Invalid number types %d", size);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2008-03-30 09:00:22 +02:00
|
|
|
REGISTER_NATIVES(coreNatives)
|
|
|
|
{
|
|
|
|
{"AutoExecConfig", AutoExecConfig},
|
|
|
|
{"GetPluginFilename", GetPluginFilename},
|
|
|
|
{"GetPluginInfo", GetPluginInfo},
|
|
|
|
{"GetPluginIterator", GetPluginIterator},
|
|
|
|
{"GetPluginStatus", GetPluginStatus},
|
|
|
|
{"GetSysTickCount", GetSysTickCount},
|
|
|
|
{"GetTime", GetTime},
|
|
|
|
{"IsPluginDebugging", IsPluginDebugging},
|
|
|
|
{"MorePlugins", MorePlugins},
|
|
|
|
{"ReadPlugin", ReadPlugin},
|
|
|
|
{"ThrowError", ThrowError},
|
|
|
|
{"SetFailState", SetFailState},
|
|
|
|
{"FormatTime", FormatTime},
|
|
|
|
{"MarkNativeAsOptional", MarkNativeAsOptional},
|
|
|
|
{"RegPluginLibrary", RegPluginLibrary},
|
|
|
|
{"LibraryExists", LibraryExists},
|
|
|
|
{"LogAction", sm_LogAction},
|
|
|
|
{"LogToFile", LogToFile},
|
|
|
|
{"LogToFileEx", LogToFileEx},
|
|
|
|
{"GetExtensionFileStatus", GetExtensionFileStatus},
|
|
|
|
{"FindPluginByNumber", FindPluginByNumber},
|
2008-10-06 00:20:17 +02:00
|
|
|
{"VerifyCoreVersion", VerifyCoreVersion},
|
2009-10-28 08:37:34 +01:00
|
|
|
{"GetFeatureStatus", GetFeatureStatus},
|
|
|
|
{"RequireFeature", RequireFeature},
|
2010-01-12 07:46:44 +01:00
|
|
|
{"LoadFromAddress", LoadFromAddress},
|
|
|
|
{"StoreToAddress", StoreToAddress},
|
2008-03-30 09:00:22 +02:00
|
|
|
{NULL, NULL},
|
|
|
|
};
|
|
|
|
|