2007-11-25 19:42:40 +00:00
|
|
|
/* copyright (c) 2007 magnus auvinen, see licence.txt for more info */
|
2007-11-08 09:11:32 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2007-12-15 10:24:49 +00:00
|
|
|
#include <engine/e_system.h>
|
2008-01-19 10:57:25 +00:00
|
|
|
#include <engine/e_server_interface.h>
|
|
|
|
/*#include <engine/e_client_interface.h>*/
|
2007-12-15 10:24:49 +00:00
|
|
|
#include <engine/e_config.h>
|
2008-01-16 22:14:06 +00:00
|
|
|
#include <engine/e_console.h>
|
2008-02-10 21:54:52 +00:00
|
|
|
#include <engine/e_engine.h>
|
|
|
|
#include "e_linereader.h"
|
2008-02-04 07:36:54 +00:00
|
|
|
|
|
|
|
static void con_dbg_dumpmem(void *result, void *user_data)
|
|
|
|
{
|
|
|
|
mem_debug_dump();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-11-08 09:11:32 +00:00
|
|
|
static char application_save_path[512] = {0};
|
|
|
|
|
|
|
|
const char *engine_savepath(const char *filename, char *buffer, int max)
|
|
|
|
{
|
2008-02-11 21:49:26 +00:00
|
|
|
str_format(buffer, max, "%s/%s", application_save_path, filename);
|
2007-11-08 09:11:32 +00:00
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2008-02-04 07:36:54 +00:00
|
|
|
|
2008-02-10 21:54:52 +00:00
|
|
|
void engine_init(const char *appname)
|
2007-11-08 09:11:32 +00:00
|
|
|
{
|
2008-01-12 12:27:55 +00:00
|
|
|
dbg_msg("engine", "running on %s-%s-%s", CONF_FAMILY_STRING, CONF_PLATFORM_STRING, CONF_ARCH_STRING);
|
|
|
|
#ifdef CONF_ARCH_ENDIAN_LITTLE
|
|
|
|
dbg_msg("engine", "arch is little endian");
|
|
|
|
#elif defined(CONF_ARCH_ENDIAN_BIG)
|
|
|
|
dbg_msg("engine", "arch is big endian");
|
|
|
|
#else
|
|
|
|
dbg_msg("engine", "unknown endian");
|
|
|
|
#endif
|
|
|
|
|
2007-11-08 09:11:32 +00:00
|
|
|
/* init the network */
|
|
|
|
net_init();
|
2008-02-10 21:54:52 +00:00
|
|
|
|
2007-11-08 09:11:32 +00:00
|
|
|
/* create storage location */
|
|
|
|
{
|
|
|
|
char path[1024] = {0};
|
|
|
|
fs_storage_path(appname, application_save_path, sizeof(application_save_path));
|
|
|
|
if(fs_makedir(application_save_path) == 0)
|
|
|
|
{
|
2008-02-11 21:49:26 +00:00
|
|
|
str_format(path, sizeof(path), "%s/screenshots", application_save_path);
|
2007-11-08 09:11:32 +00:00
|
|
|
fs_makedir(path);
|
2008-02-10 16:50:59 +00:00
|
|
|
|
2008-02-11 21:49:26 +00:00
|
|
|
str_format(path, sizeof(path), "%s/maps", application_save_path);
|
2008-02-10 16:50:59 +00:00
|
|
|
fs_makedir(path);
|
2007-11-08 09:11:32 +00:00
|
|
|
}
|
|
|
|
}
|
2008-01-16 22:14:06 +00:00
|
|
|
|
|
|
|
/* init console */
|
|
|
|
console_init();
|
2008-02-04 07:36:54 +00:00
|
|
|
|
|
|
|
MACRO_REGISTER_COMMAND("dbg_dumpmem", "", con_dbg_dumpmem, 0x0);
|
|
|
|
|
2007-11-08 09:11:32 +00:00
|
|
|
|
|
|
|
/* reset the config */
|
|
|
|
config_reset();
|
2008-02-02 12:38:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void engine_parse_arguments(int argc, char **argv)
|
|
|
|
{
|
2007-11-08 09:11:32 +00:00
|
|
|
/* load the configuration */
|
2008-02-02 12:38:36 +00:00
|
|
|
int i;
|
|
|
|
int abs = 0;
|
|
|
|
const char *config_filename = "default.cfg";
|
|
|
|
char buf[1024];
|
|
|
|
for(i = 1; i < argc; i++)
|
2007-11-08 09:11:32 +00:00
|
|
|
{
|
2008-02-02 12:38:36 +00:00
|
|
|
if(argv[i][0] == '-' && argv[i][1] == 'f' && argv[i][2] == 0 && argc - i > 1)
|
2007-11-08 09:11:32 +00:00
|
|
|
{
|
2008-02-02 12:38:36 +00:00
|
|
|
config_filename = argv[i+1];
|
|
|
|
abs = 1;
|
|
|
|
i++;
|
2007-11-08 09:11:32 +00:00
|
|
|
}
|
|
|
|
}
|
2008-02-02 12:38:36 +00:00
|
|
|
|
|
|
|
if(abs)
|
|
|
|
config_load(config_filename);
|
|
|
|
else
|
|
|
|
config_load(engine_savepath(config_filename, buf, sizeof(buf)));
|
2007-11-08 09:11:32 +00:00
|
|
|
|
|
|
|
/* search arguments for overrides */
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for(i = 1; i < argc; i++)
|
|
|
|
config_set(argv[i]);
|
|
|
|
}
|
2008-02-10 21:54:52 +00:00
|
|
|
|
|
|
|
/* set default servers and load from disk*/
|
|
|
|
mastersrv_default();
|
|
|
|
mastersrv_load();
|
2007-11-08 09:11:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void engine_writeconfig()
|
|
|
|
{
|
|
|
|
char buf[1024];
|
|
|
|
config_save(engine_savepath("default.cfg", buf, sizeof(buf)));
|
|
|
|
}
|
2007-12-16 15:33:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
static int perf_tick = 1;
|
|
|
|
static PERFORMACE_INFO *current = 0;
|
|
|
|
|
|
|
|
void perf_init()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void perf_next()
|
|
|
|
{
|
|
|
|
perf_tick++;
|
|
|
|
current = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void perf_start(PERFORMACE_INFO *info)
|
|
|
|
{
|
|
|
|
if(info->tick != perf_tick)
|
|
|
|
{
|
|
|
|
info->parent = current;
|
|
|
|
info->first_child = 0;
|
|
|
|
info->next_child = 0;
|
|
|
|
|
|
|
|
if(info->parent)
|
|
|
|
{
|
|
|
|
info->next_child = info->parent->first_child;
|
|
|
|
info->parent->first_child = info;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->tick = perf_tick;
|
|
|
|
info->biggest = 0;
|
|
|
|
info->total = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
current = info;
|
|
|
|
current->start = time_get();
|
|
|
|
}
|
|
|
|
|
|
|
|
void perf_end()
|
|
|
|
{
|
2008-01-13 11:15:32 +00:00
|
|
|
if(!current)
|
|
|
|
return;
|
|
|
|
|
2007-12-16 16:14:05 +00:00
|
|
|
current->last_delta = time_get()-current->start;
|
|
|
|
current->total += current->last_delta;
|
2007-12-16 15:33:44 +00:00
|
|
|
|
2007-12-16 16:14:05 +00:00
|
|
|
if(current->last_delta > current->biggest)
|
|
|
|
current->biggest = current->last_delta;
|
2007-12-16 15:33:44 +00:00
|
|
|
|
|
|
|
current = current->parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void perf_dump_imp(PERFORMACE_INFO *info, int indent)
|
|
|
|
{
|
|
|
|
char buf[512] = {0};
|
|
|
|
int64 freq = time_freq();
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i = 0; i < indent; i++)
|
|
|
|
buf[i] = ' ';
|
|
|
|
|
2008-02-11 21:49:26 +00:00
|
|
|
str_format(&buf[indent], sizeof(buf)-indent, "%-20s %8.2f %8.2f", info->name, info->total*1000/(float)freq, info->biggest*1000/(float)freq);
|
2007-12-16 15:33:44 +00:00
|
|
|
dbg_msg("perf", "%s", buf);
|
|
|
|
|
|
|
|
info = info->first_child;
|
|
|
|
while(info)
|
|
|
|
{
|
|
|
|
perf_dump_imp(info, indent+2);
|
|
|
|
info = info->next_child;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void perf_dump(PERFORMACE_INFO *top)
|
|
|
|
{
|
|
|
|
perf_dump_imp(top, 0);
|
|
|
|
}
|
2008-02-10 21:54:52 +00:00
|
|
|
|
|
|
|
/* master server functions */
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
NUM_LOOKUP_THREADS=4,
|
|
|
|
|
|
|
|
STATE_PROCESSED=0,
|
|
|
|
STATE_RESULT,
|
|
|
|
STATE_QUERYING
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
char hostname[128];
|
|
|
|
NETADDR4 addr;
|
|
|
|
|
|
|
|
/* these are used for lookups */
|
|
|
|
struct {
|
|
|
|
NETADDR4 addr;
|
|
|
|
int result;
|
|
|
|
void *thread;
|
|
|
|
volatile int state;
|
|
|
|
} lookup;
|
|
|
|
} MASTER_INFO;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
int start;
|
|
|
|
int num;
|
|
|
|
} THREAD_INFO;
|
|
|
|
|
|
|
|
static MASTER_INFO master_servers[MAX_MASTERSERVERS] = {{{0}}};
|
|
|
|
static THREAD_INFO thread_info[NUM_LOOKUP_THREADS];
|
|
|
|
static int needs_update = 0;
|
|
|
|
|
|
|
|
void lookup_thread(void *user)
|
|
|
|
{
|
|
|
|
THREAD_INFO *info = (THREAD_INFO *)user;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i = 0; i < info->num; i++)
|
|
|
|
{
|
|
|
|
int index = info->start+i;
|
|
|
|
master_servers[index].lookup.result = net_host_lookup(master_servers[index].hostname, 8300, &master_servers[index].lookup.addr);
|
|
|
|
master_servers[index].lookup.state = STATE_RESULT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int mastersrv_refresh_addresses()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
dbg_msg("engine/mastersrv", "refreshing master server addresses");
|
|
|
|
|
|
|
|
/* spawn threads that does the lookups */
|
|
|
|
for(i = 0; i < NUM_LOOKUP_THREADS; i++)
|
|
|
|
{
|
|
|
|
thread_info[i].start = MAX_MASTERSERVERS/NUM_LOOKUP_THREADS * i;
|
|
|
|
thread_info[i].num = MAX_MASTERSERVERS/NUM_LOOKUP_THREADS;
|
|
|
|
master_servers[i].lookup.state = STATE_QUERYING;
|
|
|
|
master_servers[i].lookup.thread = thread_create(lookup_thread, &thread_info[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
needs_update = 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mastersrv_update()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* check if we need to update */
|
|
|
|
if(!needs_update)
|
|
|
|
return;
|
|
|
|
needs_update = 0;
|
|
|
|
|
|
|
|
for(i = 0; i < MAX_MASTERSERVERS; i++)
|
|
|
|
{
|
|
|
|
if(master_servers[i].lookup.state == STATE_RESULT)
|
|
|
|
{
|
|
|
|
/* we got a result from the lookup ready */
|
|
|
|
if(master_servers[i].lookup.result == 0)
|
|
|
|
master_servers[i].addr = master_servers[i].lookup.addr;
|
|
|
|
master_servers[i].lookup.state = STATE_PROCESSED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the needs_update flag if we isn't done */
|
|
|
|
if(master_servers[i].lookup.state != STATE_PROCESSED)
|
|
|
|
needs_update = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!needs_update)
|
|
|
|
{
|
|
|
|
dbg_msg("engine/mastersrv", "saving addresses");
|
|
|
|
mastersrv_save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int mastersrv_refreshing()
|
|
|
|
{
|
|
|
|
return needs_update;
|
|
|
|
}
|
|
|
|
|
|
|
|
NETADDR4 mastersrv_get(int index)
|
|
|
|
{
|
|
|
|
return master_servers[index].addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *mastersrv_name(int index)
|
|
|
|
{
|
|
|
|
return master_servers[index].hostname;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mastersrv_dump_servers()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for(i = 0; i < MAX_MASTERSERVERS; i++)
|
|
|
|
{
|
|
|
|
dbg_msg("mastersrv", "#%d = %d.%d.%d.%d", i,
|
|
|
|
master_servers[i].addr.ip[0], master_servers[i].addr.ip[1],
|
|
|
|
master_servers[i].addr.ip[2], master_servers[i].addr.ip[3]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void mastersrv_default()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
mem_zero(master_servers, sizeof(master_servers));
|
|
|
|
for(i = 0; i < MAX_MASTERSERVERS; i++)
|
|
|
|
sprintf(master_servers[i].hostname, "master%d.teewars.com", i+1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int mastersrv_load()
|
|
|
|
{
|
|
|
|
LINEREADER lr;
|
|
|
|
IOHANDLE file;
|
|
|
|
int count = 0;
|
|
|
|
char filename[1024];
|
|
|
|
|
|
|
|
engine_savepath("masters.cfg", filename, sizeof(filename));
|
|
|
|
|
|
|
|
/* try to open file */
|
|
|
|
file = io_open(filename, IOFLAG_READ);
|
|
|
|
if(!file)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
linereader_init(&lr, file);
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
MASTER_INFO info = {{0}};
|
|
|
|
int ip[4];
|
|
|
|
const char *line = linereader_get(&lr);
|
|
|
|
if(!line)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* parse line */
|
|
|
|
if(sscanf(line, "%s %d.%d.%d.%d", info.hostname, &ip[0], &ip[1], &ip[2], &ip[3]) == 5)
|
|
|
|
{
|
|
|
|
info.addr.ip[0] = (unsigned char)ip[0];
|
|
|
|
info.addr.ip[1] = (unsigned char)ip[1];
|
|
|
|
info.addr.ip[2] = (unsigned char)ip[2];
|
|
|
|
info.addr.ip[3] = (unsigned char)ip[3];
|
|
|
|
info.addr.port = 8300;
|
|
|
|
if(count != MAX_MASTERSERVERS)
|
|
|
|
{
|
|
|
|
master_servers[count] = info;
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dbg_msg("engine/mastersrv", "warning: skipped master server '%s' due to limit of %d", line, MAX_MASTERSERVERS);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dbg_msg("engine/mastersrv", "warning: couldn't parse master server '%s'", line);
|
|
|
|
}
|
|
|
|
|
|
|
|
io_close(file);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mastersrv_save()
|
|
|
|
{
|
|
|
|
IOHANDLE file;
|
|
|
|
int i;
|
|
|
|
char filename[1024];
|
|
|
|
|
|
|
|
engine_savepath("masters.cfg", filename, sizeof(filename));
|
|
|
|
|
|
|
|
/* try to open file */
|
|
|
|
file = io_open(filename, IOFLAG_WRITE);
|
|
|
|
if(!file)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for(i = 0; i < MAX_MASTERSERVERS; i++)
|
|
|
|
{
|
|
|
|
char buf[1024];
|
2008-02-11 21:49:26 +00:00
|
|
|
str_format(buf, sizeof(buf), "%s %d.%d.%d.%d\n", master_servers[i].hostname,
|
2008-02-10 21:54:52 +00:00
|
|
|
master_servers[i].addr.ip[0], master_servers[i].addr.ip[1],
|
|
|
|
master_servers[i].addr.ip[2], master_servers[i].addr.ip[3]);
|
|
|
|
|
|
|
|
io_write(file, buf, strlen(buf));
|
|
|
|
}
|
|
|
|
|
|
|
|
io_close(file);
|
|
|
|
return 0;
|
|
|
|
}
|