2016-07-26 07:22:47 +07:00
|
|
|
#include "precompiled.h"
|
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
cvar_t g_meta_version = { "metamod_version", APP_VERSION_STRD, FCVAR_SERVER, 0, nullptr };
|
2016-07-04 12:07:29 +06:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
MConfig g_static_config;
|
|
|
|
MConfig *g_config = &g_static_config;
|
|
|
|
option_t g_global_options[] =
|
2016-07-26 23:31:47 +07:00
|
|
|
{
|
2017-01-18 22:14:02 +07:00
|
|
|
{ "debuglevel", CF_INT, &g_config->m_debuglevel, "0" },
|
2017-03-11 19:12:09 +03:00
|
|
|
{ "gamedll", CF_PATH, &g_config->m_gamedll, nullptr },
|
|
|
|
{ "exec_cfg", CF_STR, &g_config->m_exec_cfg, nullptr },
|
|
|
|
{ "clientmeta", CF_BOOL, &g_config->m_clientmeta, false },
|
2016-07-26 23:31:47 +07:00
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// list terminator
|
2017-03-11 19:12:09 +03:00
|
|
|
{ nullptr, CF_NONE, nullptr, nullptr }
|
2016-07-04 12:07:29 +06:00
|
|
|
};
|
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
gamedll_t g_GameDLL;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2017-02-14 01:51:47 +03:00
|
|
|
ALIGN16
|
2016-07-30 02:03:01 +03:00
|
|
|
meta_globals_t g_metaGlobals;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2017-05-09 18:34:55 +03:00
|
|
|
enginefuncs_t g_plugin_engfuncs;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2016-07-26 23:31:47 +07:00
|
|
|
MPluginList *g_plugins;
|
|
|
|
MRegCmdList *g_regCmds;
|
|
|
|
MRegCvarList *g_regCvars;
|
|
|
|
MRegMsgList *g_regMsgs;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2016-07-30 02:03:01 +03:00
|
|
|
MPlayerList g_players;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
unsigned int g_CALL_API_count = 0;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
int g_requestid_counter = 0;
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Very first metamod function that's run.
|
|
|
|
// Do startup operations...
|
2016-07-26 23:31:47 +07:00
|
|
|
void metamod_startup()
|
2016-07-26 07:22:47 +07:00
|
|
|
{
|
2017-01-18 22:14:02 +07:00
|
|
|
char configFile[MAX_PATH];
|
|
|
|
char pluginFile[MAX_PATH];
|
|
|
|
char execFile[MAX_PATH];
|
|
|
|
|
|
|
|
Q_snprintf(configFile, sizeof configFile, "%s/%s", g_config->directory(), CONFIG_INI);
|
|
|
|
Q_snprintf(pluginFile, sizeof pluginFile, "%s/%s", g_config->directory(), PLUGINS_INI);
|
|
|
|
Q_snprintf(execFile, sizeof execFile, "%s/%s", g_config->directory(), EXEC_CFG);
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
META_CONS(" ");
|
2017-01-17 22:03:18 +03:00
|
|
|
META_CONS(" Metamod-r version %s Copyright (c) 2016-2017 ReHlds Team (rebuild of original Metamod by Will Day)", APP_VERSION_STRD);
|
|
|
|
META_CONS(" Metamod-r comes with ABSOLUTELY NO WARRANTY; for details type `meta gpl'.");
|
2016-07-04 12:07:29 +06:00
|
|
|
META_CONS(" This is free software, and you are welcome to redistribute it");
|
|
|
|
META_CONS(" under certain conditions; type `meta gpl' for details.");
|
|
|
|
META_CONS(" ");
|
|
|
|
|
2017-01-17 22:03:18 +03:00
|
|
|
META_CONS("Metamod-r v%s, API (%s)", APP_VERSION_STRD, META_INTERFACE_VERSION);
|
|
|
|
META_CONS("Metamod-r build: " __TIME__ " " __DATE__ " (" APP_VERSION_STRD ")");
|
|
|
|
META_CONS("Metamod-r from: " APP_COMMITS_URL APP_COMMIT_ID " " APP_COMMIT_AUTHOR "");
|
2016-07-04 12:07:29 +06:00
|
|
|
|
|
|
|
// Get gamedir, very early on, because it seems we need it all over the
|
|
|
|
// place here at the start.
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!meta_init_gamedll()) {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_ERROR("Failure to init game DLL; exiting...");
|
2016-07-26 15:18:32 +03:00
|
|
|
do_exit(1);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Register various console commands and cvars.
|
2016-07-26 23:31:47 +07:00
|
|
|
// Can I do these here, rather than waiting for GameDLLInit() ?
|
2016-07-04 12:07:29 +06:00
|
|
|
// Looks like it works okay..
|
|
|
|
meta_register_cmdcvar();
|
|
|
|
|
|
|
|
// Set a slight debug level for developer mode, if debug level not
|
|
|
|
// already set.
|
2017-05-09 19:31:09 +03:00
|
|
|
if (CVAR_GET_FLOAT("developer") != 0.0 && g_meta_debug.value == 0.0)
|
2016-07-26 23:31:47 +07:00
|
|
|
CVAR_SET_FLOAT("meta_debug", 3.0);
|
2016-07-04 12:07:29 +06:00
|
|
|
|
|
|
|
// Init default values
|
2017-01-13 03:00:23 +03:00
|
|
|
g_config->init(g_global_options);
|
2017-01-09 02:44:49 +03:00
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Find config file
|
2017-05-09 19:31:09 +03:00
|
|
|
const char *cp = LOCALINFO("mm_configfile");
|
|
|
|
if (cp && *cp != '\0') {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_LOG("Configfile specified via localinfo: %s", cp);
|
2017-01-18 22:14:02 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
if (is_file_exists_in_gamedir(cp)) {
|
2017-01-18 22:14:02 +07:00
|
|
|
Q_strncpy(configFile, cp, sizeof configFile - 1);
|
|
|
|
configFile[sizeof configFile - 1] = '\0';
|
|
|
|
}
|
2016-07-04 12:07:29 +06:00
|
|
|
else
|
2017-01-18 22:14:02 +07:00
|
|
|
META_ERROR("Empty/missing config.ini file: %s; falling back to %s", cp, configFile);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2017-01-18 22:14:02 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!is_file_exists_in_gamedir(configFile)) {
|
2017-01-23 01:44:52 +07:00
|
|
|
Q_strncpy(configFile, g_config->directory(), sizeof configFile - 1);
|
|
|
|
configFile[sizeof configFile - 1] = '\0';
|
|
|
|
|
|
|
|
// Get out of sub directory and check
|
|
|
|
char *dir = Q_strrchr(configFile, '/');
|
|
|
|
if (dir) {
|
|
|
|
*dir = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
Q_strcat(configFile, "/" CONFIG_INI);
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!is_file_exists_in_gamedir(configFile)) {
|
2017-01-23 01:44:52 +07:00
|
|
|
META_DEBUG(2, "No config.ini file found: %s", CONFIG_INI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Load config file
|
2017-05-09 18:34:55 +03:00
|
|
|
if (is_file_exists_in_gamedir(configFile))
|
2017-01-18 22:14:02 +07:00
|
|
|
g_config->load(configFile);
|
2016-07-04 12:07:29 +06:00
|
|
|
|
|
|
|
// Now, override config options with localinfo commandline options.
|
2017-05-09 19:31:09 +03:00
|
|
|
cp = LOCALINFO("mm_debug");
|
|
|
|
if (cp && *cp != '\0') {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_LOG("Debuglevel specified via localinfo: %s", cp);
|
2016-07-26 15:18:32 +03:00
|
|
|
g_config->set("debuglevel", cp);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2017-01-18 22:14:02 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
cp = LOCALINFO("mm_gamedll");
|
|
|
|
if (cp && *cp != '\0') {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_LOG("Gamedll specified via localinfo: %s", cp);
|
2016-07-26 15:18:32 +03:00
|
|
|
g_config->set("gamedll", cp);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2017-01-18 22:14:02 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
cp = LOCALINFO("mm_pluginsfile");
|
|
|
|
if (cp && *cp != '\0') {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_LOG("Pluginsfile specified via localinfo: %s", cp);
|
2016-07-26 15:18:32 +03:00
|
|
|
g_config->set("plugins_file", cp);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2017-01-18 22:14:02 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
cp = LOCALINFO("mm_execcfg");
|
|
|
|
if (cp && *cp != '\0') {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_LOG("Execcfg specified via localinfo: %s", cp);
|
2016-07-26 15:18:32 +03:00
|
|
|
g_config->set("exec_cfg", cp);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
cp = LOCALINFO("mm_clientmeta");
|
|
|
|
if (cp && *cp != '\0') {
|
2017-03-11 19:12:09 +03:00
|
|
|
META_LOG("Clientmeta specified via localinfo: %s", cp);
|
|
|
|
g_config->set("clientmeta", cp);
|
|
|
|
}
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Check for an initial debug level, since cfg files don't get exec'd
|
|
|
|
// until later.
|
2017-05-09 19:31:09 +03:00
|
|
|
if (g_config->m_debuglevel)
|
2017-01-13 03:00:23 +03:00
|
|
|
CVAR_SET_FLOAT("meta_debug", g_config->m_debuglevel);
|
2017-03-11 19:12:09 +03:00
|
|
|
if (!g_config->m_clientmeta)
|
|
|
|
disable_clientcommand_fwd();
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Prepare for registered commands from plugins.
|
2016-07-26 15:18:32 +03:00
|
|
|
g_regCmds = new MRegCmdList();
|
|
|
|
g_regCvars = new MRegCvarList();
|
2016-07-04 12:07:29 +06:00
|
|
|
|
|
|
|
// Prepare for registered user messages from gamedll.
|
2016-07-26 15:18:32 +03:00
|
|
|
g_regMsgs = new MRegMsgList();
|
2016-07-26 07:22:47 +07:00
|
|
|
|
2016-07-26 15:18:32 +03:00
|
|
|
// Copy, and store pointer in g_engine struct. Yes, we could just store
|
|
|
|
// the actual engine_t struct in g_engine, but then it wouldn't be a
|
2016-07-04 12:07:29 +06:00
|
|
|
// pointer to match the other g_engfuncs.
|
2017-05-09 18:34:55 +03:00
|
|
|
g_plugin_engfuncs = *g_engine.funcs;
|
2016-07-26 15:18:32 +03:00
|
|
|
g_engine.pl_funcs = &g_plugin_engfuncs;
|
2016-07-04 12:07:29 +06:00
|
|
|
// substitute our special versions of various commands
|
2016-07-26 15:18:32 +03:00
|
|
|
g_engine.pl_funcs->pfnAddServerCommand = meta_AddServerCommand;
|
|
|
|
g_engine.pl_funcs->pfnCVarRegister = meta_CVarRegister;
|
|
|
|
g_engine.pl_funcs->pfnCvar_RegisterVariable = meta_CVarRegister;
|
|
|
|
g_engine.pl_funcs->pfnRegUserMsg = meta_RegUserMsg;
|
2016-07-26 23:31:47 +07:00
|
|
|
|
2016-07-26 15:18:32 +03:00
|
|
|
if (g_engine.pl_funcs->pfnQueryClientCvarValue)
|
|
|
|
g_engine.pl_funcs->pfnQueryClientCvarValue = meta_QueryClientCvarValue;
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Before, we loaded plugins before loading the game DLL, so that if no
|
|
|
|
// plugins caught engine functions, we could pass engine funcs straight
|
|
|
|
// to game dll, rather than acting as intermediary. (Should perform
|
|
|
|
// better, right?)
|
|
|
|
//
|
|
|
|
// But since a plugin can be loaded at any time, we have to go ahead
|
|
|
|
// and catch the engine funcs regardless. Also, we want to give each
|
|
|
|
// plugin a copy of the gameDLL's api tables, in case they need to call
|
|
|
|
// API functions directly.
|
|
|
|
//
|
|
|
|
// Thus, load gameDLL first, then plugins.
|
|
|
|
//
|
2016-07-26 15:18:32 +03:00
|
|
|
// However, we have to init the g_plugins object first, because if the
|
2016-07-04 12:07:29 +06:00
|
|
|
// gamedll calls engine functions during GiveFnptrsToDll (like hpb_bot
|
|
|
|
// does) then it needs to be non-null so META_ENGINE_HANDLE won't crash.
|
|
|
|
//
|
2017-01-23 01:44:52 +07:00
|
|
|
// However, having replaced valid_file with FileExistsInGameDir, we need
|
2016-07-26 23:31:47 +07:00
|
|
|
// to at least initialize the gameDLL to include the gamedir, before
|
2016-07-04 12:07:29 +06:00
|
|
|
// looking for plugins.ini.
|
|
|
|
//
|
|
|
|
// In fact, we need gamedir even earlier, so moved up above.
|
|
|
|
|
2017-01-23 01:44:52 +07:00
|
|
|
// Load plugins file
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!is_file_exists_in_gamedir(pluginFile)) {
|
2017-01-23 01:44:52 +07:00
|
|
|
Q_strncpy(pluginFile, g_config->directory(), sizeof pluginFile - 1);
|
|
|
|
pluginFile[sizeof pluginFile - 1] = '\0';
|
|
|
|
|
|
|
|
// Get out of sub directory and check
|
|
|
|
char *dir = Q_strrchr(pluginFile, '/');
|
|
|
|
if (dir) {
|
|
|
|
*dir = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
Q_strcat(pluginFile, "/" PLUGINS_INI);
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!is_file_exists_in_gamedir(pluginFile)) {
|
2017-01-23 01:44:52 +07:00
|
|
|
META_DEBUG(2, "No plugins.ini file found: %s", PLUGINS_INI);
|
|
|
|
}
|
|
|
|
}
|
2016-07-04 12:07:29 +06:00
|
|
|
|
2017-01-18 22:14:02 +07:00
|
|
|
g_plugins = new MPluginList(pluginFile);
|
2016-07-04 12:07:29 +06:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!meta_load_gamedll()) {
|
2016-07-04 12:07:29 +06:00
|
|
|
META_ERROR("Failure to load game DLL; exiting...");
|
2016-07-26 15:18:32 +03:00
|
|
|
do_exit(1);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2016-07-26 23:31:47 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!g_plugins->load()) {
|
2016-07-26 15:18:32 +03:00
|
|
|
META_ERROR("Failure to load plugins...");
|
2016-07-04 12:07:29 +06:00
|
|
|
// Exit on failure here? Dunno...
|
|
|
|
}
|
|
|
|
|
2017-01-09 02:44:49 +03:00
|
|
|
meta_init_rehlds_api();
|
2016-07-26 15:18:32 +03:00
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Allow for commands to metamod plugins at startup. Autoexec.cfg is
|
|
|
|
// read too early, and server.cfg is read too late.
|
|
|
|
//
|
|
|
|
// Only attempt load if the file appears to exist and be non-empty, to
|
|
|
|
// avoid confusing users with "couldn't exec exec.cfg" console
|
|
|
|
// messages.
|
2017-05-09 19:31:09 +03:00
|
|
|
if (g_config->m_exec_cfg) {
|
2017-01-18 22:14:02 +07:00
|
|
|
Q_strncpy(execFile, g_config->m_exec_cfg, sizeof execFile - 1);
|
|
|
|
execFile[sizeof execFile - 1] = '\0';
|
|
|
|
}
|
2016-07-26 07:22:47 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
if (is_file_exists_in_gamedir(execFile)) {
|
2017-01-18 22:14:02 +07:00
|
|
|
if (execFile[0] == '/')
|
|
|
|
META_ERROR("Cannot exec absolute pathnames: %s", execFile);
|
2017-05-09 19:31:09 +03:00
|
|
|
else {
|
2016-07-26 23:31:47 +07:00
|
|
|
char cmd[NAME_MAX];
|
2017-01-18 22:14:02 +07:00
|
|
|
META_LOG("Exec'ing metamod exec.cfg: %s...", execFile);
|
|
|
|
Q_snprintf(cmd, sizeof cmd, "exec %s\n", execFile);
|
2016-07-04 12:07:29 +06:00
|
|
|
SERVER_COMMAND(cmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set initial GameDLL fields (name, gamedir).
|
|
|
|
// meta_errno values:
|
2016-07-26 15:18:32 +03:00
|
|
|
// - ME_NULLRESULT _getcwd failed
|
2017-05-08 23:52:22 +03:00
|
|
|
bool meta_init_gamedll()
|
2016-07-26 15:18:32 +03:00
|
|
|
{
|
2017-01-17 00:30:02 +03:00
|
|
|
Q_memset(&g_GameDLL, 0, sizeof g_GameDLL);
|
2016-07-04 12:07:29 +06:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
char gamedir[PATH_MAX];
|
2016-07-04 12:07:29 +06:00
|
|
|
GET_GAME_DIR(gamedir);
|
2017-05-09 18:34:55 +03:00
|
|
|
normalize_path(gamedir);
|
2017-01-23 01:44:52 +07:00
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// As of 1.1.1.1, the engine routine GET_GAME_DIR no longer returns a
|
|
|
|
// full-pathname, but rather just the path specified as the argument to
|
|
|
|
// "-game".
|
|
|
|
//
|
|
|
|
// However, since we have to work properly under both the new version
|
|
|
|
// as well as previous versions, we have to support both possibilities.
|
|
|
|
//
|
|
|
|
// Note: the code has always assumed the server op wouldn't do:
|
|
|
|
// hlds -game other/firearms
|
|
|
|
//
|
2017-05-09 19:31:09 +03:00
|
|
|
if (is_abs_path(gamedir)) {
|
2016-07-04 12:07:29 +06:00
|
|
|
// Old style; GET_GAME_DIR returned full pathname. Copy this into
|
|
|
|
// our gamedir, and truncate to get the game name.
|
|
|
|
// (note check for both linux and win32 full pathname.)
|
2017-01-13 03:00:23 +03:00
|
|
|
Q_strncpy(g_GameDLL.gamedir, gamedir, sizeof g_GameDLL.gamedir - 1);
|
|
|
|
g_GameDLL.gamedir[sizeof g_GameDLL.gamedir - 1] = '\0';
|
2016-07-26 23:31:47 +07:00
|
|
|
|
2017-05-09 19:31:09 +03:00
|
|
|
char* cp = Q_strrchr(gamedir, '/') + 1;
|
2016-07-26 23:31:47 +07:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
Q_strncpy(g_GameDLL.name, cp, sizeof g_GameDLL.name - 1);
|
|
|
|
g_GameDLL.name[sizeof g_GameDLL.name - 1] = '\0';
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2017-05-09 19:31:09 +03:00
|
|
|
else {
|
2016-07-04 12:07:29 +06:00
|
|
|
// New style; GET_GAME_DIR returned game name. Copy this into our
|
|
|
|
// game name, and prepend the current working directory.
|
2016-07-26 23:31:47 +07:00
|
|
|
char buf[PATH_MAX];
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!_getcwd(buf, sizeof buf)) {
|
2016-07-26 23:31:47 +07:00
|
|
|
META_WARNING("dll: Couldn't get cwd; %s", strerror(errno));
|
2017-01-07 21:03:16 +03:00
|
|
|
return false;
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2016-07-26 23:31:47 +07:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
Q_snprintf(g_GameDLL.gamedir, sizeof g_GameDLL.gamedir, "%s/%s", buf, gamedir);
|
|
|
|
Q_strncpy(g_GameDLL.name, gamedir, sizeof g_GameDLL.name - 1);
|
2017-01-17 00:30:02 +03:00
|
|
|
g_GameDLL.name[sizeof g_GameDLL.name - 1] = '\0';
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
META_DEBUG(3, "Game: %s", g_GameDLL.name);
|
2016-07-30 02:03:01 +03:00
|
|
|
return true;
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
2017-01-09 02:44:49 +03:00
|
|
|
template<typename ifvers_t, typename table_t>
|
|
|
|
bool get_function_table(const char* ifname, int ifvers_mm, table_t*& table, size_t table_size = sizeof(table_t))
|
|
|
|
{
|
2017-01-13 02:04:11 +03:00
|
|
|
typedef int(*getfunc_t)(table_t *pFunctionTable, ifvers_t interfaceVersion);
|
2017-01-09 02:44:49 +03:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
auto pfnGetFuncs = (getfunc_t)g_GameDLL.sys_module.getsym(ifname);
|
2017-01-09 02:44:49 +03:00
|
|
|
|
|
|
|
if (pfnGetFuncs) {
|
|
|
|
table = (table_t *)Q_calloc(1, table_size);
|
|
|
|
|
|
|
|
int ifvers_gamedll = ifvers_mm;
|
|
|
|
|
|
|
|
if (pfnGetFuncs(table, ifvers_gamedll)) {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_DEBUG(3, "dll: Game '%s': Found %s", g_GameDLL.name, ifname);
|
2017-01-09 02:44:49 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
META_ERROR("dll: Failure calling %s in game '%s'", ifname, g_GameDLL.name);
|
2017-01-09 02:44:49 +03:00
|
|
|
Q_free(table);
|
|
|
|
table = nullptr;
|
|
|
|
|
|
|
|
if (ifvers_gamedll != ifvers_mm) {
|
|
|
|
META_ERROR("dll: Interface version didn't match; we wanted %d, they had %d", ifvers_mm, ifvers_gamedll);
|
|
|
|
META_CONS("==================");
|
|
|
|
META_CONS("Game DLL version mismatch");
|
|
|
|
META_CONS("DLL version is %d, engine version is %d", ifvers_gamedll, ifvers_mm);
|
|
|
|
if (ifvers_gamedll > ifvers_mm)
|
|
|
|
META_CONS("g_engine appears to be outdated, check for updates");
|
|
|
|
else
|
2017-01-13 03:00:23 +03:00
|
|
|
META_CONS("The game DLL for %s appears to be outdated, check for updates", g_GameDLL.name);
|
2017-01-09 02:44:49 +03:00
|
|
|
META_CONS("==================");
|
|
|
|
ALERT(at_error, "Exiting...\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_DEBUG(5, "dll: Game '%s': No %s", g_GameDLL.name, ifname);
|
2017-01-09 02:44:49 +03:00
|
|
|
table = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-01-13 02:04:11 +03:00
|
|
|
template<typename table_t>
|
|
|
|
bool get_function_table_old(const char* ifname, int ifvers_mm, table_t*& table, size_t table_size = sizeof(table_t))
|
|
|
|
{
|
2017-05-09 19:31:09 +03:00
|
|
|
typedef int(*getfunc_t)(table_t *pFunctionTable, int interfaceVersion);
|
2017-01-13 02:04:11 +03:00
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
auto pfnGetFuncs = (getfunc_t)g_GameDLL.sys_module.getsym(ifname);
|
2017-01-13 02:04:11 +03:00
|
|
|
|
|
|
|
if (pfnGetFuncs) {
|
|
|
|
table = (table_t *)Q_calloc(1, table_size);
|
|
|
|
|
|
|
|
if (pfnGetFuncs(table, ifvers_mm)) {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_DEBUG(3, "dll: Game '%s': Found %s", g_GameDLL.name, ifname);
|
2017-01-13 02:04:11 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
META_ERROR("dll: Failure calling %s in game '%s'", ifname, g_GameDLL.name);
|
2017-01-13 02:04:11 +03:00
|
|
|
Q_free(table);
|
|
|
|
table = nullptr;
|
|
|
|
}
|
|
|
|
else {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_DEBUG(5, "dll: Game '%s': No %s", g_GameDLL.name, ifname);
|
2017-01-13 02:04:11 +03:00
|
|
|
table = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Load game DLL.
|
|
|
|
// meta_errno values:
|
|
|
|
// - ME_DLOPEN couldn't dlopen game dll file
|
|
|
|
// - ME_DLMISSING couldn't find required routine in game dll
|
|
|
|
// (GiveFnptrsToDll, GetEntityAPI, GetEntityAPI2)
|
2017-05-08 23:52:22 +03:00
|
|
|
bool meta_load_gamedll()
|
2016-07-26 15:18:32 +03:00
|
|
|
{
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!setup_gamedll(&g_GameDLL)) {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_ERROR("dll: Unrecognized game: %s", g_GameDLL.name);
|
2016-07-04 12:07:29 +06:00
|
|
|
// meta_errno should be already set in lookup_game()
|
2016-07-30 02:03:01 +03:00
|
|
|
return false;
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
// open the game DLL
|
2017-05-09 19:31:09 +03:00
|
|
|
if (!g_GameDLL.sys_module.load(g_GameDLL.pathname)) {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_ERROR("dll: Couldn't load game DLL %s: %s", g_GameDLL.pathname, CSysModule::getloaderror());
|
2017-01-07 21:03:16 +03:00
|
|
|
return false;
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
2016-07-30 02:03:01 +03:00
|
|
|
// prepare meta_engfuncs
|
|
|
|
compile_engine_callbacks();
|
|
|
|
|
2016-07-04 12:07:29 +06:00
|
|
|
// Used to only pass our table of engine funcs if a loaded plugin
|
|
|
|
// wanted to catch one of the functions, but now that plugins are
|
|
|
|
// dynamically loadable at any time, we have to always pass our table,
|
|
|
|
// so that any plugin loaded later can catch what they need to.
|
2017-01-13 03:00:23 +03:00
|
|
|
auto pfn_give_engfuncs = (GIVE_ENGINE_FUNCTIONS_FN)g_GameDLL.sys_module.getsym("GiveFnptrsToDll");
|
2017-05-09 19:31:09 +03:00
|
|
|
if (pfn_give_engfuncs) {
|
2017-01-13 03:00:23 +03:00
|
|
|
pfn_give_engfuncs(&g_meta_engfuncs, gpGlobals);
|
|
|
|
META_DEBUG(3, "dll: Game '%s': Called GiveFnptrsToDll", g_GameDLL.name);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
2017-05-09 19:31:09 +03:00
|
|
|
else {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_ERROR("dll: Couldn't find GiveFnptrsToDll() in game DLL '%s'", g_GameDLL.name);
|
2017-01-07 21:03:16 +03:00
|
|
|
return false;
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
2016-07-26 15:18:32 +03:00
|
|
|
// Look for API-NEW interface in Game dll. We do this before API2/API, because
|
2016-07-04 12:07:29 +06:00
|
|
|
// that's what the engine appears to do..
|
2017-01-13 03:00:23 +03:00
|
|
|
get_function_table<int&>("GetNewDLLFunctions", NEW_DLL_FUNCTIONS_VERSION, g_GameDLL.funcs.newapi_table);
|
2016-07-04 12:07:29 +06:00
|
|
|
|
|
|
|
// Look for API2 interface in plugin; preferred over API-1.
|
2017-01-13 03:00:23 +03:00
|
|
|
bool found = get_function_table<int&>("GetEntityAPI2", INTERFACE_VERSION, g_GameDLL.funcs.dllapi_table);
|
2016-07-04 12:07:29 +06:00
|
|
|
|
|
|
|
// Look for API-1 in plugin, if API2 interface wasn't found.
|
2017-01-09 02:44:49 +03:00
|
|
|
if (!found) {
|
2017-01-13 03:00:23 +03:00
|
|
|
found = get_function_table_old("GetEntityAPI", INTERFACE_VERSION, g_GameDLL.funcs.dllapi_table);
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
// If didn't find either, return failure.
|
2017-01-09 02:44:49 +03:00
|
|
|
if (!found) {
|
2017-01-13 03:00:23 +03:00
|
|
|
META_ERROR("dll: Couldn't find either GetEntityAPI nor GetEntityAPI2 in game DLL '%s'", g_GameDLL.name);
|
2017-01-07 21:03:16 +03:00
|
|
|
return false;
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|
|
|
|
|
2016-07-30 02:03:01 +03:00
|
|
|
// prepare gamedll callbacks
|
|
|
|
compile_gamedll_callbacks();
|
|
|
|
|
2017-01-13 03:00:23 +03:00
|
|
|
META_LOG("Game DLL for '%s' loaded successfully", g_GameDLL.desc);
|
2016-07-30 02:03:01 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void meta_rebuild_callbacks()
|
|
|
|
{
|
2017-05-09 00:36:28 +03:00
|
|
|
META_LOG("dll: Rebuilding callbacks...");
|
|
|
|
|
2016-07-30 02:03:01 +03:00
|
|
|
g_jit.clear_callbacks();
|
|
|
|
|
|
|
|
compile_engine_callbacks();
|
|
|
|
compile_gamedll_callbacks();
|
2016-07-04 12:07:29 +06:00
|
|
|
}
|