GNUnet 0.26.2-114-g7c6b613e3
 
Loading...
Searching...
No Matches
gnunet-service-dht.c
Go to the documentation of this file.
1/*
2 This file is part of GNUnet.
3 Copyright (C) 2009-2011, 2016, 2026 GNUnet e.V.
4
5 GNUnet is free software: you can redistribute it and/or modify it
6 under the terms of the GNU Affero General Public License as published
7 by the Free Software Foundation, either version 3 of the License,
8 or (at your option) any later version.
9
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 Affero General Public License for more details.
14
15 You should have received a copy of the GNU Affero General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
17
18 SPDX-License-Identifier: AGPL3.0-or-later
19 */
20
27#include "platform.h"
28#include "gnunet_common.h"
29#include "gnunet_block_lib.h"
30#include "gnunet_util_lib.h"
32#include "gnunet_pils_service.h"
34#include "gnunet-service-dht.h"
38#include "plugin_dhtu_ip.h"
39#include "plugin_dhtu_gnunet.h"
40#include "dht_helper.h"
41
46#define HELLO_FREQUENCY GNUNET_TIME_UNIT_HOURS
47
48
85
86
91{
95 struct MyAddress *next;
96
101
106
110 char *url;
111
116};
117
121struct PilsRequest
122{
126 struct PilsRequest *prev;
127
131 struct PilsRequest *next;
132
137};
138
143
148
149
154
158static struct GDS_Underlay *u_head;
159
163static struct GDS_Underlay *u_tail;
164
168static struct MyAddress *a_head;
169
173static struct MyAddress *a_tail;
174
180
181
191static void
194 double logestimate,
195 double std_dev)
196{
197 struct GDS_Underlay *u = cls;
198 double sum = 0.0;
199
201 "# Network size estimates received",
202 1,
203 GNUNET_NO);
204 /* do not allow estimates < 0.5 */
205 u->network_size_estimate = pow (2.0,
206 GNUNET_MAX (0.5,
207 logestimate));
208 for (struct GDS_Underlay *p = u_head; NULL != p; p = p->next)
209 sum += p->network_size_estimate;
210 if (sum <= 2.0)
212 else
213 log_of_network_size_estimate = log2 (sum);
214}
215
216
222double
224{
226}
227
228
230
231
241static void
242u_address_add (void *cls,
243 const char *address,
245 void **ctx)
246{
247 struct GDS_Underlay *u = cls;
248 struct MyAddress *a;
249
251 "Underlay adds address %s for this peer\n",
252 address);
253 a = GNUNET_new (struct MyAddress);
254 a->source = source;
256 a->u = u;
258 a_tail,
259 a);
260 *ctx = a;
261
262}
263
264
270static void
272{
273 struct MyAddress *a = ctx;
274
276 "Underlay deletes address %s for this peer\n",
277 a->url);
279 a_tail,
280 a);
281 GNUNET_free (a->url);
282 GNUNET_free (a);
283}
284
285
286void
288 const char *address)
289{
290 for (struct GDS_Underlay *u = u_head;
291 NULL != u;
292 u = u->next)
293 u->dhtu->try_connect (u->dhtu->cls,
294 pid,
295 address);
296}
297
298
299void
301 struct GNUNET_DHTU_Target *target,
302 const void *msg,
303 size_t msg_size,
305 void *finished_cb_cls)
306{
307 u->dhtu->send (u->dhtu->cls,
308 target,
309 msg,
310 msg_size,
312 finished_cb_cls);
313}
314
315
316void
319{
320 u->dhtu->drop (ph);
321}
322
323
327{
328 return u->dhtu->hold (u->dhtu->cls,
329 target);
330}
331
332
338static void
339shutdown_task (void *cls)
340{
341 struct GDS_Underlay *u;
342 struct PilsRequest *pr;
343
344 while (NULL != (u = u_head))
345 {
347 u_tail,
348 u);
349 if (0 == strcmp (u->name, "gnunet"))
350 {
351 DHTU_gnunet_done (u->dhtu);
352 }
353#ifdef LINUX
354 else if (0 == strcmp (u->name, "ip"))
355 {
356 DHTU_ip_done (u->dhtu);
357 }
358#endif
359 else
360 {
361 GNUNET_assert (0);
362 }
363 GNUNET_free (u->name);
364 GNUNET_free (u);
365 }
369 if (NULL != GDS_block_context)
370 {
372 GDS_block_context = NULL;
373 }
375 if (NULL != GDS_stats)
376 {
378 GNUNET_YES);
379 GDS_stats = NULL;
380 }
381 if (NULL != GDS_my_hello)
382 {
384 GDS_my_hello = NULL;
385 }
386 while (NULL != (pr = pils_requests_head))
387 {
390 pr);
391 if (NULL != pr->op)
393 GNUNET_free (pr);
394 }
396 if (NULL != GDS_pils)
397 {
399 GDS_pils = NULL;
400 }
401}
402
403
411static void
412load_underlay (void *cls,
413 const char *section)
414{
415 struct GDS_Underlay *u;
416
417 (void) cls;
418 if (0 != strncasecmp (section,
419 "dhtu-",
420 strlen ("dhtu-")))
421 return;
422 if (GNUNET_YES !=
424 section,
425 "ENABLED"))
426 return;
427 section += strlen ("dhtu-");
428 u = GNUNET_new (struct GDS_Underlay);
429 u->env.cls = u;
430 u->env.cfg = GDS_cfg;
431 u->env.address_add_cb = &u_address_add;
432 u->env.address_del_cb = &u_address_del;
433 u->env.network_size_cb = &update_network_size_estimate;
434 u->env.connect_cb = &GDS_u_connect;
435 u->env.disconnect_cb = &GDS_u_disconnect;
436 u->env.receive_cb = &GDS_u_receive;
437
440 if (0 == strcmp (section, "gnunet"))
441 {
442 u->dhtu = DHTU_gnunet_init (&u->env);
443 }
444#ifdef LINUX
445 else if (0 == strcmp (section, "ip"))
446 {
447 u->dhtu = DHTU_ip_init (&u->env);
448 }
449#endif
450 if (NULL == u->dhtu)
451 {
452 GNUNET_free (u);
453 return;
454 }
455 u->name = GNUNET_strdup (section);
457 u_tail,
458 u);
459}
460
461
462static void
463pid_change_cb (void *cls,
464 const struct GNUNET_HELLO_Parser *parser,
465 const struct GNUNET_HashCode *hash)
466{
467 (void) cls;
468
469 if (NULL != GDS_my_hello)
472
473 if (NULL == GDS_my_hello)
474 {
475 GNUNET_break (0);
476 return;
477 }
478
479 if (NULL == u_head)
480 {
483 NULL);
484 if (NULL == u_head)
485 {
487 "No DHT underlays configured!\n");
489 return;
490 }
491 }
492
494}
495
496
504static void
529
530
531/* Finally, define the main method */
533
534
535/* end of gnunet-service-dht.c */
struct GNUNET_MessageHeader * msg
Definition 005.c:2
static mp_limb_t u[(((256)+GMP_NUMB_BITS - 1)/GMP_NUMB_BITS)]
void GDS_helper_cleanup_operations(void)
Definition dht_helper.c:81
Helper functions for DHT.
struct GNUNET_PILS_Handle * GDS_pils
Handle for the pils service.
static void finished_cb(void *cls, enum GNUNET_GenericReturnValue rv)
static uint64_t timestamp(void)
Get current timestamp.
static char * address
GNS address for this phone.
static struct GNUNET_FS_Handle * ctx
static GstElement * source
Appsrc instance into which we write data for the pipeline.
struct GNUNET_SCHEDULER_Task * shutdown_task
static struct GNUNET_SERVICE_Handle * service
Handle to our service instance.
static void pid_change_cb(void *cls, const struct GNUNET_HELLO_Parser *parser, const struct GNUNET_HashCode *hash)
static void load_underlay(void *cls, const char *section)
Function iterating over all configuration sections.
double GDS_NSE_get(void)
Return the current NSE.
struct GNUNET_MessageHeader * GDS_my_hello
Our HELLO.
static struct MyAddress * a_head
Head of addresses of this peer.
static void u_address_add(void *cls, const char *address, struct GNUNET_DHTU_Source *source, void **ctx)
Function to call with new addresses of this peer.
static struct MyAddress * a_tail
Tail of addresses of this peer.
static struct PilsRequest * pils_requests_head
PILS Operation DLL.
static struct GDS_Underlay * u_head
Handles for the DHT underlays.
static void u_address_del(void *ctx)
Function to call with expired addresses of this peer.
struct GNUNET_DHTU_PreferenceHandle * GDS_u_hold(struct GDS_Underlay *u, struct GNUNET_DHTU_Target *target)
Create a hold on target at underlay u.
void GDS_u_send(struct GDS_Underlay *u, struct GNUNET_DHTU_Target *target, const void *msg, size_t msg_size, GNUNET_SCHEDULER_TaskCallback finished_cb, void *finished_cb_cls)
Send message to some other participant over the network.
static struct GDS_Underlay * u_tail
Handles for the DHT underlays.
static struct PilsRequest * pils_requests_tail
PILS Operation DLL.
static void run(void *cls, const struct GNUNET_CONFIGURATION_Handle *c, struct GNUNET_SERVICE_Handle *service)
Process dht requests.
void GDS_u_drop(struct GDS_Underlay *u, struct GNUNET_DHTU_PreferenceHandle *ph)
Drop a hold ph from underlay u.
void GDS_u_try_connect(const struct GNUNET_PeerIdentity *pid, const char *address)
Ask all underlays to connect to peer pid at address.
static double log_of_network_size_estimate
log of the current network size estimate, used as the point where we switch between random and determ...
GNUnet DHT globals.
struct GNUNET_STATISTICS_Handle * GDS_stats
Handle for the statistics service.
struct GNUNET_BLOCK_Context * GDS_block_context
Our handle to the BLOCK library.
struct GNUNET_SERVICE_Handle * GDS_service
Handle for the service.
const struct GNUNET_CONFIGURATION_Handle * GDS_cfg
Configuration we use.
GNUnet DHT service's client management code.
#define GDS_DHT_SERVICE_INIT(name, run)
Define "main" method using service macro.
void GDS_CLIENTS_init(void)
Initialize client subsystem.
void GDS_CLIENTS_stop(void)
Shutdown client subsystem.
void GDS_DATACACHE_init()
Initialize datacache subsystem.
void GDS_DATACACHE_done()
Shutdown datacache subsystem.
GNUnet DHT service's datacache integration.
void GDS_NEIGHBOURS_done()
Shutdown neighbours subsystem.
void GDS_u_disconnect(void *ctx)
Function to call when we disconnected from a peer and can henceforth cannot transmit to that peer any...
void GDS_NEIGHBOURS_broadcast(const struct GNUNET_MessageHeader *msg)
Send msg to all peers in our buckets.
void GDS_u_connect(void *cls, struct GNUNET_DHTU_Target *target, const struct GNUNET_PeerIdentity *pid, void **ctx)
Function to call when we connect to a peer and can henceforth transmit to that peer.
enum GNUNET_GenericReturnValue GDS_NEIGHBOURS_init()
Initialize neighbours subsystem.
void GDS_u_receive(void *cls, void **tctx, void **sctx, const void *message, size_t message_size)
Function to call when we receive a message.
GNUnet DHT routing code.
void GDS_ROUTING_done()
Shutdown routing subsystem.
void GDS_ROUTING_init()
Initialize routing subsystem.
GNUnet DHT tracking of requests for routing replies.
static void update_network_size_estimate()
We've sent on our flood message or one that we received which was validated and closer than ours.
static struct GNUNET_PeerIdentity pid
Identity of the peer we transmit to / connect to.
static struct GNUNET_Process * p
Helper process we started.
Definition gnunet-uri.c:38
Library for data block manipulation.
commonly used definitions; globals in this file are exempt from the rule that the module name ("commo...
Helper library for handling HELLO URIs.
struct GNUNET_PILS_Handle * GNUNET_PILS_connect(const struct GNUNET_CONFIGURATION_Handle *cfg, GNUNET_PILS_PidChangeCallback pid_change_cb, void *cls)
Connect to the PILS service.
Definition pils_api.c:465
void GNUNET_PILS_disconnect(struct GNUNET_PILS_Handle *handle)
Disconnect from the PILS service.
Definition pils_api.c:488
void GNUNET_PILS_cancel(struct GNUNET_PILS_Operation *op)
Cancel request.
Definition pils_api.c:623
API to create, modify and access statistics.
void GNUNET_BLOCK_context_destroy(struct GNUNET_BLOCK_Context *ctx)
Destroy the block context.
Definition block.c:158
struct GNUNET_BLOCK_Context * GNUNET_BLOCK_context_create(const struct GNUNET_CONFIGURATION_Handle *cfg)
Create a block context.
Definition block.c:140
void GNUNET_CONFIGURATION_iterate_sections(const struct GNUNET_CONFIGURATION_Handle *cfg, GNUNET_CONFIGURATION_SectionIterator iter, void *iter_cls)
Iterate over all sections in the configuration.
enum GNUNET_GenericReturnValue GNUNET_CONFIGURATION_get_value_yesno(const struct GNUNET_CONFIGURATION_Handle *cfg, const char *section, const char *option)
Get a configuration value that should be in a set of "YES" or "NO".
#define GNUNET_CONTAINER_DLL_remove(head, tail, element)
Remove an element from a DLL.
#define GNUNET_CONTAINER_DLL_insert(head, tail, element)
Insert an element at the head of a DLL.
struct GNUNET_MessageHeader * GNUNET_HELLO_parser_to_dht_hello_msg(const struct GNUNET_HELLO_Parser *parser)
Generate DHT HELLO message from a parser.
Definition hello-uri.c:1208
#define GNUNET_log(kind,...)
#define GNUNET_MAX(a, b)
@ GNUNET_OK
@ GNUNET_YES
@ GNUNET_NO
#define GNUNET_assert(cond)
Use this for fatal errors that cannot be handled.
#define GNUNET_break(cond)
Use this for internal assertion violations that are not fatal (can be handled) but should not occur.
@ GNUNET_ERROR_TYPE_ERROR
@ GNUNET_ERROR_TYPE_INFO
#define GNUNET_strdup(a)
Wrapper around GNUNET_xstrdup_.
#define GNUNET_new(type)
Allocate a struct or union of the given type.
#define GNUNET_free(ptr)
Wrapper around free.
void GNUNET_SCHEDULER_shutdown(void)
Request the shutdown of a scheduler.
Definition scheduler.c:572
struct GNUNET_SCHEDULER_Task * GNUNET_SCHEDULER_add_shutdown(GNUNET_SCHEDULER_TaskCallback task, void *task_cls)
Schedule a new task to be run on shutdown, that is when a CTRL-C signal is received,...
Definition scheduler.c:1345
void(* GNUNET_SCHEDULER_TaskCallback)(void *cls)
Signature of the main function of a task.
struct GNUNET_STATISTICS_Handle * GNUNET_STATISTICS_create(const char *subsystem, const struct GNUNET_CONFIGURATION_Handle *cfg)
Get handle for the statistics service.
void GNUNET_STATISTICS_update(struct GNUNET_STATISTICS_Handle *handle, const char *name, int64_t delta, int make_persistent)
Set statistic value for the peer.
void GNUNET_STATISTICS_destroy(struct GNUNET_STATISTICS_Handle *h, int sync_first)
Destroy a handle (free all state associated with it).
void * DHTU_gnunet_done(struct GNUNET_DHTU_PluginFunctions *api)
Exit point from the plugin.
struct GNUNET_DHTU_PluginFunctions * DHTU_gnunet_init(struct GNUNET_DHTU_PluginEnvironment *env)
Entry point for the plugin.
struct GNUNET_DHTU_PluginFunctions * DHTU_ip_init(struct GNUNET_DHTU_PluginEnvironment *env)
Entry point for the plugin.
void * DHTU_ip_done(struct GNUNET_DHTU_PluginFunctions *api)
Exit point from the plugin.
Information we keep per underlay.
struct GNUNET_DHTU_PluginEnvironment env
Environment for this underlay.
struct GDS_Underlay * prev
Kept in a DLL.
double network_size_estimate
current network size estimate for this underlay.
char * name
Name of the underlay (i.e.
struct GDS_Underlay * next
Kept in a DLL.
struct GNUNET_DHTU_PluginFunctions * dhtu
Underlay API handle.
The datastore service will pass a pointer to a struct of this type as the first and only argument to ...
struct returned by the initialization function of the plugin
Opaque handle expressing a preference of the DHT to keep a particular target connected.
struct GNUNET_DHTU_Target * target
Target a preference was expressed for.
Opaque handle that the underlay offers for our address to be used when sending messages to another pe...
Opaque handle that the underlay offers for the target peer when sending messages to another peer.
Context for parsing HELLOs.
Definition hello-uri.c:233
A 512-bit hashcode.
Header for all communications.
The identity of the host (wraps the signing key of the peer).
Handle to a service.
Definition service.c:116
Time for absolute times used by GNUnet, in microseconds.
An address of this peer.
char * url
Textual representation of the address.
struct GDS_Underlay * u
Underlay of this address.
struct MyAddress * prev
Kept in a DLL.
struct MyAddress * next
Kept in a DLL.
struct GNUNET_DHTU_Source * source
Underlay handle for the address.
struct GNUNET_PILS_Operation * op
The pils operation.
struct PilsRequest * next
DLL.
struct PilsRequest * prev
DLL.