2018-05-27 18:03:10 +00:00
|
|
|
#include <llarp.h>
|
2018-09-19 13:02:55 +00:00
|
|
|
#include <llarp/logger.hpp>
|
2018-07-13 13:36:51 +00:00
|
|
|
#include <signal.h>
|
2018-09-19 13:02:55 +00:00
|
|
|
#include <getopt.h>
|
2018-07-20 04:50:28 +00:00
|
|
|
#include <string>
|
2018-09-19 13:02:55 +00:00
|
|
|
#include <iostream>
|
2018-09-29 08:16:54 +00:00
|
|
|
#ifndef _MSC_VER
|
2018-09-19 13:02:55 +00:00
|
|
|
#include <libgen.h>
|
2018-09-29 08:16:54 +00:00
|
|
|
#endif
|
2018-09-19 13:02:55 +00:00
|
|
|
#include "fs.hpp"
|
2018-05-20 16:15:16 +00:00
|
|
|
|
2018-07-30 04:38:14 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
#define wmin(x, y) (((x) < (y)) ? (x) : (y))
|
|
|
|
#define MIN wmin
|
|
|
|
#endif
|
|
|
|
|
2018-05-27 18:03:10 +00:00
|
|
|
struct llarp_main *ctx = 0;
|
2018-04-30 16:14:20 +00:00
|
|
|
|
2018-05-22 15:54:19 +00:00
|
|
|
void
|
|
|
|
handle_signal(int sig)
|
2018-05-18 17:50:21 +00:00
|
|
|
{
|
2018-05-27 17:44:01 +00:00
|
|
|
if(ctx)
|
2018-05-27 18:03:10 +00:00
|
|
|
llarp_main_signal(ctx, sig);
|
2018-05-18 17:50:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-19 13:02:55 +00:00
|
|
|
int
|
|
|
|
printHelp(const char *argv0, int code = 1)
|
|
|
|
{
|
|
|
|
std::cout << "usage: " << argv0 << " [-h] [-g] config.ini" << std::endl;
|
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
2018-09-29 08:16:54 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
int
|
|
|
|
startWinsock()
|
|
|
|
{
|
|
|
|
WSADATA wsockd;
|
|
|
|
int err;
|
|
|
|
// We used to defer starting winsock until
|
|
|
|
// we got to the iocp event loop
|
|
|
|
// but getaddrinfo(3) requires that winsock be in core already
|
|
|
|
err = ::WSAStartup(MAKEWORD(2, 2), &wsockd);
|
|
|
|
if(err)
|
|
|
|
{
|
|
|
|
perror("Failed to start Windows Sockets");
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-05-22 15:54:19 +00:00
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
|
|
|
{
|
2018-07-20 04:50:28 +00:00
|
|
|
bool multiThreaded = true;
|
|
|
|
const char *singleThreadVar = getenv("LLARP_SHADOW");
|
|
|
|
if(singleThreadVar && std::string(singleThreadVar) == "1")
|
|
|
|
{
|
|
|
|
multiThreaded = false;
|
|
|
|
}
|
2018-07-27 00:21:57 +00:00
|
|
|
|
2018-09-29 08:16:54 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
if(startWinsock())
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
|
2018-09-19 13:02:55 +00:00
|
|
|
int opt = 0;
|
|
|
|
bool genconfigOnly = false;
|
|
|
|
while((opt = getopt(argc, argv, "hg")) != -1)
|
|
|
|
{
|
|
|
|
switch(opt)
|
|
|
|
{
|
|
|
|
case 'h':
|
|
|
|
return printHelp(argv[0], 0);
|
|
|
|
case 'g':
|
|
|
|
genconfigOnly = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return printHelp(argv[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-19 13:15:07 +00:00
|
|
|
std::string conffname;
|
2018-09-19 13:02:55 +00:00
|
|
|
|
|
|
|
if(optind < argc)
|
|
|
|
{
|
|
|
|
// when we have an explicit filepath
|
2018-09-19 13:10:14 +00:00
|
|
|
fs::path fname = fs::path(argv[optind]);
|
|
|
|
fs::path basedir = fname.parent_path();
|
2018-09-19 13:17:15 +00:00
|
|
|
conffname = fname.string();
|
2018-09-19 13:15:07 +00:00
|
|
|
if(basedir.string().empty())
|
2018-09-19 13:10:14 +00:00
|
|
|
{
|
2018-09-19 13:15:07 +00:00
|
|
|
if(!llarp_ensure_config(fname.string().c_str(), nullptr, genconfigOnly))
|
2018-09-19 13:10:14 +00:00
|
|
|
return 1;
|
2018-09-19 13:15:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::error_code ec;
|
|
|
|
if(!fs::create_directories(basedir, ec))
|
|
|
|
{
|
|
|
|
if(ec)
|
|
|
|
{
|
|
|
|
llarp::LogError("failed to create '", basedir.string(),
|
|
|
|
"': ", ec.message());
|
|
|
|
return 1;
|
|
|
|
}
|
2018-09-19 13:10:14 +00:00
|
|
|
}
|
2018-09-19 13:15:07 +00:00
|
|
|
if(!llarp_ensure_config(fname.string().c_str(), basedir.string().c_str(),
|
|
|
|
genconfigOnly))
|
|
|
|
return 1;
|
2018-09-19 13:10:14 +00:00
|
|
|
}
|
2018-09-19 13:02:55 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// no explicit config file provided
|
|
|
|
#ifdef _WIN32
|
|
|
|
fs::path homedir = fs::path(getenv("APPDATA"));
|
|
|
|
#else
|
|
|
|
fs::path homedir = fs::path(getenv("HOME"));
|
|
|
|
#endif
|
|
|
|
fs::path basepath = homedir / fs::path(".lokinet");
|
|
|
|
fs::path fpath = basepath / "lokinet.ini";
|
|
|
|
|
|
|
|
std::error_code ec;
|
2018-10-01 02:08:03 +00:00
|
|
|
// These paths are guaranteed to exist - $APPDATA or $HOME
|
|
|
|
// so only create .lokinet/*
|
2018-09-29 08:16:54 +00:00
|
|
|
if(!fs::create_directory(basepath, ec))
|
2018-09-19 13:02:55 +00:00
|
|
|
{
|
2018-09-19 13:32:50 +00:00
|
|
|
if(ec)
|
|
|
|
{
|
|
|
|
llarp::LogError("failed to create '", basepath.string(),
|
|
|
|
"': ", ec.message());
|
|
|
|
return 1;
|
|
|
|
}
|
2018-09-19 13:02:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(!llarp_ensure_config(fpath.string().c_str(), basepath.string().c_str(),
|
|
|
|
genconfigOnly))
|
|
|
|
return 1;
|
2018-09-19 13:15:07 +00:00
|
|
|
conffname = fpath.string();
|
2018-09-19 13:02:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(genconfigOnly)
|
|
|
|
return 0;
|
2018-07-13 13:36:51 +00:00
|
|
|
|
2018-09-19 13:15:07 +00:00
|
|
|
ctx = llarp_main_init(conffname.c_str(), multiThreaded);
|
2018-05-27 19:13:25 +00:00
|
|
|
int code = 1;
|
|
|
|
if(ctx)
|
2018-05-27 18:03:10 +00:00
|
|
|
{
|
|
|
|
signal(SIGINT, handle_signal);
|
2018-09-19 11:57:07 +00:00
|
|
|
#ifndef _WIN32
|
2018-09-17 11:47:34 +00:00
|
|
|
signal(SIGHUP, handle_signal);
|
2018-09-19 11:57:07 +00:00
|
|
|
#endif
|
2018-05-27 19:13:25 +00:00
|
|
|
code = llarp_main_run(ctx);
|
2018-07-20 04:50:28 +00:00
|
|
|
llarp_main_free(ctx);
|
2018-05-27 18:03:10 +00:00
|
|
|
}
|
2018-08-12 17:22:29 +00:00
|
|
|
exit(code);
|
2018-09-29 08:16:54 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
::WSACleanup();
|
|
|
|
#endif
|
2018-05-27 19:13:25 +00:00
|
|
|
return code;
|
2017-09-28 17:02:05 +00:00
|
|
|
}
|