X-Git-Url: https://git.librecmc.org/?a=blobdiff_plain;f=src%2Fvpn%2Fgnunet-daemon-vpn.c;h=19824cc589df8d390501003573394d41b9c3a7e8;hb=9a4be70c294da35d740c78dde8f1a8e516eb26ec;hp=23b50c6994a746c4bd79974fcc94ee0a6bb5b98c;hpb=729f0bb2677eb1f22ecff2284e987e99212541a3;p=oweals%2Fgnunet.git diff --git a/src/vpn/gnunet-daemon-vpn.c b/src/vpn/gnunet-daemon-vpn.c index 23b50c699..19824cc58 100644 --- a/src/vpn/gnunet-daemon-vpn.c +++ b/src/vpn/gnunet-daemon-vpn.c @@ -1,6 +1,6 @@ /* This file is part of GNUnet. - (C) 2010 Christian Grothoff (and other contributing authors) + (C) 2010 Christian Grothoff GNUnet is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published @@ -21,12 +21,18 @@ /** * @file vpn/gnunet-daemon-vpn.c * @brief - * @author Christian Grothoff + * @author Philipp Tölke */ #include "platform.h" #include "gnunet_getopt_lib.h" #include "gnunet_program_lib.h" #include "gnunet_os_lib.h" +#include "gnunet-vpn-helper-p.h" +#include "gnunet-vpn-packet.h" +#include "gnunet-vpn-pretty-print.h" +#include "gnunet_common.h" +#include "gnunet_protocols.h" +#include "gnunet_server_lib.h" /* #include "gnunet_template_service.h" */ /** @@ -35,21 +41,95 @@ static int ret; struct vpn_cls { - struct GNUNET_DISK_PipeHandle* helper_in; - struct GNUNET_DISK_PipeHandle* helper_out; + struct GNUNET_DISK_PipeHandle* helper_in; // From the helper + struct GNUNET_DISK_PipeHandle* helper_out; // To the helper + const struct GNUNET_DISK_FileHandle* fh_from_helper; + + struct GNUNET_SERVER_MessageStreamTokenizer* mst; + + struct GNUNET_SCHEDULER_Handle *sched; pid_t helper_pid; }; static void cleanup(void* cls, const struct GNUNET_SCHEDULER_TaskContext* tskctx) { struct vpn_cls* mycls = (struct vpn_cls*) cls; - if (tskctx->reason == GNUNET_SCHEDULER_REASON_SHUTDOWN) { + if (tskctx->reason & GNUNET_SCHEDULER_REASON_SHUTDOWN) { PLIBC_KILL(mycls->helper_pid, SIGTERM); GNUNET_OS_process_wait(mycls->helper_pid); } } +static void helper_read(void* cls, const struct GNUNET_SCHEDULER_TaskContext* tsdkctx); + +static void start_helper_and_schedule(struct vpn_cls* mycls) { + mycls->helper_in = GNUNET_DISK_pipe(1); + mycls->helper_out = GNUNET_DISK_pipe(1); + + mycls->helper_pid = GNUNET_OS_start_process(mycls->helper_in, mycls->helper_out, "gnunet-vpn-helper", "gnunet-vpn-helper", NULL); + + mycls->fh_from_helper = GNUNET_DISK_pipe_handle (mycls->helper_out, GNUNET_DISK_PIPE_END_READ); + + GNUNET_DISK_pipe_close_end(mycls->helper_out, GNUNET_DISK_PIPE_END_WRITE); + GNUNET_DISK_pipe_close_end(mycls->helper_in, GNUNET_DISK_PIPE_END_READ); + + GNUNET_SCHEDULER_add_read_file (mycls->sched, GNUNET_TIME_UNIT_FOREVER_REL, mycls->fh_from_helper, &helper_read, mycls); +} + + +static void restart_helper(void* cls, const struct GNUNET_SCHEDULER_TaskContext* tskctx) { + struct vpn_cls* mycls = (struct vpn_cls*) cls; + + // Kill the helper + PLIBC_KILL(mycls->helper_pid, SIGTERM); + GNUNET_OS_process_wait(mycls->helper_pid); + + // Restart the helper + start_helper_and_schedule(mycls); + +} + static void helper_read(void* cls, const struct GNUNET_SCHEDULER_TaskContext* tsdkctx) { + struct vpn_cls* mycls = (struct vpn_cls*) cls; + char buf[65535]; + + if (tsdkctx->reason & GNUNET_SCHEDULER_REASON_SHUTDOWN) + return; + + int t = GNUNET_DISK_file_read(mycls->fh_from_helper, &buf, 65535); + if (t<=0) { + fprintf(stderr, "Read error for header: %m\n"); + GNUNET_SCHEDULER_add_now(mycls->sched, restart_helper, cls); + return; + } + + /* FIXME */ GNUNET_SERVER_mst_receive(mycls->mst, NULL, buf, t, 0, 0); + + GNUNET_SCHEDULER_add_read_file (mycls->sched, GNUNET_TIME_UNIT_FOREVER_REL, mycls->fh_from_helper, &helper_read, mycls); +} + +static void message_token(void *cls, void *client, const struct GNUNET_MessageHeader *message) { + if (ntohs(message->type) != GNUNET_MESSAGE_TYPE_VPN_HELPER) return; + + struct ip6_pkt *pkt6 = (struct ip6_pkt*) message; + struct ip6_tcp *pkt6_tcp; + struct ip6_udp *pkt6_udp; + + pkt_printf(pkt6); + switch(pkt6->ip6_hdr.nxthdr) { + case 0x06: + pkt6_tcp = (struct ip6_tcp*)pkt6; + pkt_printf_ip6tcp(pkt6_tcp); + break; + case 0x11: + pkt6_udp = (struct ip6_udp*)pkt6; + pkt_printf_ip6udp(pkt6_udp); + if (ntohs(pkt6_udp->udp_hdr.dpt) == 53) { + pkt_printf_ip6dns((struct ip6_udp_dns*)pkt6_udp); + } + break; + } + } /** @@ -63,23 +143,20 @@ static void helper_read(void* cls, const struct GNUNET_SCHEDULER_TaskContext* ts */ static void run (void *cls, - struct GNUNET_SCHEDULER_Handle *sched, - char *const *args, - const char *cfgfile, - const struct GNUNET_CONFIGURATION_Handle *cfg) { + struct GNUNET_SCHEDULER_Handle *sched, + char *const *args, + const char *cfgfile, + const struct GNUNET_CONFIGURATION_Handle *cfg) { struct vpn_cls* mycls = (struct vpn_cls*) cls; - GNUNET_SCHEDULER_add_delayed(sched, GNUNET_TIME_UNIT_FOREVER_REL, &cleanup, cls); + mycls->sched = sched; - mycls->helper_in = GNUNET_DISK_pipe(1); - mycls->helper_out = GNUNET_DISK_pipe(1); + mycls->mst = GNUNET_SERVER_mst_create(&message_token, mycls); - mycls->helper_pid = GNUNET_OS_start_process(mycls->helper_in, mycls->helper_out, "gnunet-vpn-helper", "gnunet-vpn-helper", NULL); + GNUNET_SCHEDULER_add_delayed(sched, GNUNET_TIME_UNIT_FOREVER_REL, &cleanup, cls); - const struct GNUNET_DISK_FileHandle* fh = GNUNET_DISK_pipe_handle (mycls->helper_out, GNUNET_DISK_PIPE_END_READ); - - GNUNET_SCHEDULER_add_read_file (sched, GNUNET_TIME_UNIT_FOREVER_REL, fh, &helper_read, mycls); + start_helper_and_schedule(mycls); } @@ -105,6 +182,8 @@ main (int argc, char *const *argv) "gnunet-daemon-vpn", gettext_noop ("help text"), options, &run, cls)) ? ret : 1; + + free(cls); // Make clang happy } /* end of gnunet-daemon-vpn.c */