blob: d16049d043e7666e9196d5fa0c7c76a5fb2ff87e [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/**
* Copyright (c) 2015 Regents of the University of California
*
* This file is part of NFD (Named Data Networking Forwarding Daemon) Android.
* See AUTHORS.md for complete list of NFD Android authors and contributors.
*
* NFD Android 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.
*
* NFD Android 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
* NFD Android, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#include "nfd-wrapper.hpp"
#include "daemon/nfd.hpp"
#include "rib/nrd.hpp"
#include "core/global-io.hpp"
#include "core/config-file.hpp"
#include "core/logger.hpp"
#include "core/privilege-helper.hpp"
#include <stdlib.h>
#include <boost/property_tree/info_parser.hpp>
#include <boost/thread.hpp>
#include <mutex>
NFD_LOG_INIT("NfdWrapper");
namespace nfd {
// A little bit of cheating to make sure NFD can be properly restarted
namespace scheduler {
// defined in scheduler.cpp
void
resetGlobalScheduler();
} // namespace scheduler
void
resetGlobalIoService();
class Runner
{
public:
Runner()
: m_io(nullptr)
{
std::string initialConfig =
"general\n"
"{\n"
"}\n"
"\n"
"log\n"
"{\n"
" default_level ALL\n"
" NameTree INFO\n"
" BestRouteStrategy2 INFO\n"
" InternalFace INFO\n"
" Forwarder INFO\n"
" ContentStore INFO\n"
" DeadNonceList INFO\n"
"}\n"
"tables\n"
"{\n"
" cs_max_packets 100\n"
"\n"
" strategy_choice\n"
" {\n"
" / /localhost/nfd/strategy/best-route\n"
" /localhost /localhost/nfd/strategy/broadcast\n"
" /localhost/nfd /localhost/nfd/strategy/best-route\n"
" /ndn/broadcast /localhost/nfd/strategy/broadcast\n"
" }\n"
"}\n"
"\n"
"face_system\n"
"{\n"
" tcp\n"
" {\n"
" port 6363\n"
" }\n"
"\n"
" udp\n"
" {\n"
" port 6363\n"
" idle_timeout 600\n"
" keep_alive_interval 25\n"
" mcast no\n"
" }\n"
" websocket\n"
" {\n"
" listen yes\n"
" port 9696\n"
" enable_v4 yes\n"
" enable_v6 yes\n"
" }\n"
"}\n"
"\n"
"authorizations\n"
"{\n"
" authorize\n"
" {\n"
" certfile any\n"
" privileges\n"
" {\n"
" faces\n"
" fib\n"
" strategy-choice\n"
" }\n"
" }\n"
"}\n"
"\n"
"rib\n"
"{\n"
" localhost_security\n"
" {\n"
" trust-anchor\n"
" {\n"
" type any\n"
" }\n"
" }\n"
"\n"
" remote_register\n"
" {\n"
" cost 15\n"
" timeout 10000\n"
" retry 0\n"
" refresh_interval 300\n"
" }\n"
"}\n"
"\n";
std::istringstream input(initialConfig);
boost::property_tree::read_info(input, m_config);
std::unique_lock<std::mutex> lock(m_pointerMutex);
m_nfd.reset(new Nfd(m_config, m_keyChain));
m_nrd.reset(new rib::Nrd(m_config, m_keyChain));
m_nfd->initialize();
m_nrd->initialize();
}
~Runner()
{
stop();
m_io->reset();
}
void
start()
{
{
std::unique_lock<std::mutex> lock(m_pointerMutex);
m_io = &getGlobalIoService();
}
m_io->run();
m_io->reset();
}
void
stop()
{
std::unique_lock<std::mutex> lock(m_pointerMutex);
m_io->post([this] {
m_io->stop();
this->m_nrd.reset();
this->m_nfd.reset();
});
}
private:
std::mutex m_pointerMutex;
boost::asio::io_service* m_io;
ndn::KeyChain m_keyChain;
unique_ptr<Nfd> m_nfd; // will use globalIoService
unique_ptr<rib::Nrd> m_nrd; // will use globalIoService
nfd::ConfigSection m_config;
};
static unique_ptr<Runner> g_runner;
static boost::thread g_thread;
} // namespace nfd
JNIEXPORT void JNICALL
Java_net_named_1data_nfd_service_NfdService_startNfd(JNIEnv* env, jclass, jstring homePathJ)
{
if (nfd::g_runner.get() == nullptr) {
// set/update HOME environment variable
const char* homePath = env->GetStringUTFChars(homePathJ, nullptr);
::setenv("HOME", homePath, true);
env->ReleaseStringUTFChars(homePathJ, homePath);
NFD_LOG_INFO("Use [" << homePath << "] as a security storage");
nfd::g_thread = boost::thread([] {
NFD_LOG_INFO("Starting NFD...");
try {
nfd::g_runner.reset(new nfd::Runner());
nfd::g_runner->start();
}
catch (const std::exception& e) {
NFD_LOG_FATAL(e.what());
}
catch (const nfd::PrivilegeHelper::Error& e) {
NFD_LOG_FATAL("PrivilegeHelper: " << e.what());
}
catch (...) {
NFD_LOG_FATAL("Unknown fatal error");
}
nfd::g_runner.reset();
nfd::scheduler::resetGlobalScheduler();
nfd::resetGlobalIoService();
NFD_LOG_INFO("NFD stopped");
});
}
}
JNIEXPORT void JNICALL
Java_net_named_1data_nfd_service_NfdService_stopNfd(JNIEnv*, jclass)
{
if (nfd::g_runner.get() != nullptr) {
NFD_LOG_INFO("Stopping NFD...");
nfd::g_runner->stop();
// do not block anything
}
}