// Copyright (c) Hercules Dev Team, licensed under GNU GPL.
// See the LICENSE file
// Portions Copyright (c) Athena Dev Teams
#define HERCULES_CORE
#include "../config/core.h"
#include "core.h"
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "../common/cbasetypes.h"
#include "../common/console.h"
#include "../common/malloc.h"
#include "../common/mmo.h"
#include "../common/random.h"
#include "../common/showmsg.h"
#include "../common/strlib.h"
#include "../common/sysinfo.h"
#include "../common/nullpo.h"
#ifndef MINICORE
# include "../common/HPM.h"
# include "../common/conf.h"
# include "../common/db.h"
# include "../common/ers.h"
# include "../common/socket.h"
# include "../common/sql.h"
# include "../common/thread.h"
# include "../common/timer.h"
# include "../common/utils.h"
#endif
#ifndef _WIN32
# include <unistd.h>
#else
# include "../common/winapi.h" // Console close event handling
#endif
/// Called when a terminate signal is received.
void (*shutdown_callback)(void) = NULL;
int runflag = CORE_ST_RUN;
int arg_c = 0;
char **arg_v = NULL;
char *SERVER_NAME = NULL;
#ifndef MINICORE // minimalist Core
// Added by Gabuzomeu
//
// This is an implementation of signal() using sigaction() for portability.
// (sigaction() is POSIX; signal() is not.) Taken from Stevens' _Advanced
// Programming in the UNIX Environment_.
//
#ifdef WIN32 // windows don't have SIGPIPE
#define SIGPIPE SIGINT
#endif
#ifndef POSIX
#define compat_signal(signo, func) signal((signo), (func))
#else
sigfunc *compat_signal(int signo, sigfunc *func) {
struct sigaction sact, oact;
sact.sa_handler = func;
sigemptyset(&sact.sa_mask);
sact.sa_flags = 0;
#ifdef SA_INTERRUPT
sact.sa_flags |= SA_INTERRUPT; /* SunOS */
#endif
if (sigaction(signo, &sact, &oact) < 0)
return (SIG_ERR);
return (oact.sa_handler);
}
#endif
/*======================================
* CORE : Console events for Windows
*--------------------------------------*/
#ifdef _WIN32
static BOOL WINAPI console_handler(DWORD c_event) {
switch(c_event) {
case CTRL_CLOSE_EVENT:
case CTRL_LOGOFF_EVENT:
case CTRL_SHUTDOWN_EVENT:
if( shutdown_callback != NULL )
shutdown_callback();
else
runflag = CORE_ST_STOP;// auto-shutdown
break;
default:
return FALSE;
}
return TRUE;
}
static void cevents_init(void) {
if (SetConsoleCtrlHandler(console_handler,TRUE)==FALSE)
ShowWarning ("Unable to install the console handler!\n");
}
#endif
/*======================================
* CORE : Signal Sub Function
*--------------------------------------*/
static void sig_proc(int sn) {
static int is_called = 0;
switch (sn) {
case SIGINT:
case SIGTERM:
if (++is_called > 3)
exit(EXIT_SUCCESS);
if( shutdown_callback != NULL )
shutdown_callback();
else
runflag = CORE_ST_STOP;// auto-shutdown
break;
case SIGSEGV:
case SIGFPE:
do_abort();
// Pass the signal to the system's default handler
compat_signal(sn, SIG_DFL);
raise(sn);
break;
#ifndef _WIN32
case SIGXFSZ:
// ignore and allow it to set errno to EFBIG
ShowWarning ("Max file size reached!\n");
//run_flag = 0; // should we quit?
break;
case SIGPIPE:
//ShowInfo ("Broken pipe found... closing socket\n"); // set to eof in socket.c
break; // does nothing here
#endif
}
}
void signals_init (void) {
compat_signal(SIGTERM, sig_proc);
compat_signal(SIGINT, sig_proc);
#ifndef _DEBUG // need unhandled exceptions to debug on Windows
compat_signal(SIGSEGV, sig_proc);
compat_signal(SIGFPE, sig_proc);
#endif
#ifndef _WIN32
compat_signal(SIGILL, SIG_DFL);
compat_signal(SIGXFSZ, sig_proc);
compat_signal(SIGPIPE, sig_proc);
compat_signal(SIGBUS, SIG_DFL);
compat_signal(SIGTRAP, SIG_DFL);
#endif
}
#endif
/**
* Warns the user if executed as superuser (root)
*/
void usercheck(void) {
if (sysinfo->is_superuser()) {
ShowWarning("You are running Hercules with root privileges, it is not necessary.\n");
}
}
void core_defaults(void) {
nullpo_defaults();
#ifndef MINICORE
hpm_defaults();
HCache_defaults();
#endif
sysinfo_defaults();
console_defaults();
strlib_defaults();
malloc_defaults();
cmdline_defaults();
#ifndef MINICORE
libconfig_defaults();
sql_defaults();
timer_defaults();
db_defaults();
socket_defaults();
#endif
}
/**
* Returns the source (core or plugin name) for the given command-line argument
*/
const char *cmdline_arg_source(struct CmdlineArgData *arg) {
#ifdef MINICORE
return "core";
#else // !MINICORE
return HPM->pid2name(arg->pluginID);
#endif // MINICORE
}
/**
* Defines a command line argument.
*
* @param pluginID the source plugin ID (use HPM_PID_CORE if loading from the core).
* @param name the command line argument name, including the leading '--'.
* @param shortname an optional short form (single-character, it will be prefixed with '-'). Use '\0' to skip.
* @param func the triggered function.
* @param help the help string to be displayed by '--help', if any.
* @param options options associated to the command-line argument. @see enum cmdline_options.
* @return the success status.
*/
bool cmdline_arg_add(unsigned int pluginID, const char *name, char shortname, CmdlineExecFunc func, const char *help, unsigned int options) {
struct CmdlineArgData *data = NULL;
RECREATE(cmdline->args_data, struct CmdlineArgData, ++cmdline->args_data_count);
data = &cmdline->args_data[cmdline->args_data_count-1];
data->pluginID = pluginID;
data->name = aStrdup(name);
data->shortname = shortname;
data->func = func;
data->help = aStrdup(help);
data->options = options;
return true;
}
/**
* Help screen to be displayed by '--help'.
*/
static CMDLINEARG(help)
{
int i;
ShowInfo("Usage: %s [options]\n", SERVER_NAME);
ShowInfo("\n");
ShowInfo("Options:\n");
for (i = 0; i < cmdline->args_data_count; i++) {
struct CmdlineArgData *data = &cmdline->args_data[i];
char altname[16], paramnames[256];
if (data->shortname) {
snprintf(altname, sizeof(altname), " [-%c]", data->shortname);
} else {
*altname = '\0';
}
snprintf(paramnames, sizeof(paramnames), "%s%s%s", data->name, altname, data->options&CMDLINE_OPT_PARAM ? " <name>" : "");
ShowInfo(" %-30s %s [%s]\n", paramnames, data->help ? data->help : "<no description provided>", cmdline->arg_source(data));
}
return false;
}
/**
* Info screen to be displayed by '--version'.
*/
static CMDLINEARG(version)
{
ShowInfo(CL_GREEN"Website/Forum:"CL_RESET"\thttp://hercules.ws/\n");
ShowInfo(CL_GREEN"IRC Channel:"CL_RESET"\tirc://irc.rizon.net/#Hercules\n");
ShowInfo("Open "CL_WHITE"readme.txt"CL_RESET" for more information.\n");
return false;
}
/**
* Checks if there is a value available for the current argument
*
* @param name the current argument's name.
* @param current_arg the current argument's position.
* @param argc the program's argc.
* @return true if a value for the current argument is available on the command line.
*/
bool cmdline_arg_next_value(const char *name, int current_arg, int argc)
{
if (current_arg >= argc-1) {
ShowError("Missing value for option '%s'.\n", name);
return false;
}
return true;
}
/**
* Executes the command line arguments handlers.
*
* @param argc the program's argc
* @param argv the program's argv
* @param options Execution options. Allowed values:
* - CMDLINE_OPT_SILENT: Scans the argv for a command line argument that
* requires the server's silent mode, and triggers it. Invalid command line
* arguments don't cause it to abort. No command handlers are executed.
* - CMDLINE_OPT_PREINIT: Scans the argv for command line arguments with the
* CMDLINE_OPT_PREINIT flag set and executes their handlers. Invalid command
* line arguments don't cause it to abort. Handler's failure causes the
* program to abort.
* - CMDLINE_OPT_NORMAL: Scans the argv for normal command line arguments,
* skipping the pre-init ones, and executes their handlers. Invalid command
* line arguments or handler's failure cause the program to abort.
* @return the amount of command line handlers successfully executed.
*/
int cmdline_exec(int argc, char **argv, unsigned int options)
{
int count = 0, i, j;
for (i = 1; i < argc; i++) {
struct CmdlineArgData *data = NULL;
const char *arg = argv[i];
if (arg[0] != '-') { // All arguments must begin with '-'
ShowError("Invalid option '%s'.\n", argv[i]);
exit(EXIT_FAILURE);
}
if (arg[1] != '-' && strlen(arg) == 2) {
ARR_FIND(0, cmdline->args_data_count, j, cmdline->args_data[j].shortname == arg[1]);
} else {
ARR_FIND(0, cmdline->args_data_count, j, strcmpi(cmdline->args_data[j].name, arg) == 0);
}
if (j == cmdline->args_data_count) {
if (options&(CMDLINE_OPT_SILENT|CMDLINE_OPT_PREINIT))
continue;
ShowError("Unknown option '%s'.\n", arg);
exit(EXIT_FAILURE);
}
data = &cmdline->args_data[j];
if (data->options&CMDLINE_OPT_PARAM) {
if (!cmdline->arg_next_value(arg, i, argc))
exit(EXIT_FAILURE);
i++;
}
if (options&CMDLINE_OPT_SILENT) {
if (data->options&CMDLINE_OPT_SILENT) {
msg_silent = 0x7; // silence information and status messages
break;
}
} else if ((data->options&CMDLINE_OPT_PREINIT) == (options&CMDLINE_OPT_PREINIT)) {
const char *param = NULL;
if (data->options&CMDLINE_OPT_PARAM) {
param = argv[i]; // Already incremented above
}
if (!data->func(arg, param))
exit(EXIT_SUCCESS);
count++;
}
}
return count;
}
/**
* Defines the global command-line arguments.
*/
void cmdline_init(void)
{
#ifdef MINICORE
// Minicore has no HPM. This value isn't used, but the arg_add function requires it, so we're (re)defining it here
#define HPM_PID_CORE ((unsigned int)-1)
#endif
CMDLINEARG_DEF(help, 'h', "Displays this help screen", CMDLINE_OPT_NORMAL);
CMDLINEARG_DEF(version, 'v', "Displays the server's version.", CMDLINE_OPT_NORMAL);
#ifndef MINICORE
CMDLINEARG_DEF2(load-plugin, loadplugin, "Loads an additional plugin (can be repeated).", CMDLINE_OPT_PARAM|CMDLINE_OPT_PREINIT);
#endif // !MINICORE
cmdline_args_init_local();
}
void cmdline_final(void)
{
int i;
for (i = 0; i < cmdline->args_data_count; i++) {
aFree(cmdline->args_data[i].name);
aFree(cmdline->args_data[i].help);
}
if (cmdline->args_data)
aFree(cmdline->args_data);
}
struct cmdline_interface cmdline_s;
void cmdline_defaults(void)
{
cmdline = &cmdline_s;
cmdline->args_data = NULL;
cmdline->args_data_count = 0;
cmdline->init = cmdline_init;
cmdline->final = cmdline_final;
cmdline->arg_add = cmdline_arg_add;
cmdline->exec = cmdline_exec;
cmdline->arg_next_value = cmdline_arg_next_value;
cmdline->arg_source = cmdline_arg_source;
}
/*======================================
* CORE : MAINROUTINE
*--------------------------------------*/
int main (int argc, char **argv) {
int retval = EXIT_SUCCESS;
{// initialize program arguments
char *p1 = SERVER_NAME = argv[0];
char *p2 = p1;
while ((p1 = strchr(p2, '/')) != NULL || (p1 = strchr(p2, '\\')) != NULL) {
SERVER_NAME = ++p1;
p2 = p1;
}
arg_c = argc;
arg_v = argv;
}
core_defaults();
iMalloc->init();// needed for Show* in display_title() [FlavioJS]
cmdline->init();
cmdline->exec(argc, argv, CMDLINE_OPT_SILENT);
iMalloc->init_messages(); // Initialization messages (after buying us some time to suppress them if needed)
sysinfo->init();
if (!(msg_silent&0x1))
console->display_title();
usercheck();
#ifdef MINICORE // minimalist Core
do_init(argc,argv);
do_final();
#else// not MINICORE
set_server_type();
Sql_Init();
rathread_init();
DB->init();
signals_init();
#ifdef _WIN32
cevents_init();
#endif
timer->init();
/* timer first */
rnd_init();
srand((unsigned int)timer->gettick());
console->init();
HCache->init();
HPM->init();
sockt->init();
do_init(argc,argv);
{// Main runtime cycle
int next;
while (runflag != CORE_ST_STOP) {
next = timer->perform(timer->gettick_nocache());
sockt->perform(next);
}
}
console->final();
retval = do_final();
HPM->final();
timer->final();
sockt->final();
DB->final();
rathread_final();
ers_final();
#endif
cmdline->final();
//sysinfo->final(); Called by iMalloc->final()
iMalloc->final();
return retval;
}