123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577 |
- /* 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 <https://www.gnu.org/licenses/>.
- */
- #include "subnode/SubnodePathfinder.h"
- #include "subnode/AddrSet.h"
- #include "subnode/MsgCore.h"
- #include "subnode/SupernodeHunter.h"
- #include "subnode/GetPeersResponder.h"
- #include "subnode/PingResponder.h"
- #include "subnode/BoilerplateResponder.h"
- #include "subnode/ReachabilityCollector.h"
- #include "crypto/AddressCalc.h"
- #include "dht/Address.h"
- #include "wire/DataHeader.h"
- #include "wire/RouteHeader.h"
- #include "dht/dhtcore/ReplySerializer.h"
- #include "util/AddrTools.h"
- #include "util/events/Timeout.h"
- #include "net/SwitchPinger.h"
- #include "switch/LabelSplicer.h"
- #include "wire/Error.h"
- #include "wire/PFChan.h"
- #include "wire/DataHeader.h"
- #include "util/CString.h"
- #include "subnode/ReachabilityAnnouncer.h"
- struct Query {
- struct Address target;
- uint8_t routeFrom[16];
- uint8_t routeTo[16];
- };
- #define Map_NAME OfPromiseByQuery
- #define Map_KEY_TYPE struct Query
- #define Map_VALUE_TYPE struct MsgCore_Promise*
- #define Map_ENABLE_HANDLES
- #include "util/Map.h"
- struct SubnodePathfinder_pvt
- {
- struct SubnodePathfinder pub;
- struct Iface msgCoreIf;
- struct Allocator* alloc;
- struct Log* log;
- struct EventBase* base;
- struct Random* rand;
- #define SubnodePathfinder_pvt_state_INITIALIZING 0
- #define SubnodePathfinder_pvt_state_RUNNING 1
- int state;
- struct Address* myAddress;
- struct AddrSet* myPeers;
- struct MsgCore* msgCore;
- struct Admin* admin;
- struct BoilerplateResponder* br;
- struct ReachabilityAnnouncer* ra;
- struct Map_OfPromiseByQuery queryMap;
- struct SwitchPinger* sp;
- struct Iface switchPingerIf;
- struct EncodingScheme* myScheme;
- uint8_t* privateKey;
- String* encodingSchemeStr;
- Identity
- };
- static void nodeForAddress(struct PFChan_Node* nodeOut, struct Address* addr, uint32_t metric)
- {
- Bits_memset(nodeOut, 0, PFChan_Node_SIZE);
- nodeOut->version_be = Endian_hostToBigEndian32(addr->protocolVersion);
- nodeOut->metric_be = Endian_hostToBigEndian32(metric);
- nodeOut->path_be = Endian_hostToBigEndian64(addr->path);
- Bits_memcpy(nodeOut->publicKey, addr->key, 32);
- Bits_memcpy(nodeOut->ip6, addr->ip6.bytes, 16);
- }
- static Iface_DEFUN sendNode(struct Message* msg,
- struct Address* addr,
- uint32_t metric,
- enum PFChan_Pathfinder msgType,
- struct SubnodePathfinder_pvt* pf)
- {
- Message_reset(msg);
- Message_shift(msg, PFChan_Node_SIZE, NULL);
- nodeForAddress((struct PFChan_Node*) msg->bytes, addr, metric);
- if (addr->path == UINT64_MAX) {
- ((struct PFChan_Node*) msg->bytes)->path_be = 0;
- }
- Message_push32(msg, msgType, NULL);
- return Iface_next(&pf->pub.eventIf, msg);
- }
- static Iface_DEFUN connected(struct SubnodePathfinder_pvt* pf, struct Message* msg)
- {
- Log_debug(pf->log, "INIT");
- pf->state = SubnodePathfinder_pvt_state_RUNNING;
- return NULL;
- }
- static void addressForNode(struct Address* addrOut, struct Message* msg)
- {
- struct PFChan_Node node;
- Message_pop(msg, &node, PFChan_Node_SIZE, NULL);
- Assert_true(!msg->length);
- addrOut->protocolVersion = Endian_bigEndianToHost32(node.version_be);
- addrOut->path = Endian_bigEndianToHost64(node.path_be);
- Bits_memcpy(addrOut->key, node.publicKey, 32);
- Bits_memcpy(addrOut->ip6.bytes, node.ip6, 16);
- }
- static Iface_DEFUN switchErr(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- struct PFChan_Core_SwitchErr switchErr;
- Message_pop(msg, &switchErr, PFChan_Core_SwitchErr_MIN_SIZE, NULL);
- uint64_t path = Endian_bigEndianToHost64(switchErr.sh.label_be);
- if (pf->pub.snh->snodeAddr.path &&
- pf->pub.snh->snodeAddr.path != path &&
- LabelSplicer_routesThrough(pf->pub.snh->snodeAddr.path, path)) {
- uint8_t pathStr[20];
- AddrTools_printPath(pathStr, path);
- int err = Endian_bigEndianToHost32(switchErr.ctrlErr.errorType_be);
- Log_debug(pf->log, "switch err from active snode [%s] type [%s][%d]",
- pathStr, Error_strerror(err), err);
- pf->pub.snh->snodeIsReachable = false;
- if (pf->pub.snh->onSnodeUnreachable) {
- pf->pub.snh->onSnodeUnreachable(pf->pub.snh, 0, 0);
- }
- }
- return NULL;
- }
- struct SnodeQuery {
- struct SubnodePathfinder_pvt* pf;
- uint32_t mapHandle;
- Identity
- };
- static void getRouteReply(Dict* msg, struct Address* src, struct MsgCore_Promise* prom)
- {
- struct SnodeQuery* snq = Identity_check((struct SnodeQuery*) prom->userData);
- struct SubnodePathfinder_pvt* pf = Identity_check(snq->pf);
- int index = Map_OfPromiseByQuery_indexForHandle(snq->mapHandle, &pf->queryMap);
- Assert_true(index > -1);
- Map_OfPromiseByQuery_remove(index, &pf->queryMap);
- if (!src) {
- Log_debug(pf->log, "GetRoute timeout");
- return;
- }
- Log_debug(pf->log, "Search reply!");
- struct Address_List* al = ReplySerializer_parse(src, msg, pf->log, false, prom->alloc);
- if (!al || al->length == 0) { return; }
- Log_debug(pf->log, "reply with[%s]", Address_toString(&al->elems[0], prom->alloc)->bytes);
- if (al->elems[0].protocolVersion < 20) {
- Log_debug(pf->log, "not sending [%s] because version is old",
- Address_toString(&al->elems[0], prom->alloc)->bytes);
- return;
- }
- //NodeCache_discoverNode(pf->nc, &al->elems[0]);
- struct Message* msgToCore = Message_new(0, 512, prom->alloc);
- Iface_CALL(sendNode, msgToCore, &al->elems[0], 0xfff00033, PFChan_Pathfinder_NODE, pf);
- }
- static Iface_DEFUN searchReq(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- uint8_t addr[16];
- Message_pop(msg, addr, 16, NULL);
- Message_pop32(msg, NULL);
- uint32_t version = Message_pop32(msg, NULL);
- if (version && version < 20) { return NULL; }
- Assert_true(!msg->length);
- uint8_t printedAddr[40];
- AddrTools_printIp(printedAddr, addr);
- Log_debug(pf->log, "Search req [%s]", printedAddr);
- for (int i = 0; i < pf->myPeers->length; ++i) {
- struct Address* myPeer = AddrSet_get(pf->myPeers, i);
- if (!Bits_memcmp(myPeer->ip6.bytes, addr, 16)) {
- return sendNode(msg, myPeer, 0xfff00000, PFChan_Pathfinder_NODE, pf);
- }
- }
- if (!pf->pub.snh || !pf->pub.snh->snodeAddr.path) { return NULL; }
- if (!Bits_memcmp(pf->pub.snh->snodeAddr.ip6.bytes, addr, 16)) {
- return sendNode(msg, &pf->pub.snh->snodeAddr, 0xfff00000, PFChan_Pathfinder_NODE, pf);
- }
- struct Query q = { .routeFrom = { 0 } };
- Bits_memcpy(&q.target, &pf->pub.snh->snodeAddr, sizeof(struct Address));
- Bits_memcpy(q.routeFrom, pf->myAddress->ip6.bytes, 16);
- Bits_memcpy(q.routeFrom, addr, 16);
- if (Map_OfPromiseByQuery_indexForKey(&q, &pf->queryMap) > -1) {
- Log_debug(pf->log, "Skipping snode query because one is outstanding");
- return NULL;
- }
- struct MsgCore_Promise* qp = MsgCore_createQuery(pf->msgCore, 0, pf->alloc);
- struct SnodeQuery* snq = Allocator_calloc(qp->alloc, sizeof(struct SnodeQuery), 1);
- Identity_set(snq);
- snq->pf = pf;
- Dict* dict = qp->msg = Dict_new(qp->alloc);
- qp->cb = getRouteReply;
- qp->userData = snq;
- Assert_true(AddressCalc_validAddress(pf->pub.snh->snodeAddr.ip6.bytes));
- qp->target = &pf->pub.snh->snodeAddr;
- Log_debug(pf->log, "Sending getRoute to snode %s",
- Address_toString(qp->target, qp->alloc)->bytes);
- Dict_putStringCC(dict, "sq", "gr", qp->alloc);
- String* src = String_newBinary(pf->myAddress->ip6.bytes, 16, qp->alloc);
- Dict_putStringC(dict, "src", src, qp->alloc);
- String* target = String_newBinary(addr, 16, qp->alloc);
- Dict_putStringC(dict, "tar", target, qp->alloc);
- int index = Map_OfPromiseByQuery_put(&q, &qp, &pf->queryMap);
- snq->mapHandle = pf->queryMap.handles[index];
- return NULL;
- }
- static void rcChange(struct ReachabilityCollector* rc,
- uint8_t nodeIpv6[16],
- uint64_t pathThemToUs,
- uint64_t pathUsToThem,
- uint32_t mtu,
- uint16_t drops,
- uint16_t latency,
- uint16_t penalty)
- {
- struct SubnodePathfinder_pvt* pf = Identity_check((struct SubnodePathfinder_pvt*) rc->userData);
- ReachabilityAnnouncer_updatePeer(
- pf->ra, nodeIpv6, pathThemToUs, pathUsToThem, mtu, drops, latency, penalty);
- }
- static Iface_DEFUN peer(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- struct Address addr;
- addressForNode(&addr, msg);
- String* str = Address_toString(&addr, msg->alloc);
- Log_debug(pf->log, "Peer [%s]", str->bytes);
- int index = AddrSet_indexOf(pf->myPeers, &addr);
- if (index > -1) {
- struct Address* myPeer = AddrSet_get(pf->myPeers, index);
- if (myPeer->path == addr.path && myPeer->protocolVersion == addr.protocolVersion) {
- return NULL;
- }
- AddrSet_remove(pf->myPeers, myPeer);
- }
- AddrSet_add(pf->myPeers, &addr);
- //NodeCache_discoverNode(pf->nc, &addr);
- ReachabilityCollector_change(pf->pub.rc, &addr);
- return sendNode(msg, &addr, 0xfff00000, PFChan_Pathfinder_NODE, pf);
- }
- static Iface_DEFUN peerGone(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- struct Address addr;
- addressForNode(&addr, msg);
- for (int i = pf->myPeers->length - 1; i >= 0; i--) {
- struct Address* myPeer = AddrSet_get(pf->myPeers, i);
- if (myPeer->path == addr.path) {
- String* str = Address_toString(myPeer, msg->alloc);
- AddrSet_remove(pf->myPeers, myPeer);
- Log_debug(pf->log, "Peer gone [%s]", str->bytes);
- }
- }
- //NodeCache_forgetNode(pf->nc, &addr);
- struct Address zaddr;
- Bits_memcpy(&zaddr, &addr, Address_SIZE);
- zaddr.path = 0;
- ReachabilityCollector_change(pf->pub.rc, &zaddr);
- // We notify about the node but with max metric so it will be removed soon.
- return sendNode(msg, &addr, 0xffffffff, PFChan_Pathfinder_NODE, pf);
- }
- static Iface_DEFUN session(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- struct Address addr;
- addressForNode(&addr, msg);
- String* str = Address_toString(&addr, msg->alloc);
- Log_debug(pf->log, "Session [%s]", str->bytes);
- //if (addr.protocolVersion) { NodeCache_discoverNode(pf->nc, &addr); }
- return NULL;
- }
- static Iface_DEFUN sessionEnded(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- struct Address addr;
- addressForNode(&addr, msg);
- String* str = Address_toString(&addr, msg->alloc);
- Log_debug(pf->log, "Session ended [%s]", str->bytes);
- //NodeCache_forgetNode(pf->nc, &addr);
- return NULL;
- }
- static Iface_DEFUN discoveredPath(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- //struct Address addr;
- //addressForNode(&addr, msg);
- //Log_debug(pf->log, "discoveredPath(%s)", Address_toString(&addr, msg->alloc)->bytes);
- //if (addr.protocolVersion) { NodeCache_discoverNode(pf->nc, &addr); }
- return NULL;
- }
- static Iface_DEFUN handlePing(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- //Log_debug(pf->log, "Received ping");
- Message_push32(msg, PFChan_Pathfinder_PONG, NULL);
- return Iface_next(&pf->pub.eventIf, msg);
- }
- static Iface_DEFUN handlePong(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- //Log_debug(pf->log, "Received pong");
- return NULL;
- }
- static Iface_DEFUN ctrlMsgFromSwitchPinger(struct Message* msg, struct Iface* iface)
- {
- struct SubnodePathfinder_pvt* pf =
- Identity_containerOf(iface, struct SubnodePathfinder_pvt, switchPingerIf);
- Message_push32(msg, PFChan_Pathfinder_CTRL_SENDMSG, NULL);
- return Iface_next(&pf->pub.eventIf, msg);
- }
- static Iface_DEFUN ctrlMsg(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- return Iface_next(&pf->switchPingerIf, msg);
- }
- struct UnsetupSessionPing {
- struct SubnodePathfinder_pvt* pf;
- uint32_t mapHandle;
- Identity
- };
- static void unsetupSessionPingReply(Dict* msg, struct Address* src, struct MsgCore_Promise* prom)
- {
- struct UnsetupSessionPing* usp =
- Identity_check((struct UnsetupSessionPing*) prom->userData);
- struct SubnodePathfinder_pvt* pf = Identity_check(usp->pf);
- int index = Map_OfPromiseByQuery_indexForHandle(usp->mapHandle, &pf->queryMap);
- Assert_true(index > -1);
- Map_OfPromiseByQuery_remove(index, &pf->queryMap);
- if (!src) {
- //Log_debug(pf->log, "Ping timeout");
- return;
- }
- //Log_debug(pf->log, "\n\n\n\nPING reply from [%s]!\n\n\n\n",
- // Address_toString(src, prom->alloc)->bytes);
- struct Message* msgToCore = Message_new(0, 512, prom->alloc);
- Iface_CALL(sendNode, msgToCore, src, 0xfffffff0, PFChan_Pathfinder_NODE, pf);
- }
- static Iface_DEFUN unsetupSession(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- struct PFChan_Node node;
- Message_pop(msg, &node, PFChan_Node_SIZE, NULL);
- Assert_true(!msg->length);
- struct Query q = { .routeFrom = { 0 } };
- struct Address* addr = &q.target;
- Bits_memcpy(addr->ip6.bytes, node.ip6, 16);
- Bits_memcpy(addr->key, node.publicKey, 32);
- addr->protocolVersion = Endian_bigEndianToHost32(node.version_be);
- addr->path = Endian_bigEndianToHost64(node.path_be);
- if (Map_OfPromiseByQuery_indexForKey(&q, &pf->queryMap) > -1) {
- Log_debug(pf->log, "Skipping ping because one is already outstanding");
- return NULL;
- }
- // We have a path to the node but the session is not setup, lets ping them...
- struct MsgCore_Promise* qp = MsgCore_createQuery(pf->msgCore, 0, pf->alloc);
- struct UnsetupSessionPing* usp =
- Allocator_calloc(qp->alloc, sizeof(struct UnsetupSessionPing), 1);
- Identity_set(usp);
- usp->pf = pf;
- Dict* dict = qp->msg = Dict_new(qp->alloc);
- qp->cb = unsetupSessionPingReply;
- qp->userData = usp;
- Assert_true(AddressCalc_validAddress(addr->ip6.bytes));
- Assert_true(addr->path);
- qp->target = Address_clone(addr, qp->alloc);
- //Log_debug(pf->log, "unsetupSession sending ping to [%s]",
- // Address_toString(qp->target, qp->alloc)->bytes);
- Dict_putStringCC(dict, "q", "pn", qp->alloc);
- BoilerplateResponder_addBoilerplate(pf->br, dict, addr, qp->alloc);
- int index = Map_OfPromiseByQuery_put(&q, &qp, &pf->queryMap);
- usp->mapHandle = pf->queryMap.handles[index];
- return NULL;
- }
- static Iface_DEFUN incomingMsg(struct Message* msg, struct SubnodePathfinder_pvt* pf)
- {
- return Iface_next(&pf->msgCoreIf, msg);
- }
- static Iface_DEFUN incomingFromMsgCore(struct Message* msg, struct Iface* iface)
- {
- struct SubnodePathfinder_pvt* pf =
- Identity_containerOf(iface, struct SubnodePathfinder_pvt, msgCoreIf);
- Assert_true(msg->length >= (RouteHeader_SIZE + DataHeader_SIZE));
- struct RouteHeader* rh = (struct RouteHeader*) msg->bytes;
- struct DataHeader* dh = (struct DataHeader*) &rh[1];
- Assert_true(DataHeader_getContentType(dh) == ContentType_CJDHT);
- Assert_true(!Bits_isZero(rh->publicKey, 32));
- Assert_true(rh->version_be);
- Assert_true(rh->sh.label_be);
- Message_push32(msg, PFChan_Pathfinder_SENDMSG, NULL);
- return Iface_next(&pf->pub.eventIf, msg);
- }
- static Iface_DEFUN incomingFromEventIf(struct Message* msg, struct Iface* eventIf)
- {
- struct SubnodePathfinder_pvt* pf =
- Identity_containerOf(eventIf, struct SubnodePathfinder_pvt, pub.eventIf);
- enum PFChan_Core ev = Message_pop32(msg, NULL);
- if (SubnodePathfinder_pvt_state_INITIALIZING == pf->state) {
- Assert_true(ev == PFChan_Core_CONNECT);
- return connected(pf, msg);
- }
- switch (ev) {
- case PFChan_Core_SWITCH_ERR: return switchErr(msg, pf);
- case PFChan_Core_SEARCH_REQ: return searchReq(msg, pf);
- case PFChan_Core_PEER: return peer(msg, pf);
- case PFChan_Core_PEER_GONE: return peerGone(msg, pf);
- case PFChan_Core_SESSION: return session(msg, pf);
- case PFChan_Core_SESSION_ENDED: return sessionEnded(msg, pf);
- case PFChan_Core_DISCOVERED_PATH: return discoveredPath(msg, pf);
- case PFChan_Core_MSG: return incomingMsg(msg, pf);
- case PFChan_Core_PING: return handlePing(msg, pf);
- case PFChan_Core_PONG: return handlePong(msg, pf);
- case PFChan_Core_CTRL_MSG: return ctrlMsg(msg, pf);
- case PFChan_Core_UNSETUP_SESSION: return unsetupSession(msg, pf);
- default:;
- }
- Assert_failure("unexpected event [%d]", ev);
- }
- static void sendEvent(struct SubnodePathfinder_pvt* pf,
- enum PFChan_Pathfinder ev,
- void* data,
- int size)
- {
- struct Allocator* alloc = Allocator_child(pf->alloc);
- struct Message* msg = Message_new(0, 512+size, alloc);
- Message_push(msg, data, size, NULL);
- Message_push32(msg, ev, NULL);
- Iface_send(&pf->pub.eventIf, msg);
- Allocator_free(alloc);
- }
- void SubnodePathfinder_start(struct SubnodePathfinder* sp)
- {
- struct SubnodePathfinder_pvt* pf = Identity_check((struct SubnodePathfinder_pvt*) sp);
- pf->msgCore = MsgCore_new(pf->base, pf->rand, pf->alloc, pf->log, pf->myScheme);
- Iface_plumb(&pf->msgCoreIf, &pf->msgCore->interRouterIf);
- PingResponder_new(pf->alloc, pf->log, pf->msgCore, pf->br);
- GetPeersResponder_new(
- pf->alloc, pf->log, pf->myPeers, pf->myAddress, pf->msgCore, pf->br, pf->myScheme);
- pf->pub.snh = SupernodeHunter_new(
- pf->alloc, pf->log, pf->base, pf->sp, pf->myPeers, pf->msgCore, pf->myAddress);
- pf->ra = ReachabilityAnnouncer_new(
- pf->alloc, pf->log, pf->base, pf->rand, pf->msgCore, pf->pub.snh, pf->privateKey,
- pf->myScheme);
- pf->pub.rc = ReachabilityCollector_new(
- pf->alloc, pf->msgCore, pf->log, pf->base, pf->br, pf->myAddress);
- pf->pub.rc->userData = pf;
- pf->pub.rc->onChange = rcChange;
- struct PFChan_Pathfinder_Connect conn = {
- .superiority_be = Endian_hostToBigEndian32(1),
- .version_be = Endian_hostToBigEndian32(Version_CURRENT_PROTOCOL)
- };
- CString_strncpy(conn.userAgent, "Cjdns subnode pathfinder", 64);
- sendEvent(pf, PFChan_Pathfinder_CONNECT, &conn, PFChan_Pathfinder_Connect_SIZE);
- }
- static void sendCurrentSupernode(void* vsp)
- {
- struct SubnodePathfinder_pvt* pf = Identity_check((struct SubnodePathfinder_pvt*) vsp);
- struct Allocator* alloc = Allocator_child(pf->alloc);
- struct Message* msgToCore = Message_new(0, 512, alloc);
- Iface_CALL(sendNode, msgToCore, &pf->pub.snh->snodeAddr, 0, PFChan_Pathfinder_SNODE, pf);
- Allocator_free(alloc);
- }
- struct SubnodePathfinder* SubnodePathfinder_new(struct Allocator* allocator,
- struct Log* log,
- struct EventBase* base,
- struct Random* rand,
- struct Address* myAddress,
- uint8_t* privateKey,
- struct EncodingScheme* myScheme)
- {
- struct Allocator* alloc = Allocator_child(allocator);
- struct SubnodePathfinder_pvt* pf =
- Allocator_calloc(alloc, sizeof(struct SubnodePathfinder_pvt), 1);
- Identity_set(pf);
- pf->alloc = alloc;
- pf->log = log;
- pf->base = base;
- pf->rand = rand;
- pf->myAddress = myAddress;
- pf->myPeers = AddrSet_new(alloc);
- pf->pub.eventIf.send = incomingFromEventIf;
- pf->msgCoreIf.send = incomingFromMsgCore;
- pf->privateKey = privateKey;
- pf->queryMap.allocator = Allocator_child(alloc);
- pf->myScheme = myScheme;
- pf->br = BoilerplateResponder_new(myScheme, alloc);
- pf->sp = SwitchPinger_new(base, rand, log, myAddress, alloc);
- pf->switchPingerIf.send = ctrlMsgFromSwitchPinger;
- Iface_plumb(&pf->switchPingerIf, &pf->sp->controlHandlerIf);
- Timeout_setInterval(sendCurrentSupernode, pf, 3000, base, alloc);
- return &pf->pub;
- }
|