2 This file is part of GNUnet.
3 (C) 2010 Christian Grothoff
5 GNUnet is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; either version 3, or (at your
8 option) any later version.
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with GNUnet; see the file COPYING. If not, write to the
17 Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 Boston, MA 02111-1307, USA.
22 * @file vpn/gnunet-daemon-vpn-dns.c
24 * @author Philipp Toelke
27 #include <gnunet_common.h>
28 #include <gnunet_client_lib.h>
29 #include <gnunet_os_lib.h>
30 #include <gnunet_mesh_service.h>
31 #include <gnunet_protocols.h>
32 #include <gnunet_server_lib.h>
33 #include <gnunet_container_lib.h>
34 #include <block_dns.h>
36 #include "gnunet-daemon-vpn-dns.h"
37 #include "gnunet-daemon-vpn.h"
38 #include "gnunet-daemon-vpn-helper.h"
39 #include "gnunet-service-dns-p.h"
40 #include "gnunet-vpn-packet.h"
42 struct query_packet_list *head;
43 struct query_packet_list *tail;
44 struct GNUNET_CLIENT_Connection *dns_connection;
45 unsigned char restart_hijack;
46 struct answer_packet_list *answer_proc_head;
47 struct answer_packet_list *answer_proc_tail;
49 struct GNUNET_CLIENT_TransmitHandle *dns_transmit_handle;
52 * Callback called by notify_transmit_ready; sends dns-queries or rehijack-messages
57 send_query (void *cls __attribute__ ((unused)), size_t size, void *buf)
61 dns_transmit_handle = NULL;
64 * Send the rehijack-message
66 if (restart_hijack == 1)
70 * The message is just a header
72 GNUNET_assert (sizeof (struct GNUNET_MessageHeader) <= size);
73 struct GNUNET_MessageHeader *hdr = buf;
75 len = sizeof (struct GNUNET_MessageHeader);
76 hdr->size = htons (len);
77 hdr->type = htons (GNUNET_MESSAGE_TYPE_REHIJACK);
79 else if (head != NULL)
81 struct query_packet_list *query = head;
83 len = ntohs (query->pkt.hdr.size);
85 GNUNET_assert (len <= size);
87 memcpy (buf, &query->pkt.hdr, len);
89 GNUNET_CONTAINER_DLL_remove (head, tail, query);
100 * Check whether more data is to be sent
104 dns_transmit_handle =
105 GNUNET_CLIENT_notify_transmit_ready (dns_connection,
106 ntohs (head->pkt.hdr.size),
107 GNUNET_TIME_UNIT_FOREVER_REL,
108 GNUNET_YES, &send_query, NULL);
110 else if (restart_hijack == 1)
112 dns_transmit_handle =
113 GNUNET_CLIENT_notify_transmit_ready (dns_connection,
115 GNUNET_MessageHeader),
116 GNUNET_TIME_UNIT_FOREVER_REL,
117 GNUNET_YES, &send_query, NULL);
127 * Connect to the service-dns
130 connect_to_service_dns (void *cls
131 __attribute__ ((unused)),
132 const struct GNUNET_SCHEDULER_TaskContext *tc)
134 conn_task = GNUNET_SCHEDULER_NO_TASK;
135 if (0 != (tc->reason & GNUNET_SCHEDULER_REASON_SHUTDOWN))
137 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Connecting to service-dns\n");
138 GNUNET_assert (dns_connection == NULL);
139 dns_connection = GNUNET_CLIENT_connect ("dns", cfg);
140 /* This would most likely be a misconfiguration */
141 GNUNET_assert (NULL != dns_connection);
142 GNUNET_CLIENT_receive (dns_connection, &dns_answer_handler, NULL,
143 GNUNET_TIME_UNIT_FOREVER_REL);
145 /* We might not yet be connected. Yay, mps. */
146 if (NULL == dns_connection)
149 /* If a packet is already in the list, schedule to send it */
150 if (dns_transmit_handle == NULL && head != NULL)
151 dns_transmit_handle =
152 GNUNET_CLIENT_notify_transmit_ready (dns_connection,
153 ntohs (head->pkt.hdr.size),
154 GNUNET_TIME_UNIT_FOREVER_REL,
155 GNUNET_YES, &send_query, NULL);
156 else if (dns_transmit_handle == NULL && restart_hijack == 1)
158 dns_transmit_handle =
159 GNUNET_CLIENT_notify_transmit_ready (dns_connection,
161 GNUNET_MessageHeader),
162 GNUNET_TIME_UNIT_FOREVER_REL,
163 GNUNET_YES, &send_query, NULL);
168 * This receives packets from the service-dns and schedules process_answer to
172 dns_answer_handler (void *cls
173 __attribute__ ((unused)),
174 const struct GNUNET_MessageHeader *msg)
176 /* the service disconnected, reconnect after short wait */
179 if (dns_transmit_handle != NULL)
180 GNUNET_CLIENT_notify_transmit_ready_cancel (dns_transmit_handle);
181 dns_transmit_handle = NULL;
182 GNUNET_CLIENT_disconnect (dns_connection, GNUNET_NO);
183 dns_connection = NULL;
185 GNUNET_SCHEDULER_add_delayed (GNUNET_TIME_UNIT_SECONDS,
186 &connect_to_service_dns, NULL);
190 /* the service did something strange, reconnect immediately */
191 if (msg->type != htons (GNUNET_MESSAGE_TYPE_VPN_DNS_LOCAL_RESPONSE_DNS))
194 GNUNET_CLIENT_disconnect (dns_connection, GNUNET_NO);
195 dns_connection = NULL;
196 conn_task = GNUNET_SCHEDULER_add_now (&connect_to_service_dns, NULL);
199 void *pkt = GNUNET_malloc (ntohs (msg->size));
201 memcpy (pkt, msg, ntohs (msg->size));
203 GNUNET_SCHEDULER_add_now (process_answer, pkt);
204 GNUNET_CLIENT_receive (dns_connection, &dns_answer_handler, NULL,
205 GNUNET_TIME_UNIT_FOREVER_REL);