/* vim: set expandtab ts=4 sw=4: */ /* * You may redistribute this program 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 "admin/Admin.h" #include "admin/AdminLog.h" #include "admin/angel/Core.h" #include "admin/angel/InterfaceWaiter.h" #include "admin/AuthorizedPasswords.h" #include "benc/Int.h" #include "benc/serialization/standard/BencMessageReader.h" #include "benc/serialization/standard/BencMessageWriter.h" #include "crypto/AddressCalc.h" #include "crypto/random/Random.h" #include "crypto/random/nanotime/NanotimeEntropyProvider.h" #include "crypto/Sign_admin.h" #include "subnode/SubnodePathfinder.h" #include "subnode/SupernodeHunter_admin.h" #include "subnode/ReachabilityCollector_admin.h" #ifndef SUBNODE #include "dht/Pathfinder.h" #endif #include "exception/Er.h" #include "interface/Iface.h" #include "util/events/UDPAddrIface.h" #include "interface/tuntap/TUNInterface.h" #include "interface/tuntap/SocketInterface.h" #include "interface/tuntap/SocketWrapper.h" #include "interface/UDPInterface_admin.h" #ifdef HAS_ETH_INTERFACE #include "interface/ETHInterface_admin.h" #endif #include "net/InterfaceController_admin.h" #include "interface/addressable/PacketHeaderToUDPAddrIface.h" #include "interface/addressable/AddrIfaceMuxer.h" #include "interface/tuntap/TUNMessageType.h" #include "interface/ASynchronizer.h" #include "memory/Allocator.h" #include "memory/MallocAllocator.h" #include "memory/Allocator_admin.h" #include "net/SwitchPinger_admin.h" #include "net/UpperDistributor_admin.h" #include "rust/cjdns_sys/Rffi.h" #define NumberCompress_OLD_CODE #include "switch/NumberCompress.h" #include "tunnel/IpTunnel_admin.h" #include "tunnel/RouteGen_admin.h" #include "util/events/EventBase.h" #include "util/events/Pipe.h" #include "util/events/PipeServer.h" #include "util/events/Timeout.h" #include "util/Hex.h" #include "util/log/FileWriterLog.h" #include "util/log/IndirectLog.h" #include "util/platform/netdev/NetDev.h" #include "util/Security_admin.h" #include "util/Security.h" #include "util/version/Version.h" #include "util/GlobalConfig.h" #include "net/SessionManager_admin.h" #include "wire/SwitchHeader.h" #include "wire/CryptoHeader.h" #include "wire/Headers.h" #include "net/NetCore.h" #include #include #include // Failsafe: abort if more than 2^23 bytes are allocated (8MB) #define ALLOCATOR_FAILSAFE (1<<23) // TODO(cjd): we need to begin detecting MTU and informing the OS properly! /** * The worst possible packet overhead, we're in session setup with the endpoint. */ #define WORST_CASE_OVERHEAD ( \ Headers_IP4Header_SIZE \ + Headers_UDPHeader_SIZE \ + 4 /* Nonce */ \ + 16 /* Poly1305 authenticator */ \ + SwitchHeader_SIZE \ + CryptoHeader_SIZE \ + 4 /* Handle */ \ + DataHeader_SIZE \ ) /** The default MTU, assuming the external MTU is 1492 (common for PPPoE DSL) */ #define DEFAULT_MTU ( 1492 - WORST_CASE_OVERHEAD ) static void adminPing(Dict* input, void* vadmin, String* txid, struct Allocator* requestAlloc) { Dict d = Dict_CONST(String_CONST("q"), String_OBJ(String_CONST("pong")), NULL); Admin_sendMessage(&d, txid, (struct Admin*) vadmin); } static void adminPid(Dict* input, void* vadmin, String* txid, struct Allocator* requestAlloc) { int pid = getpid(); Dict d = Dict_CONST(String_CONST("pid"), Int_OBJ(pid), NULL); Admin_sendMessage(&d, txid, (struct Admin*) vadmin); } struct Context { struct Allocator* alloc; struct Admin* admin; struct Log* logger; struct EventBase* base; struct NetCore* nc; struct IpTunnel* ipTunnel; struct EncodingScheme* encodingScheme; struct GlobalConfig* globalConf; struct Iface* tunDevice; struct Allocator* tunAlloc; Identity }; static void shutdown(void* vcontext) { struct Context* context = Identity_check((struct Context*) vcontext); Allocator_free(context->alloc); } static void adminExit(Dict* input, void* vcontext, String* txid, struct Allocator* requestAlloc) { struct Context* context = Identity_check((struct Context*) vcontext); Log_info(context->logger, "Got request to exit"); Dict d = Dict_CONST(String_CONST("error"), String_OBJ(String_CONST("none")), NULL); Admin_sendMessage(&d, txid, context->admin); Timeout_setTimeout(shutdown, context, 1, context->base, context->alloc); } static void sendResponse(String* error, struct Admin* admin, String* txid, struct Allocator* tempAlloc) { Dict* output = Dict_new(tempAlloc); Dict_putStringC(output, "error", error, tempAlloc); Admin_sendMessage(output, txid, admin); } static Er_DEFUN(void initSocket2(String* socketFullPath, struct Context* ctx, uint8_t addressPrefix)) { Log_debug(ctx->logger, "Initializing socket: %s;", socketFullPath->bytes); if (ctx->tunDevice) { Iface_unplumb(ctx->tunDevice, &ctx->nc->tunAdapt->tunIf); Allocator_free(ctx->tunAlloc); ctx->tunDevice = NULL; } ctx->tunAlloc = Allocator_child(ctx->alloc); struct Iface* rawSocketIf = Er(SocketInterface_new( socketFullPath->bytes, ctx->base, ctx->logger, ctx->tunAlloc)); struct SocketWrapper* sw = SocketWrapper_new(ctx->tunAlloc, ctx->logger); Iface_plumb(&sw->externalIf, rawSocketIf); ctx->tunDevice = &sw->internalIf; Iface_plumb(ctx->tunDevice, &ctx->nc->tunAdapt->tunIf); Er(SocketWrapper_addAddress( &sw->externalIf, ctx->nc->myAddress->ip6.bytes, ctx->logger, ctx->alloc)); Er(SocketWrapper_setMTU(&sw->externalIf, DEFAULT_MTU, ctx->logger, ctx->alloc)); Er_ret(); } static Er_DEFUN(void initTunnel2(String* desiredDeviceName, struct Context* ctx, uint8_t addressPrefix, struct Allocator* errAlloc)) { Log_debug(ctx->logger, "Initializing TUN device [%s]", (desiredDeviceName) ? desiredDeviceName->bytes : ""); char assignedTunName[TUNInterface_IFNAMSIZ]; char* desiredName = (desiredDeviceName) ? desiredDeviceName->bytes : NULL; if (ctx->tunDevice) { Iface_unplumb(&ctx->nc->tunAdapt->tunIf, ctx->tunDevice); Allocator_free(ctx->tunAlloc); ctx->tunDevice = NULL; } ctx->tunAlloc = Allocator_child(ctx->alloc); ctx->tunDevice = Er(TUNInterface_new( desiredName, assignedTunName, 0, ctx->base, ctx->logger, ctx->tunAlloc)); Iface_plumb(ctx->tunDevice, &ctx->nc->tunAdapt->tunIf); GlobalConfig_setTunName(ctx->globalConf, String_CONST(assignedTunName)); struct Sockaddr* myAddr = Sockaddr_fromBytes(ctx->nc->myAddress->ip6.bytes, Sockaddr_AF_INET6, ctx->tunAlloc); myAddr->prefix = addressPrefix; myAddr->flags |= Sockaddr_flags_PREFIX; Er(NetDev_addAddress(assignedTunName, myAddr, ctx->logger, errAlloc)); Er(NetDev_setMTU(assignedTunName, DEFAULT_MTU, ctx->logger, errAlloc)); Er_ret(); } static void initTunfd(Dict* args, void* vcontext, String* txid, struct Allocator* requestAlloc) { struct Context* ctx = Identity_check((struct Context*) vcontext); int64_t* tunfd = Dict_getIntC(args, "tunfd"); int64_t* tuntype = Dict_getIntC(args, "type"); if (!tunfd || *tunfd < 0) { String* error = String_printf(requestAlloc, "Invalid tunfd"); sendResponse(error, ctx->admin, txid, requestAlloc); return; } int fileno = *tunfd; int type = (tuntype) ? *tuntype : TUNMessageType_guess(); struct Allocator* tunAlloc = Allocator_child(ctx->alloc); struct Er_Ret* er = NULL; struct Pipe* p = Er_check(&er, Pipe_forFd(fileno, false, ctx->base, ctx->logger, tunAlloc)); if (er) { Log_debug(ctx->logger, "Failed to create pipe [%s]", er->message); String* error = String_printf(requestAlloc, "Failed to configure tunnel [%s]", er->message); sendResponse(error, ctx->admin, txid, requestAlloc); return; } struct Iface* iface = NULL; if (type == TUNMessageType_NONE) { RTypes_IfWrapper_t aw = Rffi_android_create(tunAlloc); Iface_plumb(aw.external, &p->iface); iface = aw.internal; } else { iface = &p->iface; } if (ctx->tunDevice) { Iface_unplumb(&ctx->nc->tunAdapt->tunIf, ctx->tunDevice); Allocator_free(ctx->tunAlloc); ctx->tunDevice = NULL; } Assert_true(!ctx->nc->tunAdapt->tunIf.connectedIf); ctx->tunAlloc = tunAlloc; ctx->tunDevice = iface; Iface_plumb(ctx->tunDevice, &ctx->nc->tunAdapt->tunIf); sendResponse(String_CONST("none"), ctx->admin, txid, requestAlloc); } static void stopTun(Dict* args, void* vcontext, String* txid, struct Allocator* requestAlloc) { struct Context* ctx = Identity_check((struct Context*) vcontext); if (ctx->tunDevice) { Iface_unplumb(&ctx->nc->tunAdapt->tunIf, ctx->tunDevice); Allocator_free(ctx->tunAlloc); ctx->tunDevice = NULL; sendResponse(String_new("none", requestAlloc), ctx->admin, txid, requestAlloc); } else { sendResponse( String_new("no tun currently configured", requestAlloc), ctx->admin, txid, requestAlloc); } } static void initTunnel(Dict* args, void* vcontext, String* txid, struct Allocator* requestAlloc) { struct Context* const ctx = Identity_check((struct Context*) vcontext); String* desiredName = Dict_getStringC(args, "desiredTunName"); struct Er_Ret* er = NULL; Er_check(&er, initTunnel2(desiredName, ctx, AddressCalc_ADDRESS_PREFIX_BITS, requestAlloc)); if (er) { String* error = String_printf(requestAlloc, "Failed to configure tunnel [%s]", er->message); sendResponse(error, ctx->admin, txid, requestAlloc); return; } sendResponse(String_CONST("none"), ctx->admin, txid, requestAlloc); } static void initSocket(Dict* args, void* vcontext, String* txid, struct Allocator* requestAlloc) { struct Context* const ctx = Identity_check((struct Context*) vcontext); String* socketFullPath = Dict_getStringC(args, "socketFullPath"); struct Er_Ret* er = NULL; Er_check(&er, initSocket2(socketFullPath, ctx, AddressCalc_ADDRESS_PREFIX_BITS)); if (er) { String* error = String_printf(requestAlloc, "Failed to configure socket [%s]", er->message); sendResponse(error, ctx->admin, txid, requestAlloc); return; } sendResponse(String_CONST("none"), ctx->admin, txid, requestAlloc); } static void nodeInfo(Dict* args, void* vcontext, String* txid, struct Allocator* requestAlloc) { struct Context* const ctx = Identity_check((struct Context*) vcontext); String* myAddr = Address_toStringKey(ctx->nc->myAddress, requestAlloc); String* schemeStr = EncodingScheme_serialize(ctx->encodingScheme, requestAlloc); List* schemeList = EncodingScheme_asList(ctx->encodingScheme, requestAlloc); Dict* out = Dict_new(requestAlloc); Dict_putStringC(out, "myAddr", myAddr, requestAlloc); String* myIp6 = String_newBinary(NULL, 41, requestAlloc); Address_printIp(myIp6->bytes, ctx->nc->myAddress); myIp6->len = CString_strlen(myIp6->bytes); Dict_putStringC(out, "myIp6", myIp6, requestAlloc); char* schemeHex = Hex_print(schemeStr->bytes, schemeStr->len, requestAlloc); Dict_putStringCC(out, "compressedSchemeHex", schemeHex, requestAlloc); Dict_putListC(out, "encodingScheme", schemeList, requestAlloc); Dict_putIntC(out, "version", Version_CURRENT_PROTOCOL, requestAlloc); Dict_putStringCC(out, "error", "none", requestAlloc); Admin_sendMessage(out, txid, ctx->admin); } void Core_init(struct Allocator* alloc, struct Log* logger, struct EventBase* eventBase, uint8_t privateKey[32], struct Admin* admin, struct Random* rand, struct Except* eh, struct FakeNetwork* fakeNet, bool noSec) { struct Security* sec = NULL; if (!noSec) { sec = Security_new(alloc, logger, eventBase); } struct GlobalConfig* globalConf = GlobalConfig_new(alloc); struct NetCore* nc = NetCore_new(privateKey, alloc, eventBase, rand, logger, !Defined(NOISE_NO)); struct RouteGen* rg = RouteGen_new(alloc, logger); struct IpTunnel* ipTunnel = IpTunnel_new(logger, eventBase, alloc, rand, rg, globalConf); Iface_plumb(&nc->tunAdapt->ipTunnelIf, &ipTunnel->tunInterface); Iface_plumb(&nc->upper->ipTunnelIf, &ipTunnel->nodeInterface); struct EncodingScheme* encodingScheme = NumberCompress_defineScheme(alloc); // The link between the Pathfinder and the core needs to be asynchronous. struct SubnodePathfinder* spf = SubnodePathfinder_new( alloc, logger, eventBase, rand, nc->myAddress, privateKey, encodingScheme); struct ASynchronizer* spfAsync = ASynchronizer_new(alloc, eventBase, logger); Iface_plumb(&spfAsync->ifA, &spf->eventIf); EventEmitter_regPathfinderIface(nc->ee, &spfAsync->ifB); #ifndef SUBNODE struct Pathfinder* opf = Pathfinder_register(alloc, logger, eventBase, rand, admin); struct ASynchronizer* opfAsync = ASynchronizer_new(alloc, eventBase, logger); Iface_plumb(&opfAsync->ifA, &opf->eventIf); EventEmitter_regPathfinderIface(nc->ee, &opfAsync->ifB); #endif SubnodePathfinder_start(spf); // ------------------- Register RPC functions ----------------------- // UpperDistributor_admin_register(nc->upper, admin, alloc); RouteGen_admin_register(rg, admin, alloc); InterfaceController_admin_register(nc->ifController, admin, alloc); SwitchPinger_admin_register(nc->sp, admin, alloc); UDPInterface_admin_register( eventBase, alloc, logger, admin, nc->ifController, fakeNet, globalConf); #ifdef HAS_ETH_INTERFACE ETHInterface_admin_register(eventBase, alloc, logger, admin, nc->ifController); #endif SupernodeHunter_admin_register(spf->snh, admin, alloc); ReachabilityCollector_admin_register(spf->rc, admin, alloc); AuthorizedPasswords_init(admin, nc->ca, alloc); Admin_registerFunction("ping", adminPing, admin, false, NULL, admin); if (!noSec) { Security_admin_register(alloc, logger, sec, admin); } IpTunnel_admin_register(ipTunnel, admin, alloc); SessionManager_admin_register(nc->sm, admin, alloc); Allocator_admin_register(alloc, admin); Sign_admin_register(privateKey, admin, rand, alloc); struct Context* ctx = Allocator_calloc(alloc, sizeof(struct Context), 1); Identity_set(ctx); ctx->alloc = alloc; ctx->admin = admin; ctx->logger = logger; ctx->base = eventBase; ctx->ipTunnel = ipTunnel; ctx->nc = nc; ctx->encodingScheme = encodingScheme; ctx->globalConf = globalConf; Admin_registerFunction("Core_exit", adminExit, ctx, true, NULL, admin); Admin_registerFunction("Core_pid", adminPid, admin, false, NULL, admin); Admin_registerFunction("Core_initTunnel", initTunnel, ctx, true, ((struct Admin_FunctionArg[]) { { .name = "desiredTunName", .required = 0, .type = "String" } }), admin); Admin_registerFunction("Core_initTunfd", initTunfd, ctx, true, ((struct Admin_FunctionArg[]) { { .name = "tunfd", .required = 1, .type = "Int" }, { .name = "type", .required = 0, .type = "Int" } }), admin); Admin_registerFunction("Core_stopTun", stopTun, ctx, true, NULL, admin); Admin_registerFunction("Core_nodeInfo", nodeInfo, ctx, false, NULL, admin); Admin_registerFunction("Core_initSocket", initSocket, ctx, true, ((struct Admin_FunctionArg[]) { { .name = "socketFullPath", .required = 1, .type = "String" }, { .name = "socketAttemptToCreate", .required = 0, .type = "Int" } }), admin); } int Core_main(int argc, char** argv) { struct Except* eh = NULL; if (argc != 3) { Except_throw(eh, "This is internal to cjdns and shouldn't started manually."); } struct Allocator* alloc = MallocAllocator_new(ALLOCATOR_FAILSAFE); struct Log* preLogger = FileWriterLog_new(stderr, alloc); Rffi_setLogger(preLogger); struct EventBase* eventBase = EventBase_new(alloc); // -------------------- Setup the Pre-Logger ---------------------- // struct Log* logger = IndirectLog_new(alloc); IndirectLog_set(logger, preLogger); // -------------------- Setup the PRNG ---------------------- // struct Random* rand = NanotimeEntropyProvider_newDefaultRandom(eventBase, logger, eh, alloc); // -------------------- Change Canary Value ---------------------- // Allocator_setCanary(alloc, (uintptr_t)Random_uint64(rand)); struct Allocator* tempAlloc = Allocator_child(alloc); // Not using tempalloc because we're going to keep this pipe around for admin struct PipeServer* clientPipe = PipeServer_named(argv[2], eventBase, eh, logger, alloc); Log_debug(logger, "Getting pre-configuration from client"); struct Message* preConf = InterfaceWaiter_waitForData(&clientPipe->iface.iface, eventBase, tempAlloc, eh); Log_debug(logger, "Finished getting pre-configuration from client"); struct Sockaddr* addr = Sockaddr_clone(Er_assert(AddrIface_popAddr(preConf)), tempAlloc); Dict* config = Except_er(eh, BencMessageReader_read(preConf, tempAlloc)); String* privateKeyHex = Dict_getStringC(config, "privateKey"); Dict* adminConf = Dict_getDictC(config, "admin"); String* pass = Dict_getStringC(adminConf, "pass"); String* bind = Dict_getStringC(adminConf, "bind"); if (!(pass && privateKeyHex && bind)) { if (!pass) { Except_throw(eh, "Expected 'pass'"); } if (!bind) { Except_throw(eh, "Expected 'bind'"); } if (!privateKeyHex) { Except_throw(eh, "Expected 'privateKey'"); } Except_throw(eh, "Expected 'pass', 'privateKey' and 'bind' in configuration."); } Log_keys(logger, "Starting core with admin password [%s]", pass->bytes); uint8_t privateKey[32]; if (privateKeyHex->len != 64 || Hex_decode(privateKey, 32, (uint8_t*) privateKeyHex->bytes, 64) != 32) { Except_throw(eh, "privateKey must be 64 bytes of hex."); } struct Sockaddr_storage bindAddr; if (Sockaddr_parse(bind->bytes, &bindAddr)) { Except_throw(eh, "bind address [%s] unparsable", bind->bytes); } // --------------------- Bind Admin UDP --------------------- // struct UDPAddrIface* udpAdmin = Except_er(eh, UDPAddrIface_new(eventBase, &bindAddr.addr, alloc, logger)); // ---- Setup a muxer so we can get admin from socket or UDP ---- // struct AddrIfaceMuxer* muxer = AddrIfaceMuxer_new(logger, alloc); Iface_plumb(&udpAdmin->generic.iface, AddrIfaceMuxer_registerIface(muxer, alloc)); Iface_plumb(&clientPipe->iface.iface, AddrIfaceMuxer_registerIface(muxer, alloc)); // --------------------- Setup Admin --------------------- // struct Admin* admin = Admin_new(&muxer->iface, logger, eventBase, pass); // --------------------- Setup the Logger --------------------- // Dict* logging = Dict_getDictC(config, "logging"); String* logTo = Dict_getStringC(logging, "logTo"); if (logTo && String_equals(logTo, String_CONST("stdout"))) { // do nothing, continue logging to stdout. } else { struct Log* adminLogger = AdminLog_registerNew(admin, alloc, rand, eventBase); IndirectLog_set(logger, adminLogger); logger = adminLogger; Rffi_setLogger(logger); } // --------------------- Inform client of UDP Addr --------------------- // char* boundAddr = Sockaddr_print(udpAdmin->generic.addr, tempAlloc); Dict adminResponse = Dict_CONST( String_CONST("bind"), String_OBJ(String_CONST(boundAddr)), NULL ); Dict response = Dict_CONST( String_CONST("error"), String_OBJ(String_CONST("none")), Dict_CONST( String_CONST("admin"), Dict_OBJ(&adminResponse), NULL )); // This always times out because the angel doesn't respond. struct Message* clientResponse = Message_new(0, 512, tempAlloc); Er_assert(BencMessageWriter_write(&response, clientResponse)); Er_assert(AddrIface_pushAddr(clientResponse, addr)); Iface_CALL(clientPipe->iface.iface.send, clientResponse, &clientPipe->iface.iface); Allocator_free(tempAlloc); Core_init(alloc, logger, eventBase, privateKey, admin, rand, eh, NULL, false); EventBase_beginLoop(eventBase); return 0; }