OpenFusion/src/main.cpp

160 lines
3.8 KiB
C++
Raw Normal View History

2020-08-18 20:42:30 +00:00
#include "CNLoginServer.hpp"
#include "CNShardServer.hpp"
#include "PlayerManager.hpp"
#include "ChatManager.hpp"
#include "MobManager.hpp"
#include "ItemManager.hpp"
#include "MissionManager.hpp"
2020-08-20 15:43:37 +00:00
#include "NanoManager.hpp"
2020-08-20 21:43:48 +00:00
#include "NPCManager.hpp"
2020-08-29 11:43:33 +00:00
#include "TransportManager.hpp"
#include "Database.hpp"
#include "TableData.hpp"
2020-09-17 22:45:43 +00:00
#include "ChunkManager.hpp"
2020-08-18 20:42:30 +00:00
#include "settings.hpp"
#include "../version.h"
2020-08-19 20:42:44 +00:00
#if defined(__MINGW32__) && !defined(_GLIBCXX_HAS_GTHREADS)
2020-08-18 20:42:30 +00:00
#include "mingw/mingw.thread.h"
#else
#include <thread>
2020-08-18 20:42:30 +00:00
#endif
#include <string>
#include <chrono>
2020-08-30 22:30:15 +00:00
#include <signal.h>
2020-08-18 20:42:30 +00:00
// HACK
#ifdef __has_feature
#if __has_feature(address_sanitizer)
#define __SANITIZE_ADDRESS__ 1
#endif
#endif
CNShardServer *shardServer = nullptr;
std::thread *shardThread = nullptr;
2020-08-19 00:52:02 +00:00
void startShard(CNShardServer* server) {
server->start();
2020-08-18 20:42:30 +00:00
}
2020-08-30 22:30:15 +00:00
#ifndef _WIN32
// terminate gracefully on SIGINT (for gprof)
void terminate(int arg) {
2020-08-30 22:30:15 +00:00
std::cout << "OpenFusion: terminating." << std::endl;
if (shardServer != nullptr && shardThread != nullptr) {
shardServer->kill();
shardThread->join();
}
TableData::cleanup();
2020-09-17 22:45:43 +00:00
ChunkManager::cleanup();
exit(0);
}
2020-08-30 22:30:15 +00:00
void initsignals() {
struct sigaction act;
memset((void*)&act, 0, sizeof(act));
sigemptyset(&act.sa_mask);
// tell the OS to not kill us if you use a broken pipe, just let us know thru recv() or send()
act.sa_handler = SIG_IGN;
if (sigaction(SIGPIPE, &act, NULL) < 0) {
perror("sigaction");
exit(1);
}
act.sa_handler = terminate;
if (sigaction(SIGINT, &act, NULL) < 0) {
perror("sigaction");
exit(1);
}
}
#endif
2020-08-18 20:42:30 +00:00
int main() {
#ifdef _WIN32
WSADATA wsaData;
if (WSAStartup(MAKEWORD(1, 1), &wsaData) != 0) {
std::cerr << "OpenFusion: WSAStartup failed" << std::endl;
exit(EXIT_FAILURE);
2020-08-18 20:42:30 +00:00
}
2020-08-26 19:38:09 +00:00
#else
2020-08-30 22:30:15 +00:00
initsignals();
2020-08-18 20:42:30 +00:00
#endif
settings::init();
std::cout << "[INFO] OpenFusion v" GIT_VERSION << std::endl;
std::cout << "[INFO] Protocol version: " << PROTOCOL_VERSION << std::endl;
2020-08-18 20:42:30 +00:00
std::cout << "[INFO] Intializing Packet Managers..." << std::endl;
TableData::init();
2020-08-18 20:42:30 +00:00
PlayerManager::init();
ChatManager::init();
MobManager::init();
ItemManager::init();
MissionManager::init();
2020-08-20 15:43:37 +00:00
NanoManager::init();
2020-08-20 21:43:48 +00:00
NPCManager::init();
2020-08-29 11:43:33 +00:00
TransportManager::init();
2020-08-18 20:42:30 +00:00
Database::open();
2020-08-18 20:42:30 +00:00
std::cout << "[INFO] Starting Server Threads..." << std::endl;
2020-08-19 01:34:39 +00:00
CNLoginServer loginServer(settings::LOGINPORT);
shardServer = new CNShardServer(settings::SHARDPORT);
2020-08-19 00:52:02 +00:00
shardThread = new std::thread(startShard, (CNShardServer*)shardServer);
2020-08-19 00:52:02 +00:00
loginServer.start();
shardServer->kill();
shardThread->join();
2020-08-18 20:42:30 +00:00
#ifdef _WIN32
WSACleanup();
#endif
return 0;
}
// helper functions
std::string U16toU8(char16_t* src) {
try {
std::wstring_convert<std::codecvt_utf8_utf16<char16_t>,char16_t> convert;
return convert.to_bytes(src);
} catch(const std::exception& e) {
return "";
}
}
// returns number of char16_t that was written at des
size_t U8toU16(std::string src, char16_t* des) {
std::wstring_convert<std::codecvt_utf8_utf16<char16_t>,char16_t> convert;
std::u16string tmp = convert.from_bytes(src);
// copy utf16 string to buffer
memcpy(des, tmp.c_str(), sizeof(char16_t) * tmp.length());
des[tmp.length()] = '\0';
return tmp.length();
}
time_t getTime() {
using namespace std::chrono;
milliseconds value = duration_cast<milliseconds>((time_point_cast<milliseconds>(high_resolution_clock::now())).time_since_epoch());
return (time_t)value.count();
}
time_t getTimestamp() {
using namespace std::chrono;
milliseconds value = duration_cast<milliseconds>((time_point_cast<milliseconds>(system_clock::now())).time_since_epoch());
return (time_t)value.count();
}