2
0
mirror of https://github.com/rehlds/metamod-r.git synced 2025-01-16 00:28:07 +03:00
metamod-r/metamod/src/mutil.cpp

410 lines
9.3 KiB
C++
Raw Normal View History

2016-07-26 07:22:47 +07:00
#include "precompiled.h"
2016-07-04 12:07:29 +06:00
static hudtextparms_t g_default_csay_tparms = {
-1, 0.25, // x, y
2, // effect
0, 255, 0, 0, // r, g, b, a1
0, 0, 0, 0, // r2, g2, b2, a2
0, 0, 10, 10, // fadein, fadeout, hold, fxtime
1 // channel
2016-07-04 12:07:29 +06:00
};
static const char* g_engine_msg_names[] =
{
"svc_bad",
"svc_nop",
"svc_disconnect",
"svc_event",
"svc_version",
"svc_setview",
"svc_sound",
"svc_time",
"svc_print",
"svc_stufftext",
"svc_setangle",
"svc_serverinfo",
"svc_lightstyle",
"svc_updateuserinfo",
"svc_deltadescription",
"svc_clientdata",
"svc_stopsound",
"svc_pings",
"svc_particle",
"svc_damage",
"svc_spawnstatic",
"svc_event_reliable",
"svc_spawnbaseline",
"svc_temp_entity",
"svc_setpause",
"svc_signonnum",
"svc_centerprint",
"svc_killedmonster",
"svc_foundsecret",
"svc_spawnstaticsound",
"svc_intermission",
"svc_finale",
"svc_cdtrack",
"svc_restore",
"svc_cutscene",
"svc_weaponanim",
"svc_decalname",
"svc_roomtype",
"svc_addangle",
"svc_newusermsg",
"svc_packetentities",
"svc_deltapacketentities",
"svc_choke",
"svc_resourcelist",
"svc_newmovevars",
"svc_resourcerequest",
"svc_customization",
"svc_crosshairangle",
"svc_soundfade",
"svc_filetxferfailed",
"svc_hltv",
"svc_director",
"svc_voiceinit",
"svc_voicedata",
"svc_sendextrainfo",
"svc_timescale",
"svc_resourcelocation",
"svc_sendcvarvalue",
"svc_sendcvarvalue2"
};
2016-07-04 12:07:29 +06:00
// Log to console; newline added.
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_LogConsole(plid_t plid, const char *fmt, ...)
{
2016-07-04 12:07:29 +06:00
char buf[MAX_LOGMSG_LEN];
va_list ap;
2016-07-04 12:07:29 +06:00
va_start(ap, fmt);
size_t len = Q_vsnprintf(buf, sizeof buf - 1, fmt, ap);
2016-07-04 12:07:29 +06:00
va_end(ap);
2016-07-26 23:31:47 +07:00
buf[len] = '\n';
buf[len + 1] = '\0';
2016-07-04 12:07:29 +06:00
SERVER_PRINT(buf);
}
// Log regular message to logs; newline added.
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_LogMessage(plid_t plid, const char *fmt, ...)
{
2016-07-04 12:07:29 +06:00
char buf[MAX_LOGMSG_LEN];
va_list ap;
2016-07-04 12:07:29 +06:00
va_start(ap, fmt);
Q_vsnprintf(buf, sizeof buf, fmt, ap);
2016-07-04 12:07:29 +06:00
va_end(ap);
ALERT(at_logged, "[%s] %s\n", plid->logtag, buf);
2016-07-04 12:07:29 +06:00
}
// Log an error message to logs; newline added.
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_LogError(plid_t plid, const char *fmt, ...)
{
2016-07-04 12:07:29 +06:00
char buf[MAX_LOGMSG_LEN];
va_list ap;
2016-07-04 12:07:29 +06:00
va_start(ap, fmt);
Q_vsnprintf(buf, sizeof buf, fmt, ap);
2016-07-04 12:07:29 +06:00
va_end(ap);
ALERT(at_logged, "[%s] ERROR: %s\n", plid->logtag, buf);
2016-07-04 12:07:29 +06:00
}
// Log a message only if cvar "developer" set; newline added.
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_LogDeveloper(plid_t plid, const char* fmt, ...)
{
2016-07-04 12:07:29 +06:00
char buf[MAX_LOGMSG_LEN];
if ((int)CVAR_GET_FLOAT("developer") == 0)
2016-07-04 12:07:29 +06:00
return;
va_list ap;
2016-07-04 12:07:29 +06:00
va_start(ap, fmt);
Q_vsnprintf(buf, sizeof buf, fmt, ap);
2016-07-04 12:07:29 +06:00
va_end(ap);
ALERT(at_logged, "[%s] dev: %s\n", plid->logtag, buf);
2016-07-04 12:07:29 +06:00
}
// Print message on center of all player's screens. Uses default text
// parameters (color green, 10 second fade-in).
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_CenterSay(plid_t plid, const char* fmt, ...)
{
2016-07-04 12:07:29 +06:00
va_list ap;
va_start(ap, fmt);
mutil_CenterSayVarargs(plid, g_default_csay_tparms, fmt, ap);
2016-07-04 12:07:29 +06:00
va_end(ap);
}
// Print a center-message, with given text parameters.
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_CenterSayParms(plid_t plid, hudtextparms_t tparms, const char* fmt, ...)
{
2016-07-04 12:07:29 +06:00
va_list ap;
va_start(ap, fmt);
mutil_CenterSayVarargs(plid, tparms, fmt, ap);
va_end(ap);
}
// Print a center-message, with text parameters and varargs. Provides
// functionality to the above center_say interfaces.
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_CenterSayVarargs(plid_t plid, hudtextparms_t tparms, const char* fmt, va_list ap)
{
char buf[MAX_LOGMSG_LEN];
Q_vsnprintf(buf, sizeof buf, fmt, ap);
mutil_LogMessage(plid, "(centersay) %s", buf);
for (int n = 1; n <= gpGlobals->maxClients; n++)
2016-07-26 23:31:47 +07:00
{
auto pEntity = INDEXENT(n);
if (FNullEnt(pEntity) || pEntity->free)
continue;
UTIL_HudMessage(pEntity, tparms, buf);
}
}
2016-07-04 12:07:29 +06:00
// Allow plugins to call the entity functions in the GameDLL. In
// particular, calling "player()" as needed by most Bots. Suggested by
// Jussi Kivilinna.
2017-01-06 22:33:36 +03:00
qboolean EXT_FUNC mutil_CallGameEntity(plid_t plid, const char *entStr, entvars_t *pev)
{
2017-01-09 02:44:49 +03:00
META_DEBUG(8, "Looking up game entity '%s' for plugin '%s'", entStr, plid->name);
2017-01-13 03:00:23 +03:00
ENTITY_FN pfnEntity = (ENTITY_FN)g_GameDLL.sys_module.getsym(entStr);
2016-07-26 23:31:47 +07:00
if (!pfnEntity)
{
2017-01-13 03:00:23 +03:00
META_ERROR("Couldn't find game entity '%s' in game DLL '%s' for plugin '%s'", entStr, g_GameDLL.name, plid->name);
2016-07-26 23:31:47 +07:00
return false;
2016-07-04 12:07:29 +06:00
}
2016-07-26 23:31:47 +07:00
2017-01-09 02:44:49 +03:00
META_DEBUG(7, "Calling game entity '%s' for plugin '%s'", entStr, plid->name);
2016-07-04 12:07:29 +06:00
(*pfnEntity)(pev);
2016-07-26 23:31:47 +07:00
return true;
2016-07-04 12:07:29 +06:00
}
// Find a usermsg, registered by the gamedll, with the corresponding
// msgname, and return remaining info about it (msgid, size).
2017-01-06 22:33:36 +03:00
int EXT_FUNC mutil_GetUserMsgID(plid_t plid, const char* msgname, int* size)
{
META_DEBUG(8, "Looking up usermsg name '%s' for plugin '%s'", msgname, plid->name);
2016-07-04 12:07:29 +06:00
2016-07-26 23:31:47 +07:00
MRegMsg *umsg = g_regMsgs->find(msgname);
2016-07-26 23:31:47 +07:00
if (umsg)
{
if (size) *size = umsg->getsize();
return umsg->getid();
2016-07-04 12:07:29 +06:00
}
for (int n = 1; n < arraysize(g_engine_msg_names); n++) {
if (!strcmp(msgname, g_engine_msg_names[n])) {
if (size) *size = -1;
return n;
}
}
return 0;
2016-07-04 12:07:29 +06:00
}
// Find a usermsg, registered by the gamedll, with the corresponding
// msgid, and return remaining info about it (msgname, size).
2017-01-06 22:33:36 +03:00
const char* EXT_FUNC mutil_GetUserMsgName(plid_t plid, int msgid, int *size)
{
2016-07-26 23:31:47 +07:00
plugin_info_t *plinfo = (plugin_info_t *)plid;
META_DEBUG(8, "Looking up usermsg id '%d' for plugin '%s'", msgid, plinfo->name);
2016-07-04 12:07:29 +06:00
// Guess names for any built-in g_engine messages mentioned in the SDK;
2016-07-04 12:07:29 +06:00
// from dlls/util.h.
if (msgid < arraysize(g_engine_msg_names))
2016-07-26 23:31:47 +07:00
{
if (size) *size = -1;
return g_engine_msg_names[msgid];
2016-07-04 12:07:29 +06:00
}
2016-07-26 23:31:47 +07:00
MRegMsg *umsg = g_regMsgs->find(msgid);
2016-07-26 23:31:47 +07:00
if (umsg)
{
if (size) *size = umsg->getsize();
2016-07-04 12:07:29 +06:00
// 'name' is assumed to be a constant string, allocated in the
// gamedll.
return umsg->getname();
2016-07-04 12:07:29 +06:00
}
return nullptr;
2016-07-04 12:07:29 +06:00
}
// Return the full path of the plugin's loaded dll/so file.
2017-01-06 22:33:36 +03:00
const char* EXT_FUNC mutil_GetPluginPath(plid_t plid)
{
static char buf[PATH_MAX];
auto plug = g_plugins->find(plid);
2016-07-26 23:31:47 +07:00
if (!plug)
{
META_ERROR("GetPluginPath: couldn't find plugin '%s'", plid->name);
return nullptr;
2016-07-04 12:07:29 +06:00
}
2016-07-26 23:31:47 +07:00
Q_strncpy(buf, plug->m_pathname, sizeof buf - 1);
buf[sizeof buf - 1] = '\0';
2016-07-26 07:22:47 +07:00
return buf;
2016-07-04 12:07:29 +06:00
}
// Return various string-based info about the game/MOD/gamedll.
2017-01-06 22:33:36 +03:00
const char* EXT_FUNC mutil_GetGameInfo(plid_t plid, ginfo_t type)
2016-07-26 07:22:47 +07:00
{
2016-07-04 12:07:29 +06:00
static char buf[MAX_STRBUF_LEN];
2016-07-26 23:31:47 +07:00
const char *cp;
2016-07-26 23:31:47 +07:00
switch (type)
{
2016-07-26 07:22:47 +07:00
case GINFO_NAME:
2017-01-13 03:00:23 +03:00
cp = g_GameDLL.name;
2016-07-26 07:22:47 +07:00
break;
case GINFO_DESC:
2017-01-13 03:00:23 +03:00
cp = g_GameDLL.desc;
2016-07-26 07:22:47 +07:00
break;
case GINFO_GAMEDIR:
2017-01-13 03:00:23 +03:00
cp = g_GameDLL.gamedir;
2016-07-26 07:22:47 +07:00
break;
case GINFO_DLL_FULLPATH:
2017-01-13 03:00:23 +03:00
cp = g_GameDLL.pathname;
2016-07-26 07:22:47 +07:00
break;
case GINFO_DLL_FILENAME:
2017-01-13 03:00:23 +03:00
cp = g_GameDLL.file;
2016-07-26 07:22:47 +07:00
break;
case GINFO_REALDLL_FULLPATH:
2017-01-13 03:00:23 +03:00
cp = g_GameDLL.real_pathname;
2016-07-26 07:22:47 +07:00
break;
default:
2016-07-26 23:31:47 +07:00
META_ERROR("GetGameInfo: invalid request '%d' from plugin '%s'", type, plid->name);
return nullptr;
2016-07-04 12:07:29 +06:00
}
2016-07-26 23:31:47 +07:00
Q_strncpy(buf, cp, sizeof buf - 1);
buf[sizeof buf - 1] = '\0';
2016-07-26 07:22:47 +07:00
return buf;
2016-07-04 12:07:29 +06:00
}
2017-01-06 22:33:36 +03:00
int EXT_FUNC mutil_LoadMetaPlugin(plid_t plid, const char* fname, PLUG_LOADTIME now, void **plugin_handle)
2016-07-04 12:07:29 +06:00
{
2016-07-26 23:31:47 +07:00
if (!fname)
{
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
}
auto pl_loaded = g_plugins->plugin_addload(plid, fname, now);
if (!pl_loaded)
2016-07-26 23:31:47 +07:00
{
2016-07-26 07:22:47 +07:00
if (plugin_handle)
2016-07-26 23:31:47 +07:00
*plugin_handle = nullptr;
return 1; // TODO: WTF
}
2016-07-26 23:31:47 +07:00
else
{
2016-07-26 07:22:47 +07:00
if (plugin_handle)
*plugin_handle = (void *)pl_loaded->m_sys_module.gethandle();
2016-07-26 07:22:47 +07:00
return 0;
2016-07-04 12:07:29 +06:00
}
}
2017-01-06 22:33:36 +03:00
int EXT_FUNC mutil_UnloadMetaPlugin(plid_t plid, const char *fname, PLUG_LOADTIME now, PL_UNLOAD_REASON reason)
2016-07-04 12:07:29 +06:00
{
MPlugin *findp;
2016-07-04 12:07:29 +06:00
2016-07-26 23:31:47 +07:00
if (!fname)
{
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
}
char *endptr;
int pindex = strtol(fname, &endptr, 10);
bool unique = true;
2016-07-26 07:22:47 +07:00
if (*fname != '\0' && *endptr == '\0')
findp = g_plugins->find(pindex);
2016-07-04 12:07:29 +06:00
else
findp = g_plugins->find_match(fname, unique);
2016-07-04 12:07:29 +06:00
if (!findp || !unique)
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
2016-07-26 07:22:47 +07:00
if (findp->plugin_unload(plid, now, reason))
return 0;
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
}
2017-01-06 22:33:36 +03:00
int EXT_FUNC mutil_UnloadMetaPluginByHandle(plid_t plid, void *plugin_handle, PLUG_LOADTIME now, PL_UNLOAD_REASON reason)
2016-07-04 12:07:29 +06:00
{
2016-07-26 23:31:47 +07:00
if (!plugin_handle)
{
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
}
auto findp = g_plugins->find((module_handle_t)plugin_handle);
if (!findp)
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
2016-07-26 07:22:47 +07:00
if (findp->plugin_unload(plid, now, reason))
return 0;
2016-07-04 12:07:29 +06:00
2017-01-07 21:03:16 +03:00
return 1;
2016-07-04 12:07:29 +06:00
}
2017-01-06 22:33:36 +03:00
const char* EXT_FUNC mutil_IsQueryingClientCvar(plid_t plid, const edict_t* pEdict)
{
2016-07-30 02:03:01 +03:00
return g_players.is_querying_cvar(pEdict);
2016-07-04 12:07:29 +06:00
}
2017-01-06 22:33:36 +03:00
int EXT_FUNC mutil_MakeRequestId(plid_t plid)
{
//the offset is to distinguish from gamedll requests, if any
2017-01-13 03:00:23 +03:00
return abs(0xbeef << 16) + (++g_requestid_counter);
2016-07-04 12:07:29 +06:00
}
2017-01-06 22:33:36 +03:00
void EXT_FUNC mutil_GetHookTables(plid_t plid, enginefuncs_t** peng, DLL_FUNCTIONS** pdll, NEW_DLL_FUNCTIONS** pnewdll)
{
2016-07-04 12:07:29 +06:00
if (peng)
2017-01-13 03:00:23 +03:00
*peng = &g_meta_engfuncs;
2016-07-26 23:31:47 +07:00
2016-07-04 12:07:29 +06:00
if (pdll)
*pdll = pHookedDllFunctions;
2016-07-26 23:31:47 +07:00
2016-07-04 12:07:29 +06:00
if (pnewdll)
*pnewdll = pHookedNewDllFunctions;
}
2016-07-04 12:07:29 +06:00
// Meta Utility Function table.
2017-01-13 03:00:23 +03:00
mutil_funcs_t g_MetaUtilFunctions = {
2016-07-26 23:31:47 +07:00
mutil_LogConsole, // pfnLogConsole
mutil_LogMessage, // pfnLogMessage
mutil_LogError, // pfnLogError
mutil_LogDeveloper, // pfnLogDeveloper
mutil_CenterSay, // pfnCenterSay
mutil_CenterSayParms, // pfnCenterSayParms
mutil_CenterSayVarargs, // pfnCenterSayVarargs
mutil_CallGameEntity, // pfnCallGameEntity
mutil_GetUserMsgID, // pfnGetUserMsgID
mutil_GetUserMsgName, // pfnGetUserMsgName
mutil_GetPluginPath, // pfnGetPluginPath
mutil_GetGameInfo, // pfnGetGameInfo
mutil_LoadMetaPlugin, // pfnLoadPlugin
mutil_UnloadMetaPlugin, // pfnUnloadPlugin
mutil_UnloadMetaPluginByHandle, // pfnUnloadPluginByHandle
mutil_IsQueryingClientCvar, // pfnIsQueryingClientCvar
mutil_MakeRequestId, // pfnMakeRequestId
mutil_GetHookTables, // pfnGetHookTables
2016-07-04 12:07:29 +06:00
};