2008-11-26 00:19:30 -06:00
|
|
|
/**
|
2009-10-29 10:57:27 -05:00
|
|
|
* vim: set ts=4 sw=4 tw=99 noet :
|
2008-11-26 00:19:30 -06:00
|
|
|
* ======================================================
|
|
|
|
* Metamod:Source
|
2010-08-07 15:26:30 -05:00
|
|
|
* Copyright (C) 2004-2010 AlliedModders LLC and authors.
|
2008-11-26 00:19:30 -06:00
|
|
|
* All rights reserved.
|
|
|
|
* ======================================================
|
|
|
|
*
|
|
|
|
* This software is provided 'as-is', without any express or implied warranty.
|
|
|
|
* In no event will the authors be held liable for any damages arising from
|
|
|
|
* the use of this software.
|
|
|
|
*
|
|
|
|
* Permission is granted to anyone to use this software for any purpose,
|
|
|
|
* including commercial applications, and to alter it and redistribute it
|
|
|
|
* freely, subject to the following restrictions:
|
|
|
|
*
|
|
|
|
* 1. The origin of this software must not be misrepresented; you must not
|
|
|
|
* claim that you wrote the original software. If you use this software in a
|
|
|
|
* product, an acknowledgment in the product documentation would be
|
|
|
|
* appreciated but is not required.
|
|
|
|
* 2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
* misrepresented as being the original software.
|
|
|
|
* 3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
|
|
|
|
2008-11-15 18:29:24 -06:00
|
|
|
#include <time.h>
|
2008-11-23 21:21:49 -06:00
|
|
|
#include <assert.h>
|
2008-11-15 18:29:24 -06:00
|
|
|
#include <stdarg.h>
|
2008-11-23 21:21:49 -06:00
|
|
|
#include <stdio.h>
|
2008-11-15 18:29:24 -06:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "loader.h"
|
|
|
|
#include "serverplugin.h"
|
2008-11-23 21:21:49 -06:00
|
|
|
#include "gamedll.h"
|
|
|
|
#include "utility.h"
|
2013-03-22 20:47:35 -04:00
|
|
|
#if defined __APPLE__
|
|
|
|
#include <crt_externs.h>
|
|
|
|
#endif
|
2008-11-15 18:29:24 -06:00
|
|
|
|
|
|
|
static HMODULE mm_library = NULL;
|
|
|
|
static char mm_fatal_logfile[PLATFORM_MAX_PATH] = "metamod-fatal.log";
|
2009-11-22 23:54:55 -06:00
|
|
|
MetamodBackend mm_backend = MMBackend_UNKNOWN;
|
2008-11-15 18:29:24 -06:00
|
|
|
|
|
|
|
extern void
|
|
|
|
mm_LogFatal(const char *message, ...)
|
|
|
|
{
|
|
|
|
FILE *fp;
|
|
|
|
time_t t;
|
|
|
|
va_list ap;
|
|
|
|
char header[256];
|
|
|
|
|
|
|
|
fp = fopen(mm_fatal_logfile, "at");
|
|
|
|
if (!fp && (fp = fopen("metamod-fatal.log", "at")) == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
t = time(NULL);
|
2008-11-24 01:52:07 -06:00
|
|
|
strftime(header, sizeof(header), "%m/%d/%Y - %H:%M:%S", localtime(&t));
|
2008-11-15 18:29:24 -06:00
|
|
|
fprintf(fp, "L %s: ", header);
|
|
|
|
|
|
|
|
va_start(ap, message);
|
|
|
|
vfprintf(fp, message, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
fprintf(fp, "\n");
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
|
2009-02-18 01:38:23 -06:00
|
|
|
static const char *backend_names[] =
|
2008-11-15 18:29:24 -06:00
|
|
|
{
|
|
|
|
"1.ep1",
|
2009-10-29 10:57:27 -05:00
|
|
|
"2.darkm",
|
2008-11-15 18:29:24 -06:00
|
|
|
"2.ep2",
|
2011-01-10 14:09:50 -05:00
|
|
|
"2.bgt",
|
2011-08-08 08:57:03 -04:00
|
|
|
"2.eye",
|
2012-08-20 14:25:29 -04:00
|
|
|
"2.css",
|
2009-10-29 16:23:25 -05:00
|
|
|
"2.ep2v",
|
2009-02-18 01:38:23 -06:00
|
|
|
"2.l4d",
|
2010-07-22 03:22:27 -05:00
|
|
|
"2.l4d2",
|
2012-05-26 17:07:26 -04:00
|
|
|
"2.swarm",
|
|
|
|
"2.portal2",
|
2013-03-04 10:38:17 -05:00
|
|
|
"2.csgo",
|
|
|
|
"2.dota",
|
2013-07-31 06:49:21 -04:00
|
|
|
"2.hl2dm",
|
|
|
|
"2.dods",
|
|
|
|
"2.tf2",
|
|
|
|
"2.nd",
|
2013-10-08 21:01:53 -04:00
|
|
|
"2.sdk2013",
|
2013-10-09 08:46:13 -04:00
|
|
|
"2.blade",
|
2008-11-15 18:29:24 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
#if defined _WIN32
|
|
|
|
#define LIBRARY_EXT ".dll"
|
|
|
|
#define LIBRARY_MINEXT ".dll"
|
2010-05-14 00:34:56 -05:00
|
|
|
#elif defined __APPLE__
|
|
|
|
#define LIBRARY_EXT ".dylib"
|
|
|
|
#define LIBRARY_MINEXT ".dylib"
|
2012-10-26 16:15:10 -04:00
|
|
|
#elif defined __linux__
|
|
|
|
#define LIBRARY_EXT LIB_SUFFIX
|
|
|
|
#define LIBRARY_MINEXT ".so"
|
2008-11-15 18:29:24 -06:00
|
|
|
#endif
|
|
|
|
|
|
|
|
bool
|
|
|
|
mm_LoadMetamodLibrary(MetamodBackend backend, char *buffer, size_t maxlength)
|
|
|
|
{
|
|
|
|
size_t len, temp_len;
|
|
|
|
char mm_path[PLATFORM_MAX_PATH * 2];
|
|
|
|
|
|
|
|
/* Get our path */
|
|
|
|
if (!mm_GetFileOfAddress((void*)mm_GetFileOfAddress, mm_path, sizeof(mm_path)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
len = strlen(mm_path);
|
|
|
|
temp_len = strlen("server" LIBRARY_EXT);
|
|
|
|
if (len < temp_len)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Build log file name */
|
|
|
|
mm_path[len - temp_len] = '\0';
|
|
|
|
mm_Format(mm_fatal_logfile,
|
|
|
|
sizeof(mm_fatal_logfile),
|
|
|
|
"%smetamod-fatal.log",
|
|
|
|
mm_path);
|
|
|
|
|
|
|
|
/* Replace server.dll with the new binary we want */
|
|
|
|
mm_Format(&mm_path[len - temp_len],
|
|
|
|
sizeof(mm_path) - (len - temp_len),
|
|
|
|
"metamod.%s" LIBRARY_MINEXT,
|
|
|
|
backend_names[backend]);
|
|
|
|
|
2008-11-23 21:25:45 -06:00
|
|
|
mm_library = (HMODULE)mm_LoadLibrary(mm_path, buffer, maxlength);
|
2008-11-15 18:29:24 -06:00
|
|
|
|
2008-11-23 21:21:49 -06:00
|
|
|
return (mm_library != NULL);
|
2008-11-15 18:29:24 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
mm_UnloadMetamodLibrary()
|
|
|
|
{
|
2008-11-23 21:21:49 -06:00
|
|
|
mm_UnloadLibrary(mm_library);
|
|
|
|
mm_library = NULL;
|
2008-11-15 18:29:24 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
#if defined _WIN32
|
|
|
|
#define EXPORT extern "C" __declspec(dllexport)
|
|
|
|
#elif defined __GNUC__
|
|
|
|
#if __GNUC__ == 4
|
|
|
|
#define EXPORT extern "C" __attribute__ ((visibility("default")))
|
|
|
|
#else
|
|
|
|
#define EXPORT extern "C"
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
EXPORT void *
|
|
|
|
CreateInterface(const char *name, int *ret)
|
|
|
|
{
|
2008-11-23 21:21:49 -06:00
|
|
|
/* If we've got a VSP bridge, do nothing. */
|
|
|
|
if (vsp_bridge != NULL)
|
2008-11-15 18:29:24 -06:00
|
|
|
{
|
|
|
|
if (ret != NULL)
|
|
|
|
*ret = 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *ptr;
|
|
|
|
if (strncmp(name, "ISERVERPLUGINCALLBACKS", 22) == 0)
|
2009-11-22 23:54:55 -06:00
|
|
|
{
|
|
|
|
/* Either load as VSP or start VSP listener */
|
2008-11-15 18:29:24 -06:00
|
|
|
ptr = mm_GetVspCallbacks(atoi(&name[22]));
|
2009-11-22 23:54:55 -06:00
|
|
|
}
|
|
|
|
else if (gamedll_bridge == NULL)
|
|
|
|
{
|
|
|
|
/* Load as gamedll */
|
2008-11-23 21:21:49 -06:00
|
|
|
ptr = mm_GameDllRequest(name, ret);
|
2009-11-22 23:54:55 -06:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* If we've got a gamedll bridge, forward the request. */
|
|
|
|
return gamedll_bridge->QueryInterface(name, ret);
|
|
|
|
}
|
2008-11-15 18:29:24 -06:00
|
|
|
|
|
|
|
if (ret != NULL)
|
2008-11-23 21:21:49 -06:00
|
|
|
*ret = (ptr != NULL) ? 0 : 1;
|
2008-11-15 18:29:24 -06:00
|
|
|
|
2008-11-23 21:21:49 -06:00
|
|
|
return ptr;
|
2008-11-15 18:29:24 -06:00
|
|
|
}
|
|
|
|
|
2008-11-23 21:21:49 -06:00
|
|
|
void *
|
2008-11-15 18:29:24 -06:00
|
|
|
mm_GetProcAddress(const char *name)
|
|
|
|
{
|
2008-11-23 21:21:49 -06:00
|
|
|
return mm_GetLibAddress(mm_library, name);
|
|
|
|
}
|
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
bool
|
|
|
|
mm_GetGameName(char *buffer, size_t size)
|
|
|
|
{
|
|
|
|
buffer[0] = '\0';
|
|
|
|
|
2009-02-18 01:38:23 -06:00
|
|
|
#if defined _WIN32
|
2013-03-22 20:47:35 -04:00
|
|
|
static char game[128];
|
2009-02-18 01:38:23 -06:00
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
LPWSTR pCmdLine = GetCommandLineW();
|
|
|
|
int argc;
|
|
|
|
LPWSTR *wargv = CommandLineToArgvW(pCmdLine, &argc);
|
|
|
|
for (int i = 0; i < argc; ++i)
|
2009-02-18 01:38:23 -06:00
|
|
|
{
|
2013-03-22 20:47:35 -04:00
|
|
|
if (wcscmp(wargv[i], L"-game") != 0)
|
|
|
|
continue;
|
2009-02-18 01:38:23 -06:00
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
if (++i >= argc)
|
|
|
|
break;
|
|
|
|
|
|
|
|
wcstombs(buffer, wargv[i], size);
|
|
|
|
buffer[size-1] = '\0';
|
|
|
|
break;
|
2009-02-18 01:38:23 -06:00
|
|
|
}
|
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
LocalFree(wargv);
|
2010-07-22 03:22:27 -05:00
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
return buffer[0] != 0;
|
|
|
|
#elif defined __APPLE__
|
|
|
|
int argc = *_NSGetArgc();
|
|
|
|
char **argv = *_NSGetArgv();
|
|
|
|
for (int i = 0; i < argc; ++i)
|
2010-07-22 03:22:27 -05:00
|
|
|
{
|
2013-03-22 20:47:35 -04:00
|
|
|
if (strcmp(argv[i], "-game") != 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (++i >= argc)
|
|
|
|
break;
|
|
|
|
|
|
|
|
strncpy(buffer, argv[i], size);
|
|
|
|
buffer[size-1] = '\0';
|
|
|
|
break;
|
2010-07-22 03:22:27 -05:00
|
|
|
}
|
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
return buffer[0] != 0;
|
|
|
|
#elif defined __linux__
|
|
|
|
FILE *pFile = fopen("/proc/self/cmdline", "rb");
|
|
|
|
if (!pFile)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
char *arg = NULL;
|
|
|
|
size_t argsize = 0;
|
|
|
|
bool bNextIsGame = false;
|
|
|
|
|
2013-07-06 12:27:14 -04:00
|
|
|
while (getdelim(&arg, &argsize, 0, pFile) != -1)
|
2009-02-18 01:38:23 -06:00
|
|
|
{
|
2013-03-22 20:47:35 -04:00
|
|
|
if (bNextIsGame)
|
2009-02-18 01:38:23 -06:00
|
|
|
{
|
2013-03-22 20:47:35 -04:00
|
|
|
strncpy(buffer, arg, size);
|
|
|
|
buffer[size-1] = '\0';
|
|
|
|
break;
|
2009-02-18 01:38:23 -06:00
|
|
|
}
|
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
if (strcmp(arg, "-game") == 0)
|
2009-02-18 01:38:23 -06:00
|
|
|
{
|
2013-03-22 20:47:35 -04:00
|
|
|
bNextIsGame = true;
|
2009-02-18 01:38:23 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
free(arg);
|
|
|
|
fclose(pFile);
|
2009-02-18 01:38:23 -06:00
|
|
|
|
2013-03-22 20:47:35 -04:00
|
|
|
return buffer[0] != 0;
|
|
|
|
#else
|
|
|
|
#error unsupported platform
|
|
|
|
#endif
|
2009-02-18 01:38:23 -06:00
|
|
|
}
|
|
|
|
|
2008-11-23 21:21:49 -06:00
|
|
|
MetamodBackend
|
2009-02-18 01:38:23 -06:00
|
|
|
mm_DetermineBackend(QueryValveInterface engineFactory, const char *game_name)
|
2008-11-23 21:21:49 -06:00
|
|
|
{
|
2013-06-18 12:15:29 -04:00
|
|
|
if (engineFactory("VEngineServer024", NULL) != NULL)
|
2012-05-26 17:07:26 -04:00
|
|
|
{
|
2013-06-18 12:15:29 -04:00
|
|
|
return MMBackend_DOTA;
|
|
|
|
}
|
|
|
|
else if (engineFactory("VEngineServer023", NULL) != NULL)
|
|
|
|
{
|
|
|
|
return MMBackend_CSGO;
|
2012-05-26 17:07:26 -04:00
|
|
|
}
|
|
|
|
else if (engineFactory("VEngineServer022", NULL) != NULL &&
|
2008-11-23 21:21:49 -06:00
|
|
|
engineFactory("VEngineCvar007", NULL) != NULL)
|
|
|
|
{
|
2013-10-09 08:46:13 -04:00
|
|
|
if (strcmp(game_name, "berimbau") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_Blade;
|
|
|
|
}
|
2012-05-26 17:07:26 -04:00
|
|
|
if (strcmp(game_name, "portal2") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_Portal2;
|
|
|
|
}
|
2010-07-22 03:22:27 -05:00
|
|
|
if (engineFactory("EngineTraceServer004", NULL) != NULL)
|
|
|
|
{
|
|
|
|
return MMBackend_AlienSwarm;
|
|
|
|
}
|
|
|
|
else if (engineFactory("VPrecacheSystem001", NULL) != NULL)
|
2009-10-29 10:57:27 -05:00
|
|
|
{
|
2013-07-31 06:49:21 -04:00
|
|
|
if (strcmp(game_name, "nucleardawn") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_NuclearDawn;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return MMBackend_Left4Dead2;
|
|
|
|
}
|
2009-10-29 10:57:27 -05:00
|
|
|
}
|
2008-11-23 21:21:49 -06:00
|
|
|
return MMBackend_Left4Dead;
|
|
|
|
}
|
|
|
|
else if (engineFactory("VEngineServer021", NULL) != NULL)
|
|
|
|
{
|
|
|
|
/* Check for OB */
|
2010-12-17 22:20:10 -05:00
|
|
|
if (engineFactory("VEngineCvar004", NULL) != NULL)
|
2008-11-23 21:21:49 -06:00
|
|
|
{
|
2010-12-17 22:20:10 -05:00
|
|
|
if (engineFactory("VModelInfoServer002", NULL) != NULL)
|
|
|
|
{
|
2011-01-10 14:09:50 -05:00
|
|
|
/* BGT has same iface version numbers and libs as ep2 */
|
|
|
|
if (strcmp(game_name, "pm") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_BloodyGoodTime;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return MMBackend_Episode2;
|
|
|
|
}
|
2010-12-17 22:20:10 -05:00
|
|
|
}
|
|
|
|
else if (engineFactory("VModelInfoServer003", NULL) != NULL)
|
2009-10-29 16:23:25 -05:00
|
|
|
{
|
2011-08-08 08:57:03 -04:00
|
|
|
if (engineFactory("VFileSystem017", NULL) != NULL)
|
|
|
|
{
|
|
|
|
return MMBackend_EYE;
|
|
|
|
}
|
2012-08-20 14:25:29 -04:00
|
|
|
else if (strcmp(game_name, "cstrike") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_CSS;
|
|
|
|
}
|
2013-07-31 06:49:21 -04:00
|
|
|
else if (strcmp(game_name, "tf") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_TF2;
|
|
|
|
}
|
|
|
|
else if (strcmp(game_name, "dod") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_DODS;
|
|
|
|
}
|
|
|
|
else if (strcmp(game_name, "hl2mp") == 0)
|
2011-08-08 08:57:03 -04:00
|
|
|
{
|
2013-07-31 06:49:21 -04:00
|
|
|
return MMBackend_HL2DM;
|
2011-08-08 08:57:03 -04:00
|
|
|
}
|
2013-10-08 21:01:53 -04:00
|
|
|
else
|
|
|
|
{
|
|
|
|
return MMBackend_SDK2013;
|
|
|
|
}
|
2009-10-29 16:23:25 -05:00
|
|
|
}
|
2008-11-23 21:21:49 -06:00
|
|
|
}
|
2009-02-18 01:38:23 -06:00
|
|
|
/* Check for Episode One/Old Engine */
|
2008-11-23 21:21:49 -06:00
|
|
|
else if (engineFactory("VModelInfoServer001", NULL) != NULL &&
|
|
|
|
(engineFactory("VEngineCvar003", NULL) != NULL ||
|
|
|
|
engineFactory("VEngineCvar002", NULL) != NULL))
|
|
|
|
{
|
2009-02-18 01:38:23 -06:00
|
|
|
/* Check for Dark Messiah which has a weird directory structure */
|
|
|
|
if (strcmp(game_name, ".") == 0)
|
|
|
|
{
|
|
|
|
return MMBackend_DarkMessiah;
|
|
|
|
}
|
2008-11-23 21:21:49 -06:00
|
|
|
return MMBackend_Episode1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return MMBackend_UNKNOWN;
|
2008-11-15 18:29:24 -06:00
|
|
|
}
|
|
|
|
|