// 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"
#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() {
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) {
#ifndef MINICORE
hpm_defaults();
HCache_defaults();
#endif
sysinfo_defaults();
console_defaults();
strlib_defaults();
malloc_defaults();
#ifndef MINICORE
libconfig_defaults();
sql_defaults();
timer_defaults();
db_defaults();
socket_defaults();
#endif
}
/*======================================
* 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();
{
int i;
for(i = 0; i < argc; i++) {
if( strcmp(argv[i], "--script-check") == 0 ) {
msg_silent = 0x7; // silence information and status messages
}
}
}
iMalloc->init();// needed for Show* in display_title() [FlavioJS]
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
sysinfo->final();
iMalloc->final();
return retval;
}