/* XMRig * Copyright 2010 Jeff Garzik * Copyright 2012-2014 pooler * Copyright 2014 Lucas Jones * Copyright 2014-2016 Wolf9466 * Copyright 2016 Jay D Dee * Copyright 2016-2017 XMRig * * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include "App.h" #include "Console.h" #include "Cpu.h" #include "crypto/CryptoNight.h" #include "log/ConsoleLog.h" #include "log/FileLog.h" #include "log/Log.h" #include "Mem.h" #include "net/Network.h" #include "Options.h" #include "Platform.h" #include "Summary.h" #include "version.h" #include "workers/Workers.h" #ifdef HAVE_SYSLOG_H # include "log/SysLog.h" #endif App *App::m_self = nullptr; App::App(int argc, char **argv) : m_console(nullptr), m_network(nullptr), m_options(nullptr) { m_self = this; Cpu::init(); m_options = Options::parse(argc, argv); if (!m_options) { return; } Log::init(); if (!m_options->background()) { Log::add(new ConsoleLog(m_options->colors())); m_console = new Console(this); } if (m_options->logFile()) { Log::add(new FileLog(m_options->logFile())); } # ifdef HAVE_SYSLOG_H if (m_options->syslog()) { Log::add(new SysLog()); } # endif Platform::init(); Platform::setProcessPriority(m_options->priority()); m_network = new Network(m_options); uv_signal_init(uv_default_loop(), &m_signal); } App::~App() { delete m_console; } int App::exec() { if (!m_options) { return 0; } uv_signal_start(&m_signal, App::onSignal, SIGHUP); uv_signal_start(&m_signal, App::onSignal, SIGTERM); uv_signal_start(&m_signal, App::onSignal, SIGINT); background(); if (!CryptoNight::init(m_options->algo(), m_options->algoVariant())) { LOG_ERR("\"%s\" hash self-test failed.", m_options->algoName()); return 1; } Mem::allocate(m_options->algo(), m_options->threads(), m_options->doubleHash()); Summary::print(); Workers::start(m_options->affinity(), m_options->priority()); m_network->connect(); const int r = uv_run(uv_default_loop(), UV_RUN_DEFAULT); uv_loop_close(uv_default_loop()); uv_tty_reset_mode(); delete m_network; Options::release(); Mem::release(); Platform::release(); return r; } void App::onConsoleCommand(char command) { switch (command) { case 'h': case 'H': Workers::printHashrate(true); break; case 'p': case 'P': LOG_INFO(m_options->colors() ? "\x1B[01;33mpaused\x1B[0m, press \x1B[01;35mr\x1B[0m to resume" : "paused, press 'r' to resume"); Workers::setEnabled(false); break; case 'r': case 'R': if (!Workers::isEnabled()) { LOG_INFO(m_options->colors() ? "\x1B[01;32mresumed" : "resumed"); Workers::setEnabled(true); } break; case 3: LOG_WARN("Ctrl+C received, exiting"); close(); break; default: break; } } void App::close() { m_network->stop(); Workers::stop(); uv_stop(uv_default_loop()); } void App::onSignal(uv_signal_t *handle, int signum) { switch (signum) { case SIGHUP: LOG_WARN("SIGHUP received, exiting"); break; case SIGTERM: LOG_WARN("SIGTERM received, exiting"); break; case SIGINT: LOG_WARN("SIGINT received, exiting"); break; default: break; } uv_signal_stop(handle); m_self->close(); }