GNUnet 0.27.0
 
Loading...
Searching...
No Matches
gnunet-service-fs_cp.c File Reference

API to handle 'connected peers'. More...

Include dependency graph for gnunet-service-fs_cp.c:

Go to the source code of this file.

Data Structures

struct  GSF_PeerTransmitHandle
 Handle to cancel a transmission request. More...
 
struct  GSF_DelayedHandle
 Handle for an entry in our delay list. More...
 
struct  PeerRequest
 Information per peer and request. More...
 
struct  GSF_ConnectedPeer
 A connected peer. More...
 
struct  TestExistClosure
 Closure for test_exist_cb(). More...
 
struct  IterationContext
 Closure for call_iterator(). More...
 

Macros

#define RUNAVG_DELAY_N   16
 Ratio for moving average delay calculation.
 
#define RESPECT_FLUSH_FREQ
 How often do we flush respect values to disk?
 
#define REPLY_TIMEOUT
 After how long do we discard a reply?
 
#define INSANE_STATISTICS   GNUNET_NO
 Collect an instance number of statistics? May cause excessive IPC.
 
#define N   ((double) 128.0)
 

Functions

void GSF_update_peer_latency_ (const struct GNUNET_PeerIdentity *id, struct GNUNET_TIME_Relative latency)
 Update the latency information kept for the given peer.
 
struct GSF_PeerPerformanceDataGSF_get_peer_performance_data_ (struct GSF_ConnectedPeer *cp)
 Return the performance data record for the given peer.
 
static void peer_transmit (struct GSF_ConnectedPeer *cp)
 Core is ready to transmit to a peer, get the message.
 
static void schedule_transmission (struct GSF_PeerTransmitHandle *pth)
 If ready (bandwidth reserved), try to schedule transmission via core for the given handle.
 
static void peer_respect_cb (void *cls, const struct GNUNET_PEERSTORE_Record *record, const char *emsg)
 Function called by PEERSTORE with peer respect record.
 
static int consider_peer_for_forwarding (void *cls, const struct GNUNET_HashCode *key, struct GSF_PendingRequest *pr)
 Function called for each pending request whenever a new peer connects, giving us a chance to decide about submitting the existing request to the new peer.
 
void * GSF_peer_connect_handler (void *cls, const struct GNUNET_PeerIdentity *peer, struct GNUNET_MQ_Handle *mq, enum GNUNET_CORE_PeerClass class)
 A peer connected to us.
 
static void revive_migration (void *cls)
 It may be time to re-start migrating content to this peer.
 
struct GSF_ConnectedPeerGSF_peer_get_ (const struct GNUNET_PeerIdentity *peer)
 Get a handle for a connected peer.
 
void handle_p2p_migration_stop (void *cls, const struct MigrationStopMessage *msm)
 Handle P2P GNUNET_MESSAGE_TYPE_FS_MIGRATION_STOP message.
 
static void free_pending_request (struct PeerRequest *peerreq)
 Free resources associated with the given peer request.
 
static int cancel_pending_request (void *cls, const struct GNUNET_HashCode *query, void *value)
 Cancel all requests associated with the peer.
 
static void peer_request_destroy (void *cls)
 Free the given request.
 
static void transmit_delayed_now (void *cls)
 The artificial delay is over, transmit the message now.
 
static struct GNUNET_TIME_Relative get_randomized_delay ()
 Get the randomized delay a response should be subjected to.
 
static void handle_p2p_reply (void *cls, enum GNUNET_BLOCK_ReplyEvaluationResult eval, struct GSF_PendingRequest *pr, uint32_t reply_anonymity_level, struct GNUNET_TIME_Absolute expiration, struct GNUNET_TIME_Absolute last_transmission, enum GNUNET_BLOCK_Type type, const void *data, size_t data_len)
 Handle a reply to a pending request.
 
static int change_peer_respect (struct GSF_ConnectedPeer *cp, int value)
 Increase the peer's respect by a value.
 
static int32_t bound_priority (uint32_t prio_in, struct GSF_ConnectedPeer *cp)
 We've received a request with the specified priority.
 
static int32_t bound_ttl (int32_t ttl_in, uint32_t prio)
 The priority level imposes a bound on the maximum value for the ttl that can be requested.
 
static int test_exist_cb (void *cls, const struct GNUNET_HashCode *hc, void *value)
 Test if the query already exists.
 
void handle_p2p_get (void *cls, const struct GetMessage *gm)
 Handle P2P "QUERY" message.
 
void GSF_peer_transmit_ (struct GSF_ConnectedPeer *cp, int is_query, uint32_t priority, struct GNUNET_MQ_Envelope *env)
 Transmit a message to the given peer as soon as possible.
 
void GSF_peer_update_performance_ (struct GSF_ConnectedPeer *cp, struct GNUNET_TIME_Absolute request_time, uint32_t request_priority)
 Report on receiving a reply; update the performance record of the given peer.
 
void GSF_peer_update_responder_client_ (struct GSF_ConnectedPeer *cp, struct GSF_LocalClient *initiator_client)
 Report on receiving a reply in response to an initiating client.
 
void GSF_peer_update_responder_peer_ (struct GSF_ConnectedPeer *cp, const struct GSF_ConnectedPeer *initiator_peer)
 Report on receiving a reply in response to an initiating peer.
 
static int flush_respect (void *cls, const struct GNUNET_PeerIdentity *key, void *value)
 Write peer-respect information to a file - flush the buffer entry!
 
void GSF_peer_disconnect_handler (void *cls, const struct GNUNET_PeerIdentity *peer, void *internal_cls)
 A peer disconnected from us.
 
static int call_iterator (void *cls, const struct GNUNET_PeerIdentity *key, void *value)
 Function that calls the callback for each peer.
 
void GSF_iterate_connected_peers_ (GSF_ConnectedPeerIterator it, void *it_cls)
 Iterate over all connected peers.
 
void GSF_connected_peer_get_identity_ (const struct GSF_ConnectedPeer *cp, struct GNUNET_PeerIdentity *id)
 Obtain the identity of a connected peer.
 
const struct GNUNET_PeerIdentityGSF_connected_peer_get_identity2_ (const struct GSF_ConnectedPeer *cp)
 Obtain the identity of a connected peer.
 
void GSF_block_peer_migration_ (struct GSF_ConnectedPeer *cp, struct GNUNET_TIME_Absolute block_time)
 Ask a peer to stop migrating data to us until the given point in time.
 
void GSF_connected_peer_change_preference_ (struct GSF_ConnectedPeer *cp, uint64_t pref)
 Notify core about a preference we have for the given peer (to allocate more resources towards it).
 
static void cron_flush_respect (void *cls)
 Call this method periodically to flush respect information to disk.
 
void GSF_connected_peer_init_ ()
 Initialize peer management subsystem.
 
void GSF_connected_peer_done_ ()
 Shutdown peer management subsystem.
 
static int clean_local_client (void *cls, const struct GNUNET_PeerIdentity *key, void *value)
 Iterator to remove references to LC entry.
 
void GSF_handle_local_client_disconnect_ (const struct GSF_LocalClient *lc)
 Notification that a local client disconnected.
 

Variables

static struct GNUNET_CONTAINER_MultiPeerMapcp_map
 Map from peer identities to struct GSF_ConnectPeer entries.
 
static struct GNUNET_PEERSTORE_Handlepeerstore
 Handle to peerstore service.
 
static struct GNUNET_SCHEDULER_Taskfr_task
 Task used to flush respect values to disk.
 

Detailed Description

API to handle 'connected peers'.

Author
Christian Grothoff

Definition in file gnunet-service-fs_cp.c.

Macro Definition Documentation

◆ RUNAVG_DELAY_N

#define RUNAVG_DELAY_N   16

Ratio for moving average delay calculation.

The previous average goes in with a factor of (n-1) into the calculation. Must be > 0.

Definition at line 43 of file gnunet-service-fs_cp.c.

◆ RESPECT_FLUSH_FREQ

#define RESPECT_FLUSH_FREQ
Value:
#define GNUNET_TIME_UNIT_MINUTES
One minute.
struct GNUNET_TIME_Relative GNUNET_TIME_relative_multiply(struct GNUNET_TIME_Relative rel, unsigned long long factor)
Multiply relative time by a given factor.
Definition time.c:486

How often do we flush respect values to disk?

Definition at line 48 of file gnunet-service-fs_cp.c.

66{
71
76
80 struct GNUNET_TIME_Absolute transmission_request_start_time;
81
85 struct GNUNET_MQ_Envelope *env;
86
90 struct GSF_ConnectedPeer *cp;
91
95 int is_query;
96
100 uint32_t priority;
101};
102
103
108{
112 struct GSF_DelayedHandle *next;
113
117 struct GSF_DelayedHandle *prev;
118
122 struct GSF_ConnectedPeer *cp;
123
127 struct GNUNET_MQ_Envelope *env;
128
133
137 size_t msize;
138};
139
140
144struct PeerRequest
145{
149 struct GSF_PendingRequest *pr;
150
154 struct GSF_ConnectedPeer *cp;
155
160};
161
162
167{
172
178
183
189
195
201
207
212
217
222 struct GNUNET_MQ_Handle *mq;
223
228 uint64_t inc_preference;
229
233 unsigned int delay_queue_size;
234
238 uint32_t disk_respect;
239
244 unsigned int last_p2p_replies_woff;
245
250 unsigned int last_client_replies_woff;
251
255 unsigned int last_request_times_off;
256
261};
262
263
268
273
277static struct GNUNET_SCHEDULER_Task *fr_task;
278
279
286void
288 struct GNUNET_TIME_Relative latency)
289{
290 struct GSF_ConnectedPeer *cp;
291
292 cp = GSF_peer_get_ (id);
293 if (NULL == cp)
294 return; /* we're not yet connected at the core level, ignore */
296 latency);
297}
298
299
308{
309 return &cp->ppd;
310}
311
312
318static void
320
321
328static void
330{
331 struct GSF_ConnectedPeer *cp;
332 struct GNUNET_PeerIdentity target;
333
334 cp = pth->cp;
335 GNUNET_assert (0 != cp->ppd.pid);
336 GNUNET_PEER_resolve (cp->ppd.pid, &target);
337
338 peer_transmit (cp);
339}
340
341
347static void
349{
350 struct GSF_PeerTransmitHandle *pth = cp->pth_head;
351 struct GSF_PeerTransmitHandle *pos;
352
353 if (NULL == pth)
354 return;
356 cp->pth_tail,
357 pth);
358 if (GNUNET_YES == pth->is_query)
359 {
364 }
365 else if (GNUNET_NO == pth->is_query)
366 {
368 }
373 pth->env);
374 GNUNET_free (pth);
375 if (NULL != (pos = cp->pth_head))
376 {
377 GNUNET_assert (pos != pth);
379 }
380}
381
382
390static void
391peer_respect_cb (void *cls,
392 const struct GNUNET_PEERSTORE_Record *record,
393 const char *emsg)
394{
395 struct GSF_ConnectedPeer *cp = cls;
396
398 if ((NULL == record) && (NULL == emsg))
399 {
400 cp->respect_iterate_req = NULL;
401 return;
402 }
403 if ((NULL != record) &&
404 (sizeof(cp->disk_respect) == record->value_size))
405 {
406 cp->disk_respect = *((uint32_t *) record->value);
407 cp->ppd.respect += *((uint32_t *) record->value);
408 }
409 GSF_push_start_ (cp);
410 if (NULL != record)
411 {
413 cp->respect_iterate_req = NULL;
414 return;
415 }
417}
418
419
430static int
432 const struct GNUNET_HashCode *key,
433 struct GSF_PendingRequest *pr)
434{
435 struct GSF_ConnectedPeer *cp = cls;
437
438 if (GNUNET_YES !=
440 return GNUNET_YES; /* request is not actually active, skip! */
442 if (GNUNET_YES !=
444 {
446 gettext_noop ("# Loopback routes suppressed"),
447 1,
448 GNUNET_NO);
449 return GNUNET_YES;
450 }
451 GSF_plan_add_ (cp, pr);
452 return GNUNET_YES;
453}
454
455
456void *
457GSF_peer_connect_handler (void *cls,
458 const struct GNUNET_PeerIdentity *peer,
459 struct GNUNET_MQ_Handle *mq,
460 enum GNUNET_CORE_PeerClass class)
461{
462 const struct GNUNET_PeerIdentity *my_identity;
463 struct GSF_ConnectedPeer *cp;
464
467
468 if (0 == GNUNET_memcmp (my_identity, peer))
469 return NULL;
471 "Connected to peer %s\n",
472 GNUNET_i2s (peer));
473 cp = GNUNET_new (struct GSF_ConnectedPeer);
474 cp->ppd.pid = GNUNET_PEER_intern (peer);
475 cp->ppd.peer = peer;
476 cp->mq = mq;
478
480 GNUNET_YES);
484 (
485 cp),
486 cp,
489 gettext_noop ("# peers connected"),
491 GNUNET_NO);
494 "fs",
495 peer,
496 "respect",
498 cp);
500 cp);
501 return cp;
502}
503
504
511static void
512revive_migration (void *cls)
513{
514 struct GSF_ConnectedPeer *cp = cls;
515 struct GNUNET_TIME_Relative bt;
516
517 cp->mig_revive_task = NULL;
519 if (0 != bt.rel_value_us)
520 {
521 /* still time left... */
522 cp->mig_revive_task =
524 return;
525 }
526 GSF_push_start_ (cp);
527}
528
529
530struct GSF_ConnectedPeer *
531GSF_peer_get_ (const struct GNUNET_PeerIdentity *peer)
532{
533 if (NULL == cp_map)
534 return NULL;
536}
537
538
545void
547 const struct MigrationStopMessage *msm)
548{
549 struct GSF_ConnectedPeer *cp = cls;
550 struct GNUNET_TIME_Relative bt;
551
553 gettext_noop ("# migration stop messages received"),
554 1, GNUNET_NO);
557 _ ("Migration of content to peer `%s' blocked for %s\n"),
558 GNUNET_i2s (cp->ppd.peer),
561 if ((NULL == cp->mig_revive_task) &&
562 (NULL == cp->respect_iterate_req))
563 {
564 GSF_push_stop_ (cp);
565 cp->mig_revive_task =
567 &revive_migration, cp);
568 }
569}
570
571
577static void
578free_pending_request (struct PeerRequest *peerreq)
579{
580 struct GSF_ConnectedPeer *cp = peerreq->cp;
581 struct GSF_PendingRequestData *prd;
582
583 prd = GSF_pending_request_get_data_ (peerreq->pr);
584 if (NULL != peerreq->kill_task)
585 {
587 peerreq->kill_task = NULL;
588 }
590 gettext_noop ("# P2P searches active"),
591 -1,
592 GNUNET_NO);
595 &prd->query,
596 peerreq));
597 GNUNET_free (peerreq);
598}
599
600
609static int
610cancel_pending_request (void *cls,
611 const struct GNUNET_HashCode *query,
612 void *value)
613{
614 struct PeerRequest *peerreq = value;
615 struct GSF_PendingRequest *pr = peerreq->pr;
616
617 free_pending_request (peerreq);
619 GNUNET_NO);
620 return GNUNET_OK;
621}
622
623
629static void
630peer_request_destroy (void *cls)
631{
632 struct PeerRequest *peerreq = cls;
633 struct GSF_PendingRequest *pr = peerreq->pr;
634 struct GSF_PendingRequestData *prd;
635
636 peerreq->kill_task = NULL;
639 &prd->query,
640 peerreq);
641}
642
643
649static void
650transmit_delayed_now (void *cls)
651{
652 struct GSF_DelayedHandle *dh = cls;
653 struct GSF_ConnectedPeer *cp = dh->cp;
654
656 cp->delayed_tail,
657 dh);
658 cp->delay_queue_size--;
660 GNUNET_NO,
661 UINT32_MAX,
662 dh->env);
663 GNUNET_free (dh);
664}
665
666
672static struct GNUNET_TIME_Relative
674{
676
677 ret =
682#if INSANE_STATISTICS
685 ("# artificial delays introduced (ms)"),
686 ret.rel_value_us / 1000LL, GNUNET_NO);
687#endif
688 return ret;
689}
690
691
710static void
711handle_p2p_reply (void *cls,
713 struct GSF_PendingRequest *pr,
714 uint32_t reply_anonymity_level,
716 struct GNUNET_TIME_Absolute last_transmission,
718 const void *data,
719 size_t data_len)
720{
721 struct PeerRequest *peerreq = cls;
722 struct GSF_ConnectedPeer *cp = peerreq->cp;
723 struct GSF_PendingRequestData *prd;
724 struct GNUNET_MQ_Envelope *env;
725 struct PutMessage *pm;
726 size_t msize;
727
728 GNUNET_assert (data_len + sizeof(struct PutMessage) <
730 GNUNET_assert (peerreq->pr == pr);
732 if (NULL == data)
733 {
734 free_pending_request (peerreq);
735 return;
736 }
738 if ( (prd->type != type) &&
739 (GNUNET_BLOCK_TYPE_ANY != prd->type) )
740 {
742 "# replies dropped due to type mismatch",
743 1, GNUNET_NO);
744 return;
745 }
747 "Transmitting result for query `%s' to peer\n",
748 GNUNET_h2s (&prd->query));
750 "# replies received for other peers",
751 1,
752 GNUNET_NO);
753 msize = sizeof(struct PutMessage) + data_len;
754 if (msize >= GNUNET_MAX_MESSAGE_SIZE)
755 {
756 GNUNET_break (0);
757 return;
758 }
759 if ( (UINT32_MAX != reply_anonymity_level) &&
760 (reply_anonymity_level > 1) )
761 {
762 if (reply_anonymity_level - 1 > GSF_cover_content_count)
763 {
765 "# replies dropped due to insufficient cover traffic",
766 1, GNUNET_NO);
767 return;
768 }
769 GSF_cover_content_count -= (reply_anonymity_level - 1);
770 }
771
773 data_len,
775 pm->type = htonl (type);
777 GNUNET_memcpy (&pm[1],
778 data,
779 data_len);
780 if ((UINT32_MAX != reply_anonymity_level) &&
781 (0 != reply_anonymity_level) &&
783 {
784 struct GSF_DelayedHandle *dh;
785
786 dh = GNUNET_new (struct GSF_DelayedHandle);
787 dh->cp = cp;
788 dh->env = env;
789 dh->msize = msize;
792 dh);
794 dh->delay_task =
797 dh);
798 }
799 else
800 {
802 GNUNET_NO,
803 UINT32_MAX,
804 env);
805 }
806 if (GNUNET_BLOCK_REPLY_OK_LAST != eval)
807 return;
808 if (NULL == peerreq->kill_task)
809 {
811 "# P2P searches destroyed due to ultimate reply",
812 1,
813 GNUNET_NO);
814 peerreq->kill_task =
816 peerreq);
817 }
818}
819
820
829static int
831{
832 if (0 == value)
833 return 0;
834 GNUNET_assert (NULL != cp);
835 if (value > 0)
836 {
837 if (cp->ppd.respect + value < cp->ppd.respect)
838 {
839 value = UINT32_MAX - cp->ppd.respect;
840 cp->ppd.respect = UINT32_MAX;
841 }
842 else
843 cp->ppd.respect += value;
844 }
845 else
846 {
847 if (cp->ppd.respect < -value)
848 {
849 value = -cp->ppd.respect;
850 cp->ppd.respect = 0;
851 }
852 else
853 cp->ppd.respect += value;
854 }
855 return value;
856}
857
858
867static int32_t
868bound_priority (uint32_t prio_in,
869 struct GSF_ConnectedPeer *cp)
870{
871#define N ((double) 128.0)
872 uint32_t ret;
873 double rret;
874 int ld;
875
877 if (GNUNET_SYSERR == ld)
878 {
879#if INSANE_STATISTICS
882 ("# requests done for free (low load)"), 1,
883 GNUNET_NO);
884#endif
885 return 0; /* excess resources */
886 }
887 if (prio_in > INT32_MAX)
888 prio_in = INT32_MAX;
889 ret = -change_peer_respect (cp, -(int) prio_in);
890 if (ret > 0)
891 {
893 rret = GSF_current_priorities + N;
894 else
895 rret = ret;
897 }
898 if ((GNUNET_YES == ld) && (ret > 0))
899 {
900 /* try with charging */
902 }
903 if (GNUNET_YES == ld)
904 {
907 ("# request dropped, priority insufficient"), 1,
908 GNUNET_NO);
909 /* undo charge */
910 change_peer_respect (cp, (int) ret);
911 return -1; /* not enough resources */
912 }
913 else
914 {
917 ("# requests done for a price (normal load)"),
918 1,
919 GNUNET_NO);
920 }
921#undef N
922 return ret;
923}
924
925
935static int32_t
936bound_ttl (int32_t ttl_in,
937 uint32_t prio)
938{
939 unsigned long long allowed;
940
941 if (ttl_in <= 0)
942 return ttl_in;
943 allowed = ((unsigned long long) prio) * TTL_DECREMENT / 1000;
944 if (ttl_in > allowed)
945 {
946 if (allowed >= (1 << 30))
947 return 1 << 30;
948 return allowed;
949 }
950 return ttl_in;
951}
952
953
957struct TestExistClosure
958{
962 int32_t priority;
963
967 int32_t ttl;
968
973
977 int finished;
978};
979
980
991static int
992test_exist_cb (void *cls,
993 const struct GNUNET_HashCode *hc,
994 void *value)
995{
996 struct TestExistClosure *tec = cls;
997 struct PeerRequest *peerreq = value;
998 struct GSF_PendingRequest *pr;
999 struct GSF_PendingRequestData *prd;
1000
1001 pr = peerreq->pr;
1003 if (prd->type != tec->type)
1004 return GNUNET_YES;
1005 if (prd->ttl.abs_value_us >=
1006 GNUNET_TIME_absolute_get ().abs_value_us + tec->ttl * 1000LL)
1007 {
1008 /* existing request has higher TTL, drop new one! */
1009 prd->priority += tec->priority;
1011 "Have existing request with higher TTL, dropping new request.\n");
1014 ("# requests dropped due to higher-TTL request")
1015 ,
1016 1, GNUNET_NO);
1017 tec->finished = GNUNET_YES;
1018 return GNUNET_NO;
1019 }
1020 /* existing request has lower TTL, drop old one! */
1021 tec->priority += prd->priority;
1022 free_pending_request (peerreq);
1024 GNUNET_YES);
1025 return GNUNET_NO;
1026}
1027
1028
1038void
1039handle_p2p_get (void *cls,
1040 const struct GetMessage *gm)
1041{
1042 struct GSF_ConnectedPeer *cps = cls;
1043 struct PeerRequest *peerreq;
1044 struct GSF_PendingRequest *pr;
1045 struct GSF_ConnectedPeer *cp;
1046 const struct GNUNET_PeerIdentity *target;
1048 uint16_t msize;
1049 unsigned int bits;
1050 const struct GNUNET_PeerIdentity *opt;
1051 uint32_t bm;
1052 size_t bfsize;
1053 uint32_t ttl_decrement;
1054 struct TestExistClosure tec;
1055 GNUNET_PEER_Id spid;
1056 const struct GSF_PendingRequestData *prd;
1057
1058 msize = ntohs (gm->header.size);
1059 tec.type = ntohl (gm->type);
1060 bm = ntohl (gm->hash_bitmap);
1061 bits = 0;
1062 while (bm > 0)
1063 {
1064 if (1 == (bm & 1))
1065 bits++;
1066 bm >>= 1;
1067 }
1068 opt = (const struct GNUNET_PeerIdentity *) &gm[1];
1069 bfsize = msize - sizeof(struct GetMessage) - bits * sizeof(struct
1073 ("# GET requests received (from other peers)"),
1074 1,
1075 GNUNET_NO);
1077 bm = ntohl (gm->hash_bitmap);
1078 bits = 0;
1079 if (0 != (bm & GET_MESSAGE_BIT_RETURN_TO))
1080 cp = GSF_peer_get_ (&opt[bits++]);
1081 else
1082 cp = cps;
1083 if (NULL == cp)
1084 {
1085 if (0 != (bm & GET_MESSAGE_BIT_RETURN_TO))
1087 "Failed to find RETURN-TO peer `%s' in connection set. Dropping query.\n",
1088 GNUNET_i2s (&opt[bits - 1]));
1089
1090 else
1092 "Failed to find peer `%s' in connection set. Dropping query.\n",
1093 GNUNET_i2s (cps->ppd.peer));
1096 (
1097 "# requests dropped due to missing reverse route"),
1098 1,
1099 GNUNET_NO);
1100 return;
1101 }
1102 unsigned int queue_size = GNUNET_MQ_get_length (cp->mq);
1103 queue_size += cp->ppd.pending_replies + cp->delay_queue_size;
1104 if (queue_size > MAX_QUEUE_PER_PEER)
1105 {
1107 "Peer `%s' has too many replies queued already. Dropping query.\n",
1108 GNUNET_i2s (cps->ppd.peer));
1110 gettext_noop (
1111 "# requests dropped due to full reply queue"),
1112 1,
1113 GNUNET_NO);
1114 return;
1115 }
1116 /* note that we can really only check load here since otherwise
1117 * peers could find out that we are overloaded by not being
1118 * disconnected after sending us a malformed query... */
1119 tec.priority = bound_priority (ntohl (gm->priority),
1120 cps);
1121 if (tec.priority < 0)
1122 {
1124 "Dropping query from `%s', this peer is too busy.\n",
1125 GNUNET_i2s (cps->ppd.peer));
1126 return;
1127 }
1129 "Received request for `%s' of type %u from peer `%s' with flags %u\n",
1130 GNUNET_h2s (&gm->query),
1131 (unsigned int) tec.type,
1132 GNUNET_i2s (cps->ppd.peer),
1133 (unsigned int) bm);
1134 target =
1135 (0 !=
1136 (bm & GET_MESSAGE_BIT_TRANSMIT_TO)) ? (&opt[bits++]) : NULL;
1138 spid = 0;
1139 if ((GNUNET_LOAD_get_load (cp->ppd.transmission_delay) > 3 * (1
1140 + tec.priority))
1142 GNUNET_CONSTANTS_MAX_CORK_DELAY.rel_value_us * 2
1144 {
1145 /* don't have BW to send to peer, or would likely take longer than we have for it,
1146 * so at best indirect the query */
1147 tec.priority = 0;
1149 spid = GNUNET_PEER_intern (cps->ppd.peer);
1150 GNUNET_assert (0 != spid);
1151 }
1152 tec.ttl = bound_ttl (ntohl (gm->ttl),
1153 tec.priority);
1154 /* decrement ttl (always) */
1155 ttl_decrement =
1158 if ((tec.ttl < 0) &&
1159 (((int32_t) (tec.ttl - ttl_decrement)) > 0))
1160 {
1162 "Dropping query from `%s' due to TTL underflow (%d - %u).\n",
1163 GNUNET_i2s (cps->ppd.peer),
1164 tec.ttl,
1165 ttl_decrement);
1168 ("# requests dropped due TTL underflow"), 1,
1169 GNUNET_NO);
1170 /* integer underflow => drop (should be very rare)! */
1171 return;
1172 }
1173 tec.ttl -= ttl_decrement;
1174
1175 /* test if the request already exists */
1176 tec.finished = GNUNET_NO;
1178 &gm->query,
1180 &tec);
1181 if (GNUNET_YES == tec.finished)
1182 return; /* merged into existing request, we're done */
1183
1184 peerreq = GNUNET_new (struct PeerRequest);
1185 peerreq->cp = cp;
1187 tec.type,
1188 &gm->query,
1189 target,
1190 (bfsize > 0)
1191 ? (const char *) &opt[bits]
1192 : NULL,
1193 bfsize,
1194 1 /* anonymity */,
1195 (uint32_t) tec.priority,
1196 tec.ttl,
1197 spid,
1198 GNUNET_PEER_intern (cps->ppd.peer),
1199 NULL, 0, /* replies_seen */
1201 peerreq);
1202 GNUNET_assert (NULL != pr);
1204 peerreq->pr = pr;
1207 &prd->query,
1208 peerreq,
1211 gettext_noop (
1212 "# P2P query messages received and processed"),
1213 1,
1214 GNUNET_NO);
1216 gettext_noop ("# P2P searches active"),
1217 1,
1218 GNUNET_NO);
1222 NULL);
1223}
1224
1225
1236void
1238 int is_query,
1239 uint32_t priority,
1240 struct GNUNET_MQ_Envelope *env)
1241{
1242 struct GSF_PeerTransmitHandle *pth;
1243 struct GSF_PeerTransmitHandle *pos;
1245
1246 pth = GNUNET_new (struct GSF_PeerTransmitHandle);
1248 pth->env = env;
1249 pth->is_query = is_query;
1250 pth->priority = priority;
1251 pth->cp = cp;
1252 /* insertion sort (by priority, descending) */
1253 prev = NULL;
1254 pos = cp->pth_head;
1255 while ((NULL != pos) && (pos->priority > priority))
1256 {
1257 prev = pos;
1258 pos = pos->next;
1259 }
1261 cp->pth_tail,
1262 prev,
1263 pth);
1264 if (GNUNET_YES == is_query)
1266 else if (GNUNET_NO == is_query)
1269}
1270
1271
1279void
1281 struct GNUNET_TIME_Absolute request_time,
1282 uint32_t request_priority)
1283{
1284 struct GNUNET_TIME_Relative delay;
1285
1286 delay = GNUNET_TIME_absolute_get_duration (request_time);
1289 + delay.rel_value_us) / RUNAVG_DELAY_N;
1290 cp->ppd.avg_priority =
1291 (cp->ppd.avg_priority * (RUNAVG_DELAY_N - 1)
1292 + request_priority) / RUNAVG_DELAY_N;
1293}
1294
1295
1303void
1305 struct GSF_LocalClient *initiator_client)
1306{
1308 % CS2P_SUCCESS_LIST_SIZE] = initiator_client;
1309}
1310
1311
1319void
1321 const struct GSF_ConnectedPeer *initiator_peer)
1322{
1323 unsigned int woff;
1324
1327 cp->ppd.last_p2p_replies[woff] = initiator_peer->ppd.pid;
1328 GNUNET_PEER_change_rc (initiator_peer->ppd.pid, 1);
1330}
1331
1332
1341static int
1342flush_respect (void *cls,
1343 const struct GNUNET_PeerIdentity *key,
1344 void *value)
1345{
1346 struct GSF_ConnectedPeer *cp = value;
1347 struct GNUNET_PeerIdentity pid;
1348
1349 if (cp->ppd.respect == cp->disk_respect)
1350 return GNUNET_OK; /* unchanged */
1351 GNUNET_assert (0 != cp->ppd.pid);
1353 GNUNET_PEERSTORE_store (peerstore, "fs", &pid, "respect", &cp->ppd.respect,
1354 sizeof(cp->ppd.respect),
1357 NULL,
1358 NULL);
1359 return GNUNET_OK;
1360}
1361
1362
1363void
1365 const struct GNUNET_PeerIdentity *peer,
1366 void *internal_cls)
1367{
1368 struct GSF_ConnectedPeer *cp = internal_cls;
1369 struct GSF_PeerTransmitHandle *pth;
1370 struct GSF_DelayedHandle *dh;
1371
1372 if (NULL == cp)
1373 return; /* must have been disconnect from core with
1374 * 'peer' == my_id, ignore */
1375 flush_respect (NULL,
1376 peer,
1377 cp);
1380 peer,
1381 cp));
1383 gettext_noop ("# peers connected"),
1385 GNUNET_NO);
1386 if (NULL != cp->respect_iterate_req)
1387 {
1389 cp->respect_iterate_req = NULL;
1390 }
1393 cp);
1395 cp->request_map = NULL;
1400 memset (cp->ppd.last_p2p_replies,
1401 0,
1402 sizeof(cp->ppd.last_p2p_replies));
1404 while (NULL != (pth = cp->pth_head))
1405 {
1407 cp->pth_tail,
1408 pth);
1409 if (GNUNET_YES == pth->is_query)
1411 else if (GNUNET_NO == pth->is_query)
1413 GNUNET_free (pth);
1414 }
1415 while (NULL != (dh = cp->delayed_head))
1416 {
1419 dh);
1420 GNUNET_MQ_discard (dh->env);
1423 GNUNET_free (dh);
1424 }
1426 if (NULL != cp->mig_revive_task)
1427 {
1429 cp->mig_revive_task = NULL;
1430 }
1433 GNUNET_free (cp);
1434}
1435
1436
1440struct IterationContext
1441{
1446
1450 void *it_cls;
1451};
1452
1453
1462static int
1463call_iterator (void *cls,
1464 const struct GNUNET_PeerIdentity *key,
1465 void *value)
1466{
1467 struct IterationContext *ic = cls;
1468 struct GSF_ConnectedPeer *cp = value;
1469
1470 ic->it (ic->it_cls,
1471 key, cp,
1472 &cp->ppd);
1473 return GNUNET_YES;
1474}
1475
1476
1477void
1479 void *it_cls)
1480{
1481 struct IterationContext ic;
1482
1483 ic.it = it;
1484 ic.it_cls = it_cls;
1487 &ic);
1488}
1489
1490
1497void
1499 struct GNUNET_PeerIdentity *id)
1500{
1501 GNUNET_assert (0 != cp->ppd.pid);
1502 GNUNET_PEER_resolve (cp->ppd.pid, id);
1503}
1504
1505
1512const struct GNUNET_PeerIdentity *
1514{
1515 GNUNET_assert (0 != cp->ppd.pid);
1516 return GNUNET_PEER_resolve2 (cp->ppd.pid);
1517}
1518
1519
1527void
1529 struct GNUNET_TIME_Absolute block_time)
1530{
1531 struct GNUNET_MQ_Envelope *env;
1532 struct MigrationStopMessage *msm;
1533
1534 if (cp->last_migration_block.abs_value_us > block_time.abs_value_us)
1535 {
1537 "Migration already blocked for another %s\n",
1540 (cp->
1541 last_migration_block), GNUNET_YES));
1542 return; /* already blocked */
1543 }
1544 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Asking to stop migration for %s\n",
1547 GNUNET_YES));
1548 cp->last_migration_block = block_time;
1549 env = GNUNET_MQ_msg (msm,
1551 msm->reserved = htonl (0);
1552 msm->duration
1554 (cp->last_migration_block));
1556 gettext_noop ("# migration stop messages sent"),
1557 1,
1558 GNUNET_NO);
1561 UINT32_MAX,
1562 env);
1563}
1564
1565
1575void
1577 uint64_t pref)
1578{
1579 cp->inc_preference += pref;
1580}
1581
1582
1588static void
1589cron_flush_respect (void *cls)
1590{
1591 fr_task = NULL;
1594 NULL);
1598 NULL);
1599}
1600
1601
1605void
1607{
1611 &cron_flush_respect, NULL);
1612}
1613
1614
1618void
1620{
1623 NULL);
1625 fr_task = NULL;
1627 cp_map = NULL;
1629}
1630
1631
1640static int
1641clean_local_client (void *cls,
1642 const struct GNUNET_PeerIdentity *key,
1643 void *value)
1644{
1645 const struct GSF_LocalClient *lc = cls;
1646 struct GSF_ConnectedPeer *cp = value;
1647 unsigned int i;
1648
1649 for (i = 0; i < CS2P_SUCCESS_LIST_SIZE; i++)
1650 if (cp->ppd.last_client_replies[i] == lc)
1651 cp->ppd.last_client_replies[i] = NULL;
1652 return GNUNET_YES;
1653}
1654
1655
1662void
1664{
1665 if (NULL == cp_map)
1666 return; /* already cleaned up */
1669 (void *) lc);
1670}
1671
1672
1673/* end of gnunet-service-fs_cp.c */
struct GNUNET_GETOPT_CommandLineOption options[]
Definition 002.c:5
struct GNUNET_MQ_Envelope * env
Definition 005.c:1
#define gettext_noop(String)
Definition gettext.h:74
static int ret
Final status code.
Definition gnunet-arm.c:93
static void record(void *cls, size_t data_size, const void *data)
Process recorded audio data.
static char * data
The data to insert into the dht.
struct GNUNET_HashCode key
The key used in the DHT.
static struct GNUNET_TIME_Relative expiration
User supplied expiration value.
static struct GNUNET_TIME_Relative ttl
Current record $TTL to use.
static char * value
Value of the record to add/remove.
static uint32_t type
Type string converted to DNS type value.
static struct GNUNET_PeerIdentity my_identity
Identity of this peer.
int GSF_enable_randomized_delays
Are we introducing randomized delays for better anonymity?
struct GNUNET_PILS_Handle * GSF_pils
Pointer to handle of the pils service (points to NULL until we've connected to it).
int GSF_test_get_load_too_high_(uint32_t priority)
Test if the DATABASE (GET) load on this peer is too high to even consider processing the query at all...
struct GNUNET_STATISTICS_Handle * GSF_stats
Handle for reporting statistics.
double GSF_current_priorities
Typical priorities we're seeing from other peers right now.
unsigned int GSF_cover_query_count
How many query messages have we received 'recently' that have not yet been claimed as cover traffic?
const struct GNUNET_CONFIGURATION_Handle * GSF_cfg
Our configuration.
void GSF_consider_forwarding(void *cls, struct GSF_PendingRequest *pr, enum GNUNET_BLOCK_ReplyEvaluationResult result)
Function to be called after we're done processing replies from the local lookup.
unsigned int GSF_cover_content_count
How many content messages have we received 'recently' that have not yet been claimed as cover traffic...
struct GNUNET_TIME_Relative GSF_avg_latency
Running average of the observed latency to other peers (round trip).
struct GNUNET_LOAD_Value * GSF_rt_entry_lifetime
How long do requests typically stay in the routing table?
#define GET_MESSAGE_BIT_TRANSMIT_TO
The peer identity of a peer that had claimed to have the content previously is included (can be used ...
#define TTL_DECREMENT
By which amount do we decrement the TTL for simple forwarding / indirection of the query; in milli-se...
#define GET_MESSAGE_BIT_RETURN_TO
The peer identity of a peer waiting for the reply is included (used if the response should be transmi...
static void handle_p2p_reply(void *cls, enum GNUNET_BLOCK_ReplyEvaluationResult eval, struct GSF_PendingRequest *pr, uint32_t reply_anonymity_level, struct GNUNET_TIME_Absolute expiration, struct GNUNET_TIME_Absolute last_transmission, enum GNUNET_BLOCK_Type type, const void *data, size_t data_len)
Handle a reply to a pending request.
void GSF_connected_peer_done_()
Shutdown peer management subsystem.
static int flush_respect(void *cls, const struct GNUNET_PeerIdentity *key, void *value)
Write peer-respect information to a file - flush the buffer entry!
#define N
void GSF_peer_disconnect_handler(void *cls, const struct GNUNET_PeerIdentity *peer, void *internal_cls)
A peer disconnected from us.
void handle_p2p_get(void *cls, const struct GetMessage *gm)
Handle P2P "QUERY" message.
static struct GNUNET_TIME_Relative get_randomized_delay()
Get the randomized delay a response should be subjected to.
static int test_exist_cb(void *cls, const struct GNUNET_HashCode *hc, void *value)
Test if the query already exists.
static void cron_flush_respect(void *cls)
Call this method periodically to flush respect information to disk.
struct GSF_PeerPerformanceData * GSF_get_peer_performance_data_(struct GSF_ConnectedPeer *cp)
Return the performance data record for the given peer.
static void free_pending_request(struct PeerRequest *peerreq)
Free resources associated with the given peer request.
static void transmit_delayed_now(void *cls)
The artificial delay is over, transmit the message now.
void GSF_update_peer_latency_(const struct GNUNET_PeerIdentity *id, struct GNUNET_TIME_Relative latency)
Update the latency information kept for the given peer.
static void revive_migration(void *cls)
It may be time to re-start migrating content to this peer.
static int clean_local_client(void *cls, const struct GNUNET_PeerIdentity *key, void *value)
Iterator to remove references to LC entry.
static void peer_request_destroy(void *cls)
Free the given request.
static struct GNUNET_PEERSTORE_Handle * peerstore
Handle to peerstore service.
static void peer_respect_cb(void *cls, const struct GNUNET_PEERSTORE_Record *record, const char *emsg)
Function called by PEERSTORE with peer respect record.
static void peer_transmit(struct GSF_ConnectedPeer *cp)
Core is ready to transmit to a peer, get the message.
void GSF_iterate_connected_peers_(GSF_ConnectedPeerIterator it, void *it_cls)
Iterate over all connected peers.
void GSF_connected_peer_get_identity_(const struct GSF_ConnectedPeer *cp, struct GNUNET_PeerIdentity *id)
Obtain the identity of a connected peer.
void GSF_handle_local_client_disconnect_(const struct GSF_LocalClient *lc)
Notification that a local client disconnected.
const struct GNUNET_PeerIdentity * GSF_connected_peer_get_identity2_(const struct GSF_ConnectedPeer *cp)
Obtain the identity of a connected peer.
static int consider_peer_for_forwarding(void *cls, const struct GNUNET_HashCode *key, struct GSF_PendingRequest *pr)
Function called for each pending request whenever a new peer connects, giving us a chance to decide a...
static int cancel_pending_request(void *cls, const struct GNUNET_HashCode *query, void *value)
Cancel all requests associated with the peer.
static struct GNUNET_CONTAINER_MultiPeerMap * cp_map
Map from peer identities to struct GSF_ConnectPeer entries.
struct GSF_ConnectedPeer * GSF_peer_get_(const struct GNUNET_PeerIdentity *peer)
Get a handle for a connected peer.
void GSF_connected_peer_init_()
Initialize peer management subsystem.
void * GSF_peer_connect_handler(void *cls, const struct GNUNET_PeerIdentity *peer, struct GNUNET_MQ_Handle *mq, enum GNUNET_CORE_PeerClass class)
A peer connected to us.
void GSF_peer_update_responder_peer_(struct GSF_ConnectedPeer *cp, const struct GSF_ConnectedPeer *initiator_peer)
Report on receiving a reply in response to an initiating peer.
static int call_iterator(void *cls, const struct GNUNET_PeerIdentity *key, void *value)
Function that calls the callback for each peer.
void handle_p2p_migration_stop(void *cls, const struct MigrationStopMessage *msm)
Handle P2P GNUNET_MESSAGE_TYPE_FS_MIGRATION_STOP message.
void GSF_peer_update_responder_client_(struct GSF_ConnectedPeer *cp, struct GSF_LocalClient *initiator_client)
Report on receiving a reply in response to an initiating client.
static void schedule_transmission(struct GSF_PeerTransmitHandle *pth)
If ready (bandwidth reserved), try to schedule transmission via core for the given handle.
#define RESPECT_FLUSH_FREQ
How often do we flush respect values to disk?
void GSF_peer_transmit_(struct GSF_ConnectedPeer *cp, int is_query, uint32_t priority, struct GNUNET_MQ_Envelope *env)
Transmit a message to the given peer as soon as possible.
static struct GNUNET_SCHEDULER_Task * fr_task
Task used to flush respect values to disk.
#define RUNAVG_DELAY_N
Ratio for moving average delay calculation.
static int change_peer_respect(struct GSF_ConnectedPeer *cp, int value)
Increase the peer's respect by a value.
void GSF_block_peer_migration_(struct GSF_ConnectedPeer *cp, struct GNUNET_TIME_Absolute block_time)
Ask a peer to stop migrating data to us until the given point in time.
static int32_t bound_priority(uint32_t prio_in, struct GSF_ConnectedPeer *cp)
We've received a request with the specified priority.
void GSF_peer_update_performance_(struct GSF_ConnectedPeer *cp, struct GNUNET_TIME_Absolute request_time, uint32_t request_priority)
Report on receiving a reply; update the performance record of the given peer.
void GSF_connected_peer_change_preference_(struct GSF_ConnectedPeer *cp, uint64_t pref)
Notify core about a preference we have for the given peer (to allocate more resources towards it).
static int32_t bound_ttl(int32_t ttl_in, uint32_t prio)
The priority level imposes a bound on the maximum value for the ttl that can be requested.
void(* GSF_ConnectedPeerIterator)(void *cls, const struct GNUNET_PeerIdentity *peer, struct GSF_ConnectedPeer *cp, const struct GSF_PeerPerformanceData *ppd)
Signature of function called on a connected peer.
#define P2P_SUCCESS_LIST_SIZE
Length of the P2P success tracker.
#define CS2P_SUCCESS_LIST_SIZE
Length of the CS-2-P success tracker.
#define MAX_QUEUE_PER_PEER
Maximum number of outgoing messages we queue per peer.
void GSF_plan_notify_peer_disconnect_(const struct GSF_ConnectedPeer *cp)
Notify the plan about a peer being no longer available; destroy all entries associated with this peer...
void GSF_plan_add_(struct GSF_ConnectedPeer *cp, struct GSF_PendingRequest *pr)
Create a new query plan entry.
int GSF_pending_request_test_target_(struct GSF_PendingRequest *pr, const struct GNUNET_PeerIdentity *target)
Is the given target a legitimate peer for forwarding the given request?
struct GSF_PendingRequest * GSF_pending_request_create_(enum GSF_PendingRequestOptions options, enum GNUNET_BLOCK_Type type, const struct GNUNET_HashCode *query, const struct GNUNET_PeerIdentity *target, const char *bf_data, size_t bf_size, uint32_t anonymity_level, uint32_t priority, int32_t ttl, GNUNET_PEER_Id sender_pid, GNUNET_PEER_Id origin_pid, const struct GNUNET_HashCode *replies_seen, unsigned int replies_seen_count, GSF_PendingRequestReplyHandler rh, void *rh_cls)
Create a new pending request.
void GSF_iterate_pending_requests_(GSF_PendingRequestIterator it, void *cls)
Iterate over all pending requests.
void GSF_local_lookup_(struct GSF_PendingRequest *pr, GSF_LocalLookupContinuation cont, void *cont_cls)
Look up the request in the local datastore.
struct GSF_PendingRequestData * GSF_pending_request_get_data_(struct GSF_PendingRequest *pr)
Obtain the public data associated with a pending request.
int GSF_pending_request_test_active_(struct GSF_PendingRequest *pr)
Check if the given request is still active.
void GSF_pending_request_cancel_(struct GSF_PendingRequest *pr, int full_cleanup)
Explicitly cancel a pending request.
GSF_PendingRequestOptions
Options for pending requests (bits to be ORed).
@ GSF_PRO_DEFAULTS
No special options (P2P-default).
@ GSF_PRO_FORWARD_ONLY
Request must only be forwarded (no routing)
void GSF_push_stop_(struct GSF_ConnectedPeer *peer)
A peer disconnected from us.
void GSF_push_start_(struct GSF_ConnectedPeer *peer)
A peer connected to us.
static struct GNUNET_PeerIdentity pid
Identity of the peer we transmit to / connect to.
static struct GNUNET_TRANSPORT_PluginMonitor * pm
Handle if we are monitoring plugin session activity.
GNUNET_BLOCK_Type
WARNING: This header is generated! In order to add DHT block types, you must register them in GANA,...
@ GNUNET_BLOCK_TYPE_ANY
Identifier for any block.
const struct GNUNET_PeerIdentity * GNUNET_PILS_get_identity(const struct GNUNET_PILS_Handle *handle)
Return the current peer identity of a given handle.
Definition pils_api.c:727
#define GNUNET_MAX_MESSAGE_SIZE
Largest supported message (to be precise, one byte more than the largest possible message,...
GNUNET_BLOCK_ReplyEvaluationResult
Possible ways for how a block may relate to a query.
@ GNUNET_BLOCK_REPLY_OK_LAST
Last possible valid result.
#define GNUNET_CONSTANTS_MAX_CORK_DELAY
How long do we delay messages to get larger packet sizes (CORKing)?
GNUNET_CORE_PeerClass
The peer class gives a hint about the capabilities of a peer.
uint32_t GNUNET_CRYPTO_random_u32(enum GNUNET_CRYPTO_Quality mode, uint32_t i)
Produce a random value.
@ GNUNET_CRYPTO_QUALITY_WEAK
No good quality of the operation is needed (i.e., random numbers can be pseudo-random).
#define GNUNET_CONTAINER_DLL_remove(head, tail, element)
Remove an element from a DLL.
#define GNUNET_CONTAINER_DLL_insert_after(head, tail, other, element)
Insert an element into a DLL after the given other element.
#define GNUNET_CONTAINER_DLL_insert(head, tail, element)
Insert an element at the head of a DLL.
void * GNUNET_CONTAINER_multipeermap_get(const struct GNUNET_CONTAINER_MultiPeerMap *map, const struct GNUNET_PeerIdentity *key)
Given a key find a value in the map matching the key.
int GNUNET_CONTAINER_multihashmap_iterate(struct GNUNET_CONTAINER_MultiHashMap *map, GNUNET_CONTAINER_MultiHashMapIteratorCallback it, void *it_cls)
Iterate over all entries in the map.
void GNUNET_CONTAINER_multipeermap_destroy(struct GNUNET_CONTAINER_MultiPeerMap *map)
Destroy a hash map.
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multihashmap_remove(struct GNUNET_CONTAINER_MultiHashMap *map, const struct GNUNET_HashCode *key, const void *value)
Remove the given key-value pair from the map.
int GNUNET_CONTAINER_multipeermap_iterate(struct GNUNET_CONTAINER_MultiPeerMap *map, GNUNET_CONTAINER_PeerMapIterator it, void *it_cls)
Iterate over all entries in the map.
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multihashmap_put(struct GNUNET_CONTAINER_MultiHashMap *map, const struct GNUNET_HashCode *key, void *value, enum GNUNET_CONTAINER_MultiHashMapOption opt)
Store a key-value pair in the map.
void GNUNET_CONTAINER_multihashmap_destroy(struct GNUNET_CONTAINER_MultiHashMap *map)
Destroy a hash map.
struct GNUNET_CONTAINER_MultiHashMap * GNUNET_CONTAINER_multihashmap_create(unsigned int len, int do_not_copy_keys)
Create a multi hash map.
struct GNUNET_CONTAINER_MultiPeerMap * GNUNET_CONTAINER_multipeermap_create(unsigned int len, int do_not_copy_keys)
Create a multi peer map (hash map for public keys of peers).
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multihashmap_get_multiple(struct GNUNET_CONTAINER_MultiHashMap *map, const struct GNUNET_HashCode *key, GNUNET_CONTAINER_MultiHashMapIteratorCallback it, void *it_cls)
Iterate over all entries in the map that match a particular key.
unsigned int GNUNET_CONTAINER_multipeermap_size(const struct GNUNET_CONTAINER_MultiPeerMap *map)
Get the number of key-value pairs in the map.
int GNUNET_CONTAINER_multipeermap_put(struct GNUNET_CONTAINER_MultiPeerMap *map, const struct GNUNET_PeerIdentity *key, void *value, enum GNUNET_CONTAINER_MultiHashMapOption opt)
Store a key-value pair in the map.
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multipeermap_remove(struct GNUNET_CONTAINER_MultiPeerMap *map, const struct GNUNET_PeerIdentity *key, const void *value)
Remove the given key-value pair from the map.
@ GNUNET_CONTAINER_MULTIHASHMAPOPTION_MULTIPLE
Allow multiple values with the same key.
@ GNUNET_CONTAINER_MULTIHASHMAPOPTION_UNIQUE_ONLY
There must only be one value per key; storing a value should fail if a value under the same key alrea...
#define GNUNET_log(kind,...)
#define GNUNET_memcmp(a, b)
Compare memory in a and b, where both must be of the same pointer type.
#define GNUNET_memcpy(dst, src, n)
Call memcpy() but check for n being 0 first.
uint16_t size
The length of the struct (in bytes, including the length field itself), in big-endian format.
@ GNUNET_SCHEDULER_PRIORITY_HIGH
Run with high priority (important requests).
@ GNUNET_OK
@ GNUNET_YES
@ GNUNET_NO
@ GNUNET_SYSERR
void GNUNET_LOAD_value_set_decline(struct GNUNET_LOAD_Value *load, struct GNUNET_TIME_Relative autodecline)
Change the value by which the load automatically declines.
Definition load.c:142
double GNUNET_LOAD_get_average(struct GNUNET_LOAD_Value *load)
Get the average value given to update so far.
Definition load.c:215
struct GNUNET_LOAD_Value * GNUNET_LOAD_value_init(struct GNUNET_TIME_Relative autodecline)
Create a new load value.
Definition load.c:124
void GNUNET_LOAD_update(struct GNUNET_LOAD_Value *load, uint64_t data)
Update the current load.
Definition load.c:236
double GNUNET_LOAD_get_load(struct GNUNET_LOAD_Value *load)
Get the current load.
Definition load.c:200
#define GNUNET_LOAD_value_free(lv)
Free a load value.
const char * GNUNET_i2s(const struct GNUNET_PeerIdentity *pid)
Convert a peer identity to a string (for printing debug messages).
#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.
const char * GNUNET_h2s(const struct GNUNET_HashCode *hc)
Convert a hash value to a string (for printing debug messages).
@ GNUNET_ERROR_TYPE_DEBUG
@ GNUNET_ERROR_TYPE_INFO
#define GNUNET_new(type)
Allocate a struct or union of the given type.
#define GNUNET_free(ptr)
Wrapper around free.
unsigned int GNUNET_MQ_get_length(struct GNUNET_MQ_Handle *mq)
Obtain the current length of the message queue.
Definition mq.c:293
void GNUNET_MQ_send(struct GNUNET_MQ_Handle *mq, struct GNUNET_MQ_Envelope *ev)
Send a message with the given message queue.
Definition mq.c:305
void GNUNET_MQ_discard(struct GNUNET_MQ_Envelope *mqm)
Discard the message queue message, free all allocated resources.
Definition mq.c:285
#define GNUNET_MQ_msg_extra(mvar, esize, type)
Allocate an envelope, with extra space allocated after the space needed by the message struct.
#define GNUNET_MQ_msg(mvar, type)
Allocate a GNUNET_MQ_Envelope.
void GNUNET_PEER_decrement_rcs(const GNUNET_PEER_Id *ids, unsigned int count)
Decrement multiple RCs of peer identities by one.
Definition peer.c:157
unsigned int GNUNET_PEER_Id
A GNUNET_PEER_Id is simply a shorter version of a "struct GNUNET_PeerIdentifier" that can be used ins...
void GNUNET_PEER_change_rc(GNUNET_PEER_Id id, int delta)
Change the reference counter of an interned PID.
Definition peer.c:192
const struct GNUNET_PeerIdentity * GNUNET_PEER_resolve2(GNUNET_PEER_Id id)
Convert an interned PID to a normal peer identity.
Definition peer.c:234
void GNUNET_PEER_resolve(GNUNET_PEER_Id id, struct GNUNET_PeerIdentity *pid)
Convert an interned PID to a normal peer identity.
Definition peer.c:220
GNUNET_PEER_Id GNUNET_PEER_intern(const struct GNUNET_PeerIdentity *pid)
Intern an peer identity.
Definition peer.c:108
struct GNUNET_PEERSTORE_IterateContext * GNUNET_PEERSTORE_iteration_start(struct GNUNET_PEERSTORE_Handle *h, const char *sub_system, const struct GNUNET_PeerIdentity *peer, const char *key, GNUNET_PEERSTORE_Processor callback, void *callback_cls)
Iterate over peerstore entries.
void GNUNET_PEERSTORE_iteration_next(struct GNUNET_PEERSTORE_IterateContext *ic, uint64_t limit)
Continue an iteration.
void GNUNET_PEERSTORE_disconnect(struct GNUNET_PEERSTORE_Handle *h)
Disconnect from the PEERSTORE service.
struct GNUNET_PEERSTORE_Handle * GNUNET_PEERSTORE_connect(const struct GNUNET_CONFIGURATION_Handle *cfg)
Connect to the PEERSTORE service.
struct GNUNET_PEERSTORE_StoreContext * GNUNET_PEERSTORE_store(struct GNUNET_PEERSTORE_Handle *h, const char *sub_system, const struct GNUNET_PeerIdentity *peer, const char *key, const void *value, size_t size, struct GNUNET_TIME_Absolute expiry, enum GNUNET_PEERSTORE_StoreOption options, GNUNET_PEERSTORE_Continuation cont, void *cont_cls)
Store a new entry in the PEERSTORE.
void GNUNET_PEERSTORE_iteration_stop(struct GNUNET_PEERSTORE_IterateContext *ic)
Cancel an iteration.
@ GNUNET_PEERSTORE_STOREOPTION_REPLACE
Delete any previous values for the given key before storing the given value.
#define GNUNET_MESSAGE_TYPE_FS_PUT
P2P response with content or active migration of content.
#define GNUNET_MESSAGE_TYPE_FS_MIGRATION_STOP
Peer asks us to stop migrating content towards it for a while.
void * GNUNET_SCHEDULER_cancel(struct GNUNET_SCHEDULER_Task *task)
Cancel the task with the specified identifier.
Definition scheduler.c:986
struct GNUNET_SCHEDULER_Task * GNUNET_SCHEDULER_add_now(GNUNET_SCHEDULER_TaskCallback task, void *task_cls)
Schedule a new task to be run as soon as possible.
Definition scheduler.c:1310
struct GNUNET_SCHEDULER_Task * GNUNET_SCHEDULER_add_delayed(struct GNUNET_TIME_Relative delay, GNUNET_SCHEDULER_TaskCallback task, void *task_cls)
Schedule a new task to be run with a specified delay.
Definition scheduler.c:1283
struct GNUNET_SCHEDULER_Task * GNUNET_SCHEDULER_add_delayed_with_priority(struct GNUNET_TIME_Relative delay, enum GNUNET_SCHEDULER_Priority priority, GNUNET_SCHEDULER_TaskCallback task, void *task_cls)
Schedule a new task to be run with a specified delay.
Definition scheduler.c:1213
struct GNUNET_SCHEDULER_Task * GNUNET_SCHEDULER_add_with_priority(enum GNUNET_SCHEDULER_Priority prio, GNUNET_SCHEDULER_TaskCallback task, void *task_cls)
Schedule a new task to be run with a specified priority.
Definition scheduler.c:1237
void GNUNET_STATISTICS_set(struct GNUNET_STATISTICS_Handle *handle, const char *name, uint64_t value, int make_persistent)
Set statistic value for the peer.
void GNUNET_STATISTICS_update(struct GNUNET_STATISTICS_Handle *handle, const char *name, int64_t delta, int make_persistent)
Set statistic value for the peer.
struct GNUNET_TIME_Relative GNUNET_TIME_relative_ntoh(struct GNUNET_TIME_RelativeNBO a)
Convert relative time from network byte order.
Definition time.c:626
struct GNUNET_TIME_Relative GNUNET_TIME_absolute_get_duration(struct GNUNET_TIME_Absolute whence)
Get the duration of an operation as the difference of the current time and the given start time "henc...
Definition time.c:438
struct GNUNET_TIME_Relative GNUNET_TIME_absolute_get_remaining(struct GNUNET_TIME_Absolute future)
Given a timestamp in the future, how much time remains until then?
Definition time.c:406
const char * GNUNET_STRINGS_relative_time_to_string(struct GNUNET_TIME_Relative delta, int do_round)
Give relative time in human-readable fancy format.
Definition strings.c:604
#define GNUNET_TIME_UNIT_MILLISECONDS
One millisecond.
struct GNUNET_TIME_Absolute GNUNET_TIME_absolute_get(void)
Get the current time.
Definition time.c:111
struct GNUNET_TIME_Absolute GNUNET_TIME_relative_to_absolute(struct GNUNET_TIME_Relative rel)
Convert relative time to an absolute time in the future.
Definition time.c:316
#define GNUNET_TIME_UNIT_ZERO
Relative time zero.
struct GNUNET_TIME_RelativeNBO GNUNET_TIME_relative_hton(struct GNUNET_TIME_Relative a)
Convert relative time to network byte order.
Definition time.c:616
struct GNUNET_TIME_AbsoluteNBO GNUNET_TIME_absolute_hton(struct GNUNET_TIME_Absolute a)
Convert absolute time to network byte order.
Definition time.c:636
#define GNUNET_TIME_UNIT_FOREVER_ABS
Constant used to specify "forever".
#define _(String)
GNU gettext support macro.
Definition platform.h:179
static struct GNUNET_MQ_Handle * mq
Our connection to the resolver service, created on-demand, but then persists until error or shutdown.
Internal representation of the hash map.
Internal representation of the hash map.
A 512-bit hashcode.
Handle to a message queue.
Definition mq.c:87
Handle to the PEERSTORE service.
Context for a iterate request.
The identity of the host (wraps the signing key of the peer).
Entry in list of pending tasks.
Definition scheduler.c:141
Time for absolute times used by GNUnet, in microseconds.
uint64_t abs_value_us
The actual value.
Time for relative time used by GNUnet, in microseconds.
uint64_t rel_value_us
The actual value.
unsigned int last_p2p_replies_woff
Which offset in last_p2p_replies will be updated next? (we go round-robin).
struct GNUNET_TIME_Absolute last_migration_block
Time until when we blocked this peer from migrating data to us.
struct GSF_DelayedHandle * delayed_tail
Messages (replies, queries, content migration) we would like to send to this peer in the near future.
unsigned int last_client_replies_woff
Which offset in last_client_replies will be updated next? (we go round-robin).
unsigned int delay_queue_size
Number of entries in delayed_head DLL.
struct GSF_DelayedHandle * delayed_head
Messages (replies, queries, content migration) we would like to send to this peer in the near future.
struct GNUNET_PEERSTORE_IterateContext * respect_iterate_req
Handle to the PEERSTORE iterate request for peer respect value.
struct GSF_PeerTransmitHandle * pth_head
Messages (replies, queries, content migration) we would like to send to this peer in the near future.
struct GSF_PeerPerformanceData ppd
Performance data for this peer.
struct GNUNET_CONTAINER_MultiHashMap * request_map
Active requests from this neighbour, map of query to struct PeerRequest.
struct GNUNET_MQ_Handle * mq
Handle for an active request for transmission to this peer.
unsigned int last_request_times_off
Current offset into last_request_times ring buffer.
struct GNUNET_SCHEDULER_Task * rc_delay_task
Task scheduled if we need to retry bandwidth reservation later.
struct GNUNET_SCHEDULER_Task * mig_revive_task
Task scheduled to revive migration to this peer.
uint32_t disk_respect
Respect rating for this peer on disk.
struct GSF_PeerTransmitHandle * pth_tail
Messages (replies, queries, content migration) we would like to send to this peer in the near future.
uint64_t inc_preference
Increase in traffic preference still to be submitted to the core service for this peer.
Handle for an entry in our delay list.
struct GSF_DelayedHandle * next
Kept in a doubly-linked list.
struct GSF_DelayedHandle * prev
Kept in a doubly-linked list.
struct GNUNET_SCHEDULER_Task * delay_task
Task for the delay.
size_t msize
Size of the message.
struct GSF_ConnectedPeer * cp
Peer this transmission belongs to.
struct GNUNET_MQ_Envelope * env
Envelope of the message that was delayed.
A local client.
Performance data kept for a peer.
double avg_priority
Average priority of successful replies.
struct GNUNET_TIME_Relative avg_reply_delay
Average delay between sending the peer a request and getting a reply (only calculated over the reques...
unsigned int pending_replies
Number of pending replies (queries are not counted)
struct GSF_LocalClient * last_client_replies[8]
List of the last clients for which this peer successfully answered a query.
struct GNUNET_LOAD_Value * transmission_delay
How long does it typically take for us to transmit a message to this peer? (delay between the request...
struct GNUNET_TIME_Absolute migration_blocked_until
Point in time until which this peer does not want us to migrate content to it.
const struct GNUNET_PeerIdentity * peer
The peer's identity (pointer).
unsigned int pending_queries
Number of pending queries (replies are not counted)
GNUNET_PEER_Id pid
The peer's identity (interned version).
uint32_t respect
Respect rating for this peer.
struct GNUNET_TIME_Absolute last_request_times[24]
Transmission times for the last MAX_QUEUE_PER_PEER requests for this peer.
GNUNET_PEER_Id last_p2p_replies[8]
List of the last PIDs for which this peer successfully answered a query; We use 0 to indicate no succ...
Handle to cancel a transmission request.
int is_query
GNUNET_YES if this is a query, GNUNET_NO for content.
struct GSF_PeerTransmitHandle * next
Kept in a doubly-linked list.
uint32_t priority
Priority of this request.
struct GNUNET_TIME_Absolute transmission_request_start_time
Time when this transmission request was issued.
struct GSF_PeerTransmitHandle * prev
Kept in a doubly-linked list.
struct GSF_ConnectedPeer * cp
Peer this request targets.
struct GNUNET_MQ_Envelope * env
Envelope with the actual message.
Public data (in the sense of not encapsulated within 'gnunet-service-fs_pr', not in the sense of netw...
int has_started
Has this request been started yet (local/p2p operations)? Or are we still constructing it?
struct GNUNET_TIME_Absolute ttl
Current TTL for the request.
enum GNUNET_BLOCK_Type type
Type of the requested block.
struct GNUNET_HashCode query
Primary query hash for this request.
uint32_t priority
Priority that this request (still) has for us.
An active request.
Message to the datastore service asking about specific content.
Definition datastore.h:141
uint32_t type
Desired content type.
Definition datastore.h:150
struct GNUNET_MessageHeader header
Type is GNUNET_MESSAGE_TYPE_DATASTORE_GET.
Definition datastore.h:145
uint32_t priority
How important is this request (network byte order)
struct GNUNET_HashCode query
Hashcodes of the file(s) we're looking for.
int32_t ttl
Relative time to live in MILLISECONDS (network byte order)
uint32_t hash_bitmap
Which of the optional hash codes are present at the end of the message? See GET_MESSAGE_BIT_xx consta...
Closure for call_iterator().
void * it_cls
Closure for it.
GSF_ConnectedPeerIterator it
Function to call on each entry.
Message send by a peer that wants to be excluded from migration for a while.
struct GNUNET_TIME_RelativeNBO duration
How long should the block last?
uint32_t reserved
Always zero.
Information per peer and request.
struct GSF_ConnectedPeer * cp
Which specific peer issued this request?
struct GSF_PendingRequest * pr
Handle to generic request (generic: from peer or local client).
struct GNUNET_SCHEDULER_Task * kill_task
Task for asynchronous stopping of this request.
Response from FS service with a result for a previous FS search.
Definition fs.h:330
Closure for test_exist_cb().
int finished
Set to GNUNET_YES if we are done handling the query.
int32_t ttl
Relative TTL of the incoming request.
int32_t priority
Priority of the incoming request.
enum GNUNET_BLOCK_Type type
Type of the incoming request.

◆ REPLY_TIMEOUT

#define REPLY_TIMEOUT
Value:

After how long do we discard a reply?

Definition at line 54 of file gnunet-service-fs_cp.c.

◆ INSANE_STATISTICS

#define INSANE_STATISTICS   GNUNET_NO

Collect an instance number of statistics? May cause excessive IPC.

Definition at line 60 of file gnunet-service-fs_cp.c.

◆ N

#define N   ((double) 128.0)

Function Documentation

◆ GSF_update_peer_latency_()

void GSF_update_peer_latency_ ( const struct GNUNET_PeerIdentity id,
struct GNUNET_TIME_Relative  latency 
)

Update the latency information kept for the given peer.

Parameters
idpeer record to update
latencycurrent latency value

Definition at line 288 of file gnunet-service-fs_cp.c.

290{
291 struct GSF_ConnectedPeer *cp;
292
293 cp = GSF_peer_get_ (id);
294 if (NULL == cp)
295 return; /* we're not yet connected at the core level, ignore */
297 latency);
298}

References GNUNET_LOAD_value_set_decline(), GSF_peer_get_(), GSF_ConnectedPeer::ppd, and GSF_PeerPerformanceData::transmission_delay.

Here is the call graph for this function:

◆ GSF_get_peer_performance_data_()

struct GSF_PeerPerformanceData * GSF_get_peer_performance_data_ ( struct GSF_ConnectedPeer cp)

Return the performance data record for the given peer.

Parameters
cppeer to query
Returns
performance data record for the peer

Definition at line 308 of file gnunet-service-fs_cp.c.

309{
310 return &cp->ppd;
311}

References GSF_ConnectedPeer::ppd.

Referenced by handle_p2p_put(), put_migration_continuation(), score_content(), and transmit_content().

Here is the caller graph for this function:

◆ peer_transmit()

static void peer_transmit ( struct GSF_ConnectedPeer cp)
static

Core is ready to transmit to a peer, get the message.

Parameters
cpwhich peer to send a message to

Definition at line 349 of file gnunet-service-fs_cp.c.

350{
351 struct GSF_PeerTransmitHandle *pth = cp->pth_head;
352 struct GSF_PeerTransmitHandle *pos;
353
354 if (NULL == pth)
355 return;
357 cp->pth_tail,
358 pth);
359 if (GNUNET_YES == pth->is_query)
360 {
365 }
366 else if (GNUNET_NO == pth->is_query)
367 {
369 }
374 pth->env);
375 GNUNET_free (pth);
376 if (NULL != (pos = cp->pth_head))
377 {
378 GNUNET_assert (pos != pth);
380 }
381}

References GSF_PeerTransmitHandle::cp, GSF_PeerTransmitHandle::env, GNUNET_assert, GNUNET_CONTAINER_DLL_remove, GNUNET_free, GNUNET_LOAD_update(), GNUNET_MQ_send(), GNUNET_NO, GNUNET_TIME_absolute_get(), GNUNET_TIME_absolute_get_duration(), GNUNET_YES, GSF_PeerTransmitHandle::is_query, GSF_PeerPerformanceData::last_request_times, GSF_ConnectedPeer::last_request_times_off, MAX_QUEUE_PER_PEER, GSF_ConnectedPeer::mq, GSF_PeerPerformanceData::pending_queries, GSF_PeerPerformanceData::pending_replies, GSF_ConnectedPeer::ppd, GSF_ConnectedPeer::pth_head, GSF_ConnectedPeer::pth_tail, GNUNET_TIME_Relative::rel_value_us, schedule_transmission(), GSF_PeerPerformanceData::transmission_delay, and GSF_PeerTransmitHandle::transmission_request_start_time.

Referenced by schedule_transmission().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ schedule_transmission()

static void schedule_transmission ( struct GSF_PeerTransmitHandle pth)
static

If ready (bandwidth reserved), try to schedule transmission via core for the given handle.

Parameters
pthtransmission handle to schedule

Definition at line 330 of file gnunet-service-fs_cp.c.

331{
332 struct GSF_ConnectedPeer *cp;
333 struct GNUNET_PeerIdentity target;
334
335 cp = pth->cp;
336 GNUNET_assert (0 != cp->ppd.pid);
337 GNUNET_PEER_resolve (cp->ppd.pid, &target);
338
339 peer_transmit (cp);
340}

References GSF_PeerTransmitHandle::cp, GNUNET_assert, GNUNET_PEER_resolve(), peer_transmit(), GSF_PeerPerformanceData::pid, and GSF_ConnectedPeer::ppd.

Referenced by GSF_peer_transmit_(), and peer_transmit().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ peer_respect_cb()

static void peer_respect_cb ( void *  cls,
const struct GNUNET_PEERSTORE_Record record,
const char *  emsg 
)
static

Function called by PEERSTORE with peer respect record.

Parameters
clshandle to connected peer entry
recordpeerstore record information
emsgerror message, or NULL if no errors

Definition at line 392 of file gnunet-service-fs_cp.c.

395{
396 struct GSF_ConnectedPeer *cp = cls;
397
399 if ((NULL == record) && (NULL == emsg))
400 {
401 cp->respect_iterate_req = NULL;
402 return;
403 }
404 if ((NULL != record) &&
405 (sizeof(cp->disk_respect) == record->value_size))
406 {
407 cp->disk_respect = *((uint32_t *) record->value);
408 cp->ppd.respect += *((uint32_t *) record->value);
409 }
410 GSF_push_start_ (cp);
411 if (NULL != record)
412 {
414 cp->respect_iterate_req = NULL;
415 return;
416 }
418}

References GSF_ConnectedPeer::disk_respect, GNUNET_assert, GNUNET_PEERSTORE_iteration_next(), GNUNET_PEERSTORE_iteration_stop(), GSF_push_start_(), GSF_ConnectedPeer::ppd, record(), GSF_PeerPerformanceData::respect, and GSF_ConnectedPeer::respect_iterate_req.

Referenced by GSF_peer_connect_handler().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ consider_peer_for_forwarding()

static int consider_peer_for_forwarding ( void *  cls,
const struct GNUNET_HashCode key,
struct GSF_PendingRequest pr 
)
static

Function called for each pending request whenever a new peer connects, giving us a chance to decide about submitting the existing request to the new peer.

Parameters
clsthe struct GSF_ConnectedPeer of the new peer
keyquery for the request
prhandle to the pending request
Returns
GNUNET_YES to continue to iterate

Definition at line 432 of file gnunet-service-fs_cp.c.

435{
436 struct GSF_ConnectedPeer *cp = cls;
438
439 if (GNUNET_YES !=
441 return GNUNET_YES; /* request is not actually active, skip! */
443 if (GNUNET_YES !=
445 {
447 gettext_noop ("# Loopback routes suppressed"),
448 1,
449 GNUNET_NO);
450 return GNUNET_YES;
451 }
452 GSF_plan_add_ (cp, pr);
453 return GNUNET_YES;
454}

References gettext_noop, GNUNET_NO, GNUNET_STATISTICS_update(), GNUNET_YES, GSF_connected_peer_get_identity_(), GSF_pending_request_test_active_(), GSF_pending_request_test_target_(), GSF_plan_add_(), GSF_stats, and pid.

Referenced by GSF_peer_connect_handler().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_peer_connect_handler()

void * GSF_peer_connect_handler ( void *  cls,
const struct GNUNET_PeerIdentity peer,
struct GNUNET_MQ_Handle mq,
enum GNUNET_CORE_PeerClass  class 
)

A peer connected to us.

Setup the connected peer records.

Parameters
clsNULL
peeridentity of peer that connected
mqmessage queue for talking to peer
classpeer class of connected peer
Returns
internal handle for the peer

Definition at line 458 of file gnunet-service-fs_cp.c.

462{
463 const struct GNUNET_PeerIdentity *my_identity;
464 struct GSF_ConnectedPeer *cp;
465
468
469 if (0 == GNUNET_memcmp (my_identity, peer))
470 return NULL;
472 "Connected to peer %s\n",
473 GNUNET_i2s (peer));
474 cp = GNUNET_new (struct GSF_ConnectedPeer);
475 cp->ppd.pid = GNUNET_PEER_intern (peer);
476 cp->ppd.peer = peer;
477 cp->mq = mq;
479
481 GNUNET_YES);
485 (
486 cp),
487 cp,
490 gettext_noop ("# peers connected"),
492 GNUNET_NO);
495 "fs",
496 peer,
497 "respect",
499 cp);
501 cp);
502 return cp;
503}

References consider_peer_for_forwarding(), cp_map, gettext_noop, GNUNET_assert, GNUNET_break, GNUNET_CONTAINER_multihashmap_create(), GNUNET_CONTAINER_MULTIHASHMAPOPTION_UNIQUE_ONLY, GNUNET_CONTAINER_multipeermap_put(), GNUNET_CONTAINER_multipeermap_size(), GNUNET_ERROR_TYPE_DEBUG, GNUNET_i2s(), GNUNET_LOAD_value_init(), GNUNET_log, GNUNET_memcmp, GNUNET_new, GNUNET_NO, GNUNET_OK, GNUNET_PEER_intern(), GNUNET_PEERSTORE_iteration_start(), GNUNET_PILS_get_identity(), GNUNET_STATISTICS_set(), GNUNET_TIME_UNIT_ZERO, GNUNET_YES, GSF_connected_peer_get_identity2_(), GSF_iterate_pending_requests_(), GSF_pils, GSF_stats, mq, GSF_ConnectedPeer::mq, my_identity, GSF_PeerPerformanceData::peer, peer_respect_cb(), peerstore, GSF_PeerPerformanceData::pid, GSF_ConnectedPeer::ppd, GSF_ConnectedPeer::request_map, GSF_ConnectedPeer::respect_iterate_req, and GSF_PeerPerformanceData::transmission_delay.

Referenced by main_init().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ revive_migration()

static void revive_migration ( void *  cls)
static

It may be time to re-start migrating content to this peer.

Check, and if so, restart migration.

Parameters
clsthe struct GSF_ConnectedPeer

Definition at line 513 of file gnunet-service-fs_cp.c.

514{
515 struct GSF_ConnectedPeer *cp = cls;
516 struct GNUNET_TIME_Relative bt;
517
518 cp->mig_revive_task = NULL;
520 if (0 != bt.rel_value_us)
521 {
522 /* still time left... */
523 cp->mig_revive_task =
525 return;
526 }
527 GSF_push_start_ (cp);
528}

References GNUNET_SCHEDULER_add_delayed(), GNUNET_TIME_absolute_get_remaining(), GSF_push_start_(), GSF_ConnectedPeer::mig_revive_task, GSF_PeerPerformanceData::migration_blocked_until, GSF_ConnectedPeer::ppd, GNUNET_TIME_Relative::rel_value_us, and revive_migration().

Referenced by handle_p2p_migration_stop(), and revive_migration().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_peer_get_()

struct GSF_ConnectedPeer * GSF_peer_get_ ( const struct GNUNET_PeerIdentity peer)

Get a handle for a connected peer.

Parameters
peerpeer's identity
Returns
NULL if this peer is not currently connected

Definition at line 532 of file gnunet-service-fs_cp.c.

533{
534 if (NULL == cp_map)
535 return NULL;
537}

References cp_map, and GNUNET_CONTAINER_multipeermap_get().

Referenced by GSF_update_peer_latency_(), handle_p2p_get(), and put_migration_continuation().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ handle_p2p_migration_stop()

void handle_p2p_migration_stop ( void *  cls,
const struct MigrationStopMessage msm 
)

Handle P2P GNUNET_MESSAGE_TYPE_FS_MIGRATION_STOP message.

Parameters
clsclosure, the struct GSF_ConnectedPeer
msmthe actual message

Definition at line 547 of file gnunet-service-fs_cp.c.

549{
550 struct GSF_ConnectedPeer *cp = cls;
551 struct GNUNET_TIME_Relative bt;
552
554 gettext_noop ("# migration stop messages received"),
555 1, GNUNET_NO);
558 _ ("Migration of content to peer `%s' blocked for %s\n"),
559 GNUNET_i2s (cp->ppd.peer),
562 if ((NULL == cp->mig_revive_task) &&
563 (NULL == cp->respect_iterate_req))
564 {
565 GSF_push_stop_ (cp);
566 cp->mig_revive_task =
568 &revive_migration, cp);
569 }
570}

References _, MigrationStopMessage::duration, gettext_noop, GNUNET_ERROR_TYPE_INFO, GNUNET_i2s(), GNUNET_log, GNUNET_NO, GNUNET_SCHEDULER_add_delayed(), GNUNET_STATISTICS_update(), GNUNET_STRINGS_relative_time_to_string(), GNUNET_TIME_relative_ntoh(), GNUNET_TIME_relative_to_absolute(), GNUNET_YES, GSF_push_stop_(), GSF_stats, GSF_ConnectedPeer::mig_revive_task, GSF_PeerPerformanceData::migration_blocked_until, GSF_PeerPerformanceData::peer, GSF_ConnectedPeer::ppd, GSF_ConnectedPeer::respect_iterate_req, and revive_migration().

Here is the call graph for this function:

◆ free_pending_request()

static void free_pending_request ( struct PeerRequest peerreq)
static

Free resources associated with the given peer request.

Parameters
peerreqrequest to free

Definition at line 579 of file gnunet-service-fs_cp.c.

580{
581 struct GSF_ConnectedPeer *cp = peerreq->cp;
582 struct GSF_PendingRequestData *prd;
583
584 prd = GSF_pending_request_get_data_ (peerreq->pr);
585 if (NULL != peerreq->kill_task)
586 {
588 peerreq->kill_task = NULL;
589 }
591 gettext_noop ("# P2P searches active"),
592 -1,
593 GNUNET_NO);
596 &prd->query,
597 peerreq));
598 GNUNET_free (peerreq);
599}

References PeerRequest::cp, gettext_noop, GNUNET_break, GNUNET_CONTAINER_multihashmap_remove(), GNUNET_free, GNUNET_NO, GNUNET_SCHEDULER_cancel(), GNUNET_STATISTICS_update(), GNUNET_YES, GSF_pending_request_get_data_(), GSF_stats, PeerRequest::kill_task, PeerRequest::pr, GSF_PendingRequestData::query, and GSF_ConnectedPeer::request_map.

Referenced by cancel_pending_request(), handle_p2p_reply(), and test_exist_cb().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ cancel_pending_request()

static int cancel_pending_request ( void *  cls,
const struct GNUNET_HashCode query,
void *  value 
)
static

Cancel all requests associated with the peer.

Parameters
clsunused
queryhash code of the request
valuethe struct GSF_PendingRequest
Returns
GNUNET_YES (continue to iterate)

Definition at line 611 of file gnunet-service-fs_cp.c.

614{
615 struct PeerRequest *peerreq = value;
616 struct GSF_PendingRequest *pr = peerreq->pr;
617
618 free_pending_request (peerreq);
620 GNUNET_NO);
621 return GNUNET_OK;
622}

References free_pending_request(), GNUNET_NO, GNUNET_OK, GSF_pending_request_cancel_(), PeerRequest::pr, and value.

Referenced by GSF_peer_disconnect_handler(), and peer_request_destroy().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ peer_request_destroy()

static void peer_request_destroy ( void *  cls)
static

Free the given request.

Parameters
clsthe request to free

Definition at line 631 of file gnunet-service-fs_cp.c.

632{
633 struct PeerRequest *peerreq = cls;
634 struct GSF_PendingRequest *pr = peerreq->pr;
635 struct GSF_PendingRequestData *prd;
636
637 peerreq->kill_task = NULL;
640 &prd->query,
641 peerreq);
642}

References cancel_pending_request(), GSF_pending_request_get_data_(), PeerRequest::kill_task, PeerRequest::pr, and GSF_PendingRequestData::query.

Referenced by handle_p2p_reply().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ transmit_delayed_now()

static void transmit_delayed_now ( void *  cls)
static

The artificial delay is over, transmit the message now.

Parameters
clsthe struct GSF_DelayedHandle with the message

Definition at line 651 of file gnunet-service-fs_cp.c.

652{
653 struct GSF_DelayedHandle *dh = cls;
654 struct GSF_ConnectedPeer *cp = dh->cp;
655
657 cp->delayed_tail,
658 dh);
659 cp->delay_queue_size--;
661 GNUNET_NO,
662 UINT32_MAX,
663 dh->env);
664 GNUNET_free (dh);
665}

References GSF_DelayedHandle::cp, GSF_ConnectedPeer::delay_queue_size, GSF_ConnectedPeer::delayed_head, GSF_ConnectedPeer::delayed_tail, GSF_DelayedHandle::env, GNUNET_CONTAINER_DLL_remove, GNUNET_free, GNUNET_NO, and GSF_peer_transmit_().

Referenced by handle_p2p_reply().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ get_randomized_delay()

static struct GNUNET_TIME_Relative get_randomized_delay ( )
static

Get the randomized delay a response should be subjected to.

Returns
desired delay

Definition at line 674 of file gnunet-service-fs_cp.c.

675{
677
678 ret =
683#if INSANE_STATISTICS
686 ("# artificial delays introduced (ms)"),
687 ret.rel_value_us / 1000LL, GNUNET_NO);
688#endif
689 return ret;
690}

References gettext_noop, GNUNET_CRYPTO_QUALITY_WEAK, GNUNET_CRYPTO_random_u32(), GNUNET_NO, GNUNET_STATISTICS_update(), GNUNET_TIME_relative_multiply(), GNUNET_TIME_UNIT_MILLISECONDS, GSF_avg_latency, GSF_stats, GNUNET_TIME_Relative::rel_value_us, and ret.

Referenced by handle_p2p_reply().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ handle_p2p_reply()

static void handle_p2p_reply ( void *  cls,
enum GNUNET_BLOCK_ReplyEvaluationResult  eval,
struct GSF_PendingRequest pr,
uint32_t  reply_anonymity_level,
struct GNUNET_TIME_Absolute  expiration,
struct GNUNET_TIME_Absolute  last_transmission,
enum GNUNET_BLOCK_Type  type,
const void *  data,
size_t  data_len 
)
static

Handle a reply to a pending request.

Also called if a request expires (then with data == NULL). The handler may be called many times (depending on the request type), but will not be called during or after a call to GSF_pending_request_cancel and will also not be called anymore after a call signalling expiration.

Parameters
clsstruct PeerRequest this is an answer for
evalevaluation of the result
prhandle to the original pending request
reply_anonymity_levelanonymity level for the reply, UINT32_MAX for "unknown"
expirationwhen does data expire?
last_transmissionwhen did we last transmit a request for this block
typetype of the block
dataresponse data, NULL on request expiration
data_lennumber of bytes in data

Definition at line 712 of file gnunet-service-fs_cp.c.

721{
722 struct PeerRequest *peerreq = cls;
723 struct GSF_ConnectedPeer *cp = peerreq->cp;
724 struct GSF_PendingRequestData *prd;
725 struct GNUNET_MQ_Envelope *env;
726 struct PutMessage *pm;
727 size_t msize;
728
729 GNUNET_assert (data_len + sizeof(struct PutMessage) <
731 GNUNET_assert (peerreq->pr == pr);
733 if (NULL == data)
734 {
735 free_pending_request (peerreq);
736 return;
737 }
739 if ( (prd->type != type) &&
740 (GNUNET_BLOCK_TYPE_ANY != prd->type) )
741 {
743 "# replies dropped due to type mismatch",
744 1, GNUNET_NO);
745 return;
746 }
748 "Transmitting result for query `%s' to peer\n",
749 GNUNET_h2s (&prd->query));
751 "# replies received for other peers",
752 1,
753 GNUNET_NO);
754 msize = sizeof(struct PutMessage) + data_len;
755 if (msize >= GNUNET_MAX_MESSAGE_SIZE)
756 {
757 GNUNET_break (0);
758 return;
759 }
760 if ( (UINT32_MAX != reply_anonymity_level) &&
761 (reply_anonymity_level > 1) )
762 {
763 if (reply_anonymity_level - 1 > GSF_cover_content_count)
764 {
766 "# replies dropped due to insufficient cover traffic",
767 1, GNUNET_NO);
768 return;
769 }
770 GSF_cover_content_count -= (reply_anonymity_level - 1);
771 }
772
774 data_len,
776 pm->type = htonl (type);
778 GNUNET_memcpy (&pm[1],
779 data,
780 data_len);
781 if ((UINT32_MAX != reply_anonymity_level) &&
782 (0 != reply_anonymity_level) &&
784 {
785 struct GSF_DelayedHandle *dh;
786
787 dh = GNUNET_new (struct GSF_DelayedHandle);
788 dh->cp = cp;
789 dh->env = env;
790 dh->msize = msize;
793 dh);
795 dh->delay_task =
798 dh);
799 }
800 else
801 {
803 GNUNET_NO,
804 UINT32_MAX,
805 env);
806 }
807 if (GNUNET_BLOCK_REPLY_OK_LAST != eval)
808 return;
809 if (NULL == peerreq->kill_task)
810 {
812 "# P2P searches destroyed due to ultimate reply",
813 1,
814 GNUNET_NO);
815 peerreq->kill_task =
817 peerreq);
818 }
819}

References GSF_DelayedHandle::cp, PeerRequest::cp, data, GSF_ConnectedPeer::delay_queue_size, GSF_DelayedHandle::delay_task, GSF_ConnectedPeer::delayed_head, GSF_ConnectedPeer::delayed_tail, env, GSF_DelayedHandle::env, expiration, free_pending_request(), get_randomized_delay(), GNUNET_assert, GNUNET_BLOCK_REPLY_OK_LAST, GNUNET_BLOCK_TYPE_ANY, GNUNET_break, GNUNET_CONTAINER_DLL_insert, GNUNET_ERROR_TYPE_DEBUG, GNUNET_h2s(), GNUNET_log, GNUNET_MAX_MESSAGE_SIZE, GNUNET_memcpy, GNUNET_MESSAGE_TYPE_FS_PUT, GNUNET_MQ_msg_extra, GNUNET_new, GNUNET_NO, GNUNET_SCHEDULER_add_delayed(), GNUNET_SCHEDULER_add_now(), GNUNET_STATISTICS_update(), GNUNET_TIME_absolute_hton(), GNUNET_YES, GSF_cover_content_count, GSF_enable_randomized_delays, GSF_peer_transmit_(), GSF_pending_request_get_data_(), GSF_stats, PeerRequest::kill_task, GSF_DelayedHandle::msize, peer_request_destroy(), pm, PeerRequest::pr, GSF_PendingRequestData::query, transmit_delayed_now(), type, and GSF_PendingRequestData::type.

Referenced by handle_p2p_get().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ change_peer_respect()

static int change_peer_respect ( struct GSF_ConnectedPeer cp,
int  value 
)
static

Increase the peer's respect by a value.

Parameters
cpwhich peer to change the respect value on
valueis the int value by which the peer's credit is to be increased or decreased
Returns
the actual change in respect (positive or negative)

Definition at line 831 of file gnunet-service-fs_cp.c.

832{
833 if (0 == value)
834 return 0;
835 GNUNET_assert (NULL != cp);
836 if (value > 0)
837 {
838 if (cp->ppd.respect + value < cp->ppd.respect)
839 {
840 value = UINT32_MAX - cp->ppd.respect;
841 cp->ppd.respect = UINT32_MAX;
842 }
843 else
844 cp->ppd.respect += value;
845 }
846 else
847 {
848 if (cp->ppd.respect < -value)
849 {
850 value = -cp->ppd.respect;
851 cp->ppd.respect = 0;
852 }
853 else
854 cp->ppd.respect += value;
855 }
856 return value;
857}

References GSF_DelayedHandle::cp, GNUNET_assert, GSF_ConnectedPeer::ppd, GSF_PeerPerformanceData::respect, and value.

Referenced by bound_priority().

Here is the caller graph for this function:

◆ bound_priority()

static int32_t bound_priority ( uint32_t  prio_in,
struct GSF_ConnectedPeer cp 
)
static

We've received a request with the specified priority.

Bound it according to how much we respect the given peer.

Parameters
prio_inrequested priority
cpthe peer making the request
Returns
effective priority

Definition at line 869 of file gnunet-service-fs_cp.c.

871{
872#define N ((double) 128.0)
873 uint32_t ret;
874 double rret;
875 int ld;
876
878 if (GNUNET_SYSERR == ld)
879 {
880#if INSANE_STATISTICS
883 ("# requests done for free (low load)"), 1,
884 GNUNET_NO);
885#endif
886 return 0; /* excess resources */
887 }
888 if (prio_in > INT32_MAX)
889 prio_in = INT32_MAX;
890 ret = -change_peer_respect (cp, -(int) prio_in);
891 if (ret > 0)
892 {
894 rret = GSF_current_priorities + N;
895 else
896 rret = ret;
898 }
899 if ((GNUNET_YES == ld) && (ret > 0))
900 {
901 /* try with charging */
903 }
904 if (GNUNET_YES == ld)
905 {
908 ("# request dropped, priority insufficient"), 1,
909 GNUNET_NO);
910 /* undo charge */
911 change_peer_respect (cp, (int) ret);
912 return -1; /* not enough resources */
913 }
914 else
915 {
918 ("# requests done for a price (normal load)"),
919 1,
920 GNUNET_NO);
921 }
922#undef N
923 return ret;
924}

References change_peer_respect(), GSF_DelayedHandle::cp, gettext_noop, GNUNET_NO, GNUNET_STATISTICS_update(), GNUNET_SYSERR, GNUNET_YES, GSF_current_priorities, GSF_stats, GSF_test_get_load_too_high_(), N, and ret.

Referenced by handle_p2p_get().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ bound_ttl()

static int32_t bound_ttl ( int32_t  ttl_in,
uint32_t  prio 
)
static

The priority level imposes a bound on the maximum value for the ttl that can be requested.

Parameters
ttl_inrequested ttl
priogiven priority
Returns
ttl_in if ttl_in is below the limit, otherwise the ttl-limit for the given prio

Definition at line 937 of file gnunet-service-fs_cp.c.

939{
940 unsigned long long allowed;
941
942 if (ttl_in <= 0)
943 return ttl_in;
944 allowed = ((unsigned long long) prio) * TTL_DECREMENT / 1000;
945 if (ttl_in > allowed)
946 {
947 if (allowed >= (1 << 30))
948 return 1 << 30;
949 return allowed;
950 }
951 return ttl_in;
952}

References TTL_DECREMENT.

Referenced by handle_p2p_get().

Here is the caller graph for this function:

◆ test_exist_cb()

static int test_exist_cb ( void *  cls,
const struct GNUNET_HashCode hc,
void *  value 
)
static

Test if the query already exists.

If so, merge it, otherwise keep finished at GNUNET_NO.

Parameters
clsour struct TestExistClosure
hcthe key of the query
valuethe existing struct PeerRequest.
Returns
GNUNET_YES to continue to iterate, GNUNET_NO if we successfully merged

Definition at line 993 of file gnunet-service-fs_cp.c.

996{
997 struct TestExistClosure *tec = cls;
998 struct PeerRequest *peerreq = value;
999 struct GSF_PendingRequest *pr;
1000 struct GSF_PendingRequestData *prd;
1001
1002 pr = peerreq->pr;
1004 if (prd->type != tec->type)
1005 return GNUNET_YES;
1006 if (prd->ttl.abs_value_us >=
1007 GNUNET_TIME_absolute_get ().abs_value_us + tec->ttl * 1000LL)
1008 {
1009 /* existing request has higher TTL, drop new one! */
1010 prd->priority += tec->priority;
1012 "Have existing request with higher TTL, dropping new request.\n");
1015 ("# requests dropped due to higher-TTL request")
1016 ,
1017 1, GNUNET_NO);
1018 tec->finished = GNUNET_YES;
1019 return GNUNET_NO;
1020 }
1021 /* existing request has lower TTL, drop old one! */
1022 tec->priority += prd->priority;
1023 free_pending_request (peerreq);
1025 GNUNET_YES);
1026 return GNUNET_NO;
1027}

References GNUNET_TIME_Absolute::abs_value_us, TestExistClosure::finished, free_pending_request(), gettext_noop, GNUNET_ERROR_TYPE_DEBUG, GNUNET_log, GNUNET_NO, GNUNET_STATISTICS_update(), GNUNET_TIME_absolute_get(), GNUNET_YES, GSF_pending_request_cancel_(), GSF_pending_request_get_data_(), GSF_stats, PeerRequest::pr, TestExistClosure::priority, GSF_PendingRequestData::priority, TestExistClosure::ttl, GSF_PendingRequestData::ttl, TestExistClosure::type, GSF_PendingRequestData::type, and value.

Referenced by handle_p2p_get().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ handle_p2p_get()

void handle_p2p_get ( void *  cls,
const struct GetMessage gm 
)

Handle P2P "QUERY" message.

Creates the pending request entry and sets up all of the data structures to that we will process replies properly. Does not initiate forwarding or local database lookups.

Parameters
clsthe other peer involved (sender of the message)
gmthe GET message

Definition at line 1040 of file gnunet-service-fs_cp.c.

1042{
1043 struct GSF_ConnectedPeer *cps = cls;
1044 struct PeerRequest *peerreq;
1045 struct GSF_PendingRequest *pr;
1046 struct GSF_ConnectedPeer *cp;
1047 const struct GNUNET_PeerIdentity *target;
1049 uint16_t msize;
1050 unsigned int bits;
1051 const struct GNUNET_PeerIdentity *opt;
1052 uint32_t bm;
1053 size_t bfsize;
1054 uint32_t ttl_decrement;
1055 struct TestExistClosure tec;
1056 GNUNET_PEER_Id spid;
1057 const struct GSF_PendingRequestData *prd;
1058
1059 msize = ntohs (gm->header.size);
1060 tec.type = ntohl (gm->type);
1061 bm = ntohl (gm->hash_bitmap);
1062 bits = 0;
1063 while (bm > 0)
1064 {
1065 if (1 == (bm & 1))
1066 bits++;
1067 bm >>= 1;
1068 }
1069 opt = (const struct GNUNET_PeerIdentity *) &gm[1];
1070 bfsize = msize - sizeof(struct GetMessage) - bits * sizeof(struct
1074 ("# GET requests received (from other peers)"),
1075 1,
1076 GNUNET_NO);
1078 bm = ntohl (gm->hash_bitmap);
1079 bits = 0;
1080 if (0 != (bm & GET_MESSAGE_BIT_RETURN_TO))
1081 cp = GSF_peer_get_ (&opt[bits++]);
1082 else
1083 cp = cps;
1084 if (NULL == cp)
1085 {
1086 if (0 != (bm & GET_MESSAGE_BIT_RETURN_TO))
1088 "Failed to find RETURN-TO peer `%s' in connection set. Dropping query.\n",
1089 GNUNET_i2s (&opt[bits - 1]));
1090
1091 else
1093 "Failed to find peer `%s' in connection set. Dropping query.\n",
1094 GNUNET_i2s (cps->ppd.peer));
1097 (
1098 "# requests dropped due to missing reverse route"),
1099 1,
1100 GNUNET_NO);
1101 return;
1102 }
1103 unsigned int queue_size = GNUNET_MQ_get_length (cp->mq);
1104 queue_size += cp->ppd.pending_replies + cp->delay_queue_size;
1105 if (queue_size > MAX_QUEUE_PER_PEER)
1106 {
1108 "Peer `%s' has too many replies queued already. Dropping query.\n",
1109 GNUNET_i2s (cps->ppd.peer));
1111 gettext_noop (
1112 "# requests dropped due to full reply queue"),
1113 1,
1114 GNUNET_NO);
1115 return;
1116 }
1117 /* note that we can really only check load here since otherwise
1118 * peers could find out that we are overloaded by not being
1119 * disconnected after sending us a malformed query... */
1120 tec.priority = bound_priority (ntohl (gm->priority),
1121 cps);
1122 if (tec.priority < 0)
1123 {
1125 "Dropping query from `%s', this peer is too busy.\n",
1126 GNUNET_i2s (cps->ppd.peer));
1127 return;
1128 }
1130 "Received request for `%s' of type %u from peer `%s' with flags %u\n",
1131 GNUNET_h2s (&gm->query),
1132 (unsigned int) tec.type,
1133 GNUNET_i2s (cps->ppd.peer),
1134 (unsigned int) bm);
1135 target =
1136 (0 !=
1137 (bm & GET_MESSAGE_BIT_TRANSMIT_TO)) ? (&opt[bits++]) : NULL;
1139 spid = 0;
1140 if ((GNUNET_LOAD_get_load (cp->ppd.transmission_delay) > 3 * (1
1141 + tec.priority))
1143 GNUNET_CONSTANTS_MAX_CORK_DELAY.rel_value_us * 2
1145 {
1146 /* don't have BW to send to peer, or would likely take longer than we have for it,
1147 * so at best indirect the query */
1148 tec.priority = 0;
1150 spid = GNUNET_PEER_intern (cps->ppd.peer);
1151 GNUNET_assert (0 != spid);
1152 }
1153 tec.ttl = bound_ttl (ntohl (gm->ttl),
1154 tec.priority);
1155 /* decrement ttl (always) */
1156 ttl_decrement =
1159 if ((tec.ttl < 0) &&
1160 (((int32_t) (tec.ttl - ttl_decrement)) > 0))
1161 {
1163 "Dropping query from `%s' due to TTL underflow (%d - %u).\n",
1164 GNUNET_i2s (cps->ppd.peer),
1165 tec.ttl,
1166 ttl_decrement);
1169 ("# requests dropped due TTL underflow"), 1,
1170 GNUNET_NO);
1171 /* integer underflow => drop (should be very rare)! */
1172 return;
1173 }
1174 tec.ttl -= ttl_decrement;
1175
1176 /* test if the request already exists */
1177 tec.finished = GNUNET_NO;
1179 &gm->query,
1181 &tec);
1182 if (GNUNET_YES == tec.finished)
1183 return; /* merged into existing request, we're done */
1184
1185 peerreq = GNUNET_new (struct PeerRequest);
1186 peerreq->cp = cp;
1188 tec.type,
1189 &gm->query,
1190 target,
1191 (bfsize > 0)
1192 ? (const char *) &opt[bits]
1193 : NULL,
1194 bfsize,
1195 1 /* anonymity */,
1196 (uint32_t) tec.priority,
1197 tec.ttl,
1198 spid,
1199 GNUNET_PEER_intern (cps->ppd.peer),
1200 NULL, 0, /* replies_seen */
1202 peerreq);
1203 GNUNET_assert (NULL != pr);
1205 peerreq->pr = pr;
1208 &prd->query,
1209 peerreq,
1212 gettext_noop (
1213 "# P2P query messages received and processed"),
1214 1,
1215 GNUNET_NO);
1217 gettext_noop ("# P2P searches active"),
1218 1,
1219 GNUNET_NO);
1223 NULL);
1224}

References bound_priority(), bound_ttl(), PeerRequest::cp, GSF_ConnectedPeer::delay_queue_size, TestExistClosure::finished, GET_MESSAGE_BIT_RETURN_TO, GET_MESSAGE_BIT_TRANSMIT_TO, gettext_noop, GNUNET_assert, GNUNET_break, GNUNET_CONSTANTS_MAX_CORK_DELAY, GNUNET_CONTAINER_multihashmap_get_multiple(), GNUNET_CONTAINER_multihashmap_put(), GNUNET_CONTAINER_MULTIHASHMAPOPTION_MULTIPLE, GNUNET_CRYPTO_QUALITY_WEAK, GNUNET_CRYPTO_random_u32(), GNUNET_ERROR_TYPE_DEBUG, GNUNET_h2s(), GNUNET_i2s(), GNUNET_LOAD_get_average(), GNUNET_LOAD_get_load(), GNUNET_log, GNUNET_MQ_get_length(), GNUNET_new, GNUNET_NO, GNUNET_OK, GNUNET_PEER_intern(), GNUNET_STATISTICS_update(), GNUNET_YES, GSF_consider_forwarding(), GSF_cover_query_count, GSF_local_lookup_(), GSF_peer_get_(), GSF_pending_request_create_(), GSF_pending_request_get_data_(), GSF_PRO_DEFAULTS, GSF_PRO_FORWARD_ONLY, GSF_rt_entry_lifetime, GSF_stats, handle_p2p_reply(), GSF_PendingRequestData::has_started, GetMessage::hash_bitmap, GetMessage::header, MAX_QUEUE_PER_PEER, GSF_ConnectedPeer::mq, options, GSF_PeerPerformanceData::peer, GSF_PeerPerformanceData::pending_replies, GSF_ConnectedPeer::ppd, PeerRequest::pr, GetMessage::priority, TestExistClosure::priority, GetMessage::query, GSF_PendingRequestData::query, GSF_ConnectedPeer::request_map, GNUNET_MessageHeader::size, test_exist_cb(), GSF_PeerPerformanceData::transmission_delay, GetMessage::ttl, TestExistClosure::ttl, TTL_DECREMENT, GetMessage::type, and TestExistClosure::type.

Here is the call graph for this function:

◆ GSF_peer_transmit_()

void GSF_peer_transmit_ ( struct GSF_ConnectedPeer cp,
int  is_query,
uint32_t  priority,
struct GNUNET_MQ_Envelope env 
)

Transmit a message to the given peer as soon as possible.

If the peer disconnects before the transmission can happen, the callback is invoked with a NULL buffer.

Parameters
cptarget peer
is_queryis this a query (GNUNET_YES) or content (GNUNET_NO) or neither (GNUNET_SYSERR)
priorityhow important is this request?
envmessage to send

Definition at line 1238 of file gnunet-service-fs_cp.c.

1242{
1243 struct GSF_PeerTransmitHandle *pth;
1244 struct GSF_PeerTransmitHandle *pos;
1246
1247 pth = GNUNET_new (struct GSF_PeerTransmitHandle);
1249 pth->env = env;
1250 pth->is_query = is_query;
1251 pth->priority = priority;
1252 pth->cp = cp;
1253 /* insertion sort (by priority, descending) */
1254 prev = NULL;
1255 pos = cp->pth_head;
1256 while ((NULL != pos) && (pos->priority > priority))
1257 {
1258 prev = pos;
1259 pos = pos->next;
1260 }
1262 cp->pth_tail,
1263 prev,
1264 pth);
1265 if (GNUNET_YES == is_query)
1267 else if (GNUNET_NO == is_query)
1270}

References GSF_PeerTransmitHandle::cp, env, GSF_PeerTransmitHandle::env, GNUNET_CONTAINER_DLL_insert_after, GNUNET_new, GNUNET_NO, GNUNET_TIME_absolute_get(), GNUNET_YES, GSF_PeerTransmitHandle::is_query, GSF_PeerTransmitHandle::next, GSF_PeerPerformanceData::pending_queries, GSF_PeerPerformanceData::pending_replies, GSF_ConnectedPeer::ppd, GSF_PeerTransmitHandle::prev, GSF_PeerTransmitHandle::priority, GSF_ConnectedPeer::pth_head, GSF_ConnectedPeer::pth_tail, schedule_transmission(), and GSF_PeerTransmitHandle::transmission_request_start_time.

Referenced by GSF_block_peer_migration_(), handle_p2p_reply(), schedule_peer_transmission(), transmit_content(), and transmit_delayed_now().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_peer_update_performance_()

void GSF_peer_update_performance_ ( struct GSF_ConnectedPeer cp,
struct GNUNET_TIME_Absolute  request_time,
uint32_t  request_priority 
)

Report on receiving a reply; update the performance record of the given peer.

Parameters
cpresponding peer (will be updated)
request_timetime at which the original query was transmitted
request_prioritypriority of the original request

Definition at line 1281 of file gnunet-service-fs_cp.c.

1284{
1285 struct GNUNET_TIME_Relative delay;
1286
1287 delay = GNUNET_TIME_absolute_get_duration (request_time);
1290 + delay.rel_value_us) / RUNAVG_DELAY_N;
1291 cp->ppd.avg_priority =
1292 (cp->ppd.avg_priority * (RUNAVG_DELAY_N - 1)
1293 + request_priority) / RUNAVG_DELAY_N;
1294}

References GSF_PeerPerformanceData::avg_priority, GSF_PeerPerformanceData::avg_reply_delay, GNUNET_TIME_absolute_get_duration(), GSF_ConnectedPeer::ppd, GNUNET_TIME_Relative::rel_value_us, and RUNAVG_DELAY_N.

Referenced by update_request_performance_data().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_peer_update_responder_client_()

void GSF_peer_update_responder_client_ ( struct GSF_ConnectedPeer cp,
struct GSF_LocalClient initiator_client 
)

Report on receiving a reply in response to an initiating client.

Remember that this peer is good for this client.

Parameters
cpresponding peer (will be updated)
initiator_clientlocal client on responsible for query

Definition at line 1305 of file gnunet-service-fs_cp.c.

1307{
1309 % CS2P_SUCCESS_LIST_SIZE] = initiator_client;
1310}

References CS2P_SUCCESS_LIST_SIZE, GSF_PeerPerformanceData::last_client_replies, GSF_ConnectedPeer::last_client_replies_woff, and GSF_ConnectedPeer::ppd.

◆ GSF_peer_update_responder_peer_()

void GSF_peer_update_responder_peer_ ( struct GSF_ConnectedPeer cp,
const struct GSF_ConnectedPeer initiator_peer 
)

Report on receiving a reply in response to an initiating peer.

Remember that this peer is good for this initiating peer.

Parameters
cpresponding peer (will be updated)
initiator_peerother peer responsible for query

Definition at line 1321 of file gnunet-service-fs_cp.c.

1323{
1324 unsigned int woff;
1325
1328 cp->ppd.last_p2p_replies[woff] = initiator_peer->ppd.pid;
1329 GNUNET_PEER_change_rc (initiator_peer->ppd.pid, 1);
1331}

References GNUNET_PEER_change_rc(), GSF_PeerPerformanceData::last_p2p_replies, GSF_ConnectedPeer::last_p2p_replies_woff, P2P_SUCCESS_LIST_SIZE, GSF_PeerPerformanceData::pid, and GSF_ConnectedPeer::ppd.

Here is the call graph for this function:

◆ flush_respect()

static int flush_respect ( void *  cls,
const struct GNUNET_PeerIdentity key,
void *  value 
)
static

Write peer-respect information to a file - flush the buffer entry!

Parameters
clsunused
keypeer identity
valuethe struct GSF_ConnectedPeer to flush
Returns
GNUNET_OK to continue iteration

Definition at line 1343 of file gnunet-service-fs_cp.c.

1346{
1347 struct GSF_ConnectedPeer *cp = value;
1348 struct GNUNET_PeerIdentity pid;
1349
1350 if (cp->ppd.respect == cp->disk_respect)
1351 return GNUNET_OK; /* unchanged */
1352 GNUNET_assert (0 != cp->ppd.pid);
1354 GNUNET_PEERSTORE_store (peerstore, "fs", &pid, "respect", &cp->ppd.respect,
1355 sizeof(cp->ppd.respect),
1358 NULL,
1359 NULL);
1360 return GNUNET_OK;
1361}

References GSF_ConnectedPeer::disk_respect, GNUNET_assert, GNUNET_OK, GNUNET_PEER_resolve(), GNUNET_PEERSTORE_store(), GNUNET_PEERSTORE_STOREOPTION_REPLACE, GNUNET_TIME_UNIT_FOREVER_ABS, peerstore, GSF_PeerPerformanceData::pid, pid, GSF_ConnectedPeer::ppd, GSF_PeerPerformanceData::respect, and value.

Referenced by cron_flush_respect(), GSF_connected_peer_done_(), and GSF_peer_disconnect_handler().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_peer_disconnect_handler()

void GSF_peer_disconnect_handler ( void *  cls,
const struct GNUNET_PeerIdentity peer,
void *  internal_cls 
)

A peer disconnected from us.

Tear down the connected peer record.

Parameters
clsunused
peeridentity of peer that disconnected
internal_clsthe corresponding struct GSF_ConnectedPeer

Definition at line 1365 of file gnunet-service-fs_cp.c.

1368{
1369 struct GSF_ConnectedPeer *cp = internal_cls;
1370 struct GSF_PeerTransmitHandle *pth;
1371 struct GSF_DelayedHandle *dh;
1372
1373 if (NULL == cp)
1374 return; /* must have been disconnect from core with
1375 * 'peer' == my_id, ignore */
1376 flush_respect (NULL,
1377 peer,
1378 cp);
1381 peer,
1382 cp));
1384 gettext_noop ("# peers connected"),
1386 GNUNET_NO);
1387 if (NULL != cp->respect_iterate_req)
1388 {
1390 cp->respect_iterate_req = NULL;
1391 }
1394 cp);
1396 cp->request_map = NULL;
1401 memset (cp->ppd.last_p2p_replies,
1402 0,
1403 sizeof(cp->ppd.last_p2p_replies));
1405 while (NULL != (pth = cp->pth_head))
1406 {
1408 cp->pth_tail,
1409 pth);
1410 if (GNUNET_YES == pth->is_query)
1412 else if (GNUNET_NO == pth->is_query)
1414 GNUNET_free (pth);
1415 }
1416 while (NULL != (dh = cp->delayed_head))
1417 {
1420 dh);
1421 GNUNET_MQ_discard (dh->env);
1424 GNUNET_free (dh);
1425 }
1427 if (NULL != cp->mig_revive_task)
1428 {
1430 cp->mig_revive_task = NULL;
1431 }
1434 GNUNET_free (cp);
1435}

References cancel_pending_request(), GSF_DelayedHandle::cp, cp_map, GSF_ConnectedPeer::delay_queue_size, GSF_DelayedHandle::delay_task, GSF_ConnectedPeer::delayed_head, GSF_ConnectedPeer::delayed_tail, GSF_DelayedHandle::env, flush_respect(), gettext_noop, GNUNET_assert, GNUNET_break, GNUNET_CONTAINER_DLL_remove, GNUNET_CONTAINER_multihashmap_destroy(), GNUNET_CONTAINER_multihashmap_iterate(), GNUNET_CONTAINER_multipeermap_remove(), GNUNET_CONTAINER_multipeermap_size(), GNUNET_free, GNUNET_LOAD_value_free, GNUNET_MQ_discard(), GNUNET_NO, GNUNET_PEER_change_rc(), GNUNET_PEER_decrement_rcs(), GNUNET_PEERSTORE_iteration_stop(), GNUNET_SCHEDULER_cancel(), GNUNET_STATISTICS_set(), GNUNET_YES, GSF_plan_notify_peer_disconnect_(), GSF_push_stop_(), GSF_stats, GSF_PeerTransmitHandle::is_query, GSF_PeerPerformanceData::last_p2p_replies, GSF_ConnectedPeer::mig_revive_task, P2P_SUCCESS_LIST_SIZE, GSF_PeerPerformanceData::pending_queries, GSF_PeerPerformanceData::pending_replies, GSF_PeerPerformanceData::pid, GSF_ConnectedPeer::ppd, GSF_ConnectedPeer::pth_head, GSF_ConnectedPeer::pth_tail, GSF_ConnectedPeer::request_map, GSF_ConnectedPeer::respect_iterate_req, and GSF_PeerPerformanceData::transmission_delay.

Referenced by main_init().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ call_iterator()

static int call_iterator ( void *  cls,
const struct GNUNET_PeerIdentity key,
void *  value 
)
static

Function that calls the callback for each peer.

Parameters
clsthe struct IterationContext *
keyidentity of the peer
valuethe struct GSF_ConnectedPeer *
Returns
GNUNET_YES to continue iteration

Definition at line 1464 of file gnunet-service-fs_cp.c.

1467{
1468 struct IterationContext *ic = cls;
1469 struct GSF_ConnectedPeer *cp = value;
1470
1471 ic->it (ic->it_cls,
1472 key, cp,
1473 &cp->ppd);
1474 return GNUNET_YES;
1475}

References GNUNET_YES, IterationContext::it, IterationContext::it_cls, key, GSF_ConnectedPeer::ppd, and value.

Referenced by GSF_iterate_connected_peers_().

Here is the caller graph for this function:

◆ GSF_iterate_connected_peers_()

void GSF_iterate_connected_peers_ ( GSF_ConnectedPeerIterator  it,
void *  it_cls 
)

Iterate over all connected peers.

Parameters
itfunction to call for each peer
it_clsclosure for it

Definition at line 1479 of file gnunet-service-fs_cp.c.

1481{
1482 struct IterationContext ic;
1483
1484 ic.it = it;
1485 ic.it_cls = it_cls;
1488 &ic);
1489}

References call_iterator(), cp_map, GNUNET_CONTAINER_multipeermap_iterate(), IterationContext::it, and IterationContext::it_cls.

Referenced by GSF_consider_forwarding().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_connected_peer_get_identity_()

void GSF_connected_peer_get_identity_ ( const struct GSF_ConnectedPeer cp,
struct GNUNET_PeerIdentity id 
)

Obtain the identity of a connected peer.

Parameters
cppeer to get identity of
ididentity to set (written to)

Definition at line 1499 of file gnunet-service-fs_cp.c.

1501{
1502 GNUNET_assert (0 != cp->ppd.pid);
1503 GNUNET_PEER_resolve (cp->ppd.pid, id);
1504}

References GNUNET_assert, GNUNET_PEER_resolve(), GSF_PeerPerformanceData::pid, and GSF_ConnectedPeer::ppd.

Referenced by consider_peer_for_forwarding().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_connected_peer_get_identity2_()

const struct GNUNET_PeerIdentity * GSF_connected_peer_get_identity2_ ( const struct GSF_ConnectedPeer cp)

Obtain the identity of a connected peer.

Parameters
cppeer to get identity of
Returns
reference to peer identity, valid until peer disconnects (!)

Definition at line 1514 of file gnunet-service-fs_cp.c.

1515{
1516 GNUNET_assert (0 != cp->ppd.pid);
1517 return GNUNET_PEER_resolve2 (cp->ppd.pid);
1518}

References GNUNET_assert, GNUNET_PEER_resolve2(), GSF_PeerPerformanceData::pid, and GSF_ConnectedPeer::ppd.

Referenced by GSF_peer_connect_handler(), GSF_plan_add_(), GSF_plan_notify_peer_disconnect_(), GSF_push_start_(), and process_migration_content().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_block_peer_migration_()

void GSF_block_peer_migration_ ( struct GSF_ConnectedPeer cp,
struct GNUNET_TIME_Absolute  block_time 
)

Ask a peer to stop migrating data to us until the given point in time.

Parameters
cppeer to ask
block_timeuntil when to block

Definition at line 1529 of file gnunet-service-fs_cp.c.

1531{
1532 struct GNUNET_MQ_Envelope *env;
1533 struct MigrationStopMessage *msm;
1534
1535 if (cp->last_migration_block.abs_value_us > block_time.abs_value_us)
1536 {
1538 "Migration already blocked for another %s\n",
1541 (cp->
1542 last_migration_block), GNUNET_YES));
1543 return; /* already blocked */
1544 }
1545 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Asking to stop migration for %s\n",
1548 GNUNET_YES));
1549 cp->last_migration_block = block_time;
1550 env = GNUNET_MQ_msg (msm,
1552 msm->reserved = htonl (0);
1553 msm->duration
1555 (cp->last_migration_block));
1557 gettext_noop ("# migration stop messages sent"),
1558 1,
1559 GNUNET_NO);
1562 UINT32_MAX,
1563 env);
1564}

References GNUNET_TIME_Absolute::abs_value_us, MigrationStopMessage::duration, env, gettext_noop, GNUNET_ERROR_TYPE_DEBUG, GNUNET_log, GNUNET_MESSAGE_TYPE_FS_MIGRATION_STOP, GNUNET_MQ_msg, GNUNET_NO, GNUNET_STATISTICS_update(), GNUNET_STRINGS_relative_time_to_string(), GNUNET_SYSERR, GNUNET_TIME_absolute_get_remaining(), GNUNET_TIME_relative_hton(), GNUNET_YES, GSF_peer_transmit_(), GSF_stats, GSF_ConnectedPeer::last_migration_block, and MigrationStopMessage::reserved.

Referenced by handle_p2p_put(), and put_migration_continuation().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_connected_peer_change_preference_()

void GSF_connected_peer_change_preference_ ( struct GSF_ConnectedPeer cp,
uint64_t  pref 
)

Notify core about a preference we have for the given peer (to allocate more resources towards it).

The change will be communicated the next time we reserve bandwidth with core (not instantly).

Parameters
cppeer to reserve bandwidth from
prefpreference change

Definition at line 1577 of file gnunet-service-fs_cp.c.

1579{
1580 cp->inc_preference += pref;
1581}

References GSF_ConnectedPeer::inc_preference.

Referenced by handle_p2p_put().

Here is the caller graph for this function:

◆ cron_flush_respect()

static void cron_flush_respect ( void *  cls)
static

Call this method periodically to flush respect information to disk.

Parameters
clsclosure, not used

Definition at line 1590 of file gnunet-service-fs_cp.c.

References cp_map, cron_flush_respect(), flush_respect(), fr_task, GNUNET_CONTAINER_multipeermap_iterate(), GNUNET_SCHEDULER_add_delayed_with_priority(), GNUNET_SCHEDULER_PRIORITY_HIGH, and RESPECT_FLUSH_FREQ.

Referenced by cron_flush_respect(), and GSF_connected_peer_init_().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_connected_peer_init_()

void GSF_connected_peer_init_ ( void  )

Initialize peer management subsystem.

Definition at line 1607 of file gnunet-service-fs_cp.c.

References cp_map, cron_flush_respect(), fr_task, GNUNET_CONTAINER_multipeermap_create(), GNUNET_PEERSTORE_connect(), GNUNET_SCHEDULER_add_with_priority(), GNUNET_SCHEDULER_PRIORITY_HIGH, GNUNET_YES, GSF_cfg, and peerstore.

Referenced by run().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSF_connected_peer_done_()

void GSF_connected_peer_done_ ( void  )

Shutdown peer management subsystem.

Definition at line 1620 of file gnunet-service-fs_cp.c.

References cp_map, flush_respect(), fr_task, GNUNET_CONTAINER_multipeermap_destroy(), GNUNET_CONTAINER_multipeermap_iterate(), GNUNET_PEERSTORE_disconnect(), GNUNET_SCHEDULER_cancel(), and peerstore.

Referenced by shutdown_task().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ clean_local_client()

static int clean_local_client ( void *  cls,
const struct GNUNET_PeerIdentity key,
void *  value 
)
static

Iterator to remove references to LC entry.

Parameters
clsthe struct GSF_LocalClient * to look for
keycurrent key code
valuevalue in the hash map (peer entry)
Returns
GNUNET_YES (we should continue to iterate)

Definition at line 1642 of file gnunet-service-fs_cp.c.

1645{
1646 const struct GSF_LocalClient *lc = cls;
1647 struct GSF_ConnectedPeer *cp = value;
1648 unsigned int i;
1649
1650 for (i = 0; i < CS2P_SUCCESS_LIST_SIZE; i++)
1651 if (cp->ppd.last_client_replies[i] == lc)
1652 cp->ppd.last_client_replies[i] = NULL;
1653 return GNUNET_YES;
1654}

References CS2P_SUCCESS_LIST_SIZE, GNUNET_YES, GSF_PeerPerformanceData::last_client_replies, GSF_ConnectedPeer::ppd, and value.

Referenced by GSF_handle_local_client_disconnect_().

Here is the caller graph for this function:

◆ GSF_handle_local_client_disconnect_()

void GSF_handle_local_client_disconnect_ ( const struct GSF_LocalClient lc)

Notification that a local client disconnected.

Clean up all of our references to the given handle.

Parameters
lchandle to the local client (henceforth invalid)

Definition at line 1664 of file gnunet-service-fs_cp.c.

1665{
1666 if (NULL == cp_map)
1667 return; /* already cleaned up */
1670 (void *) lc);
1671}

References clean_local_client(), cp_map, and GNUNET_CONTAINER_multipeermap_iterate().

Here is the call graph for this function:

Variable Documentation

◆ cp_map

◆ peerstore

struct GNUNET_PEERSTORE_Handle* peerstore
static

Handle to peerstore service.

Definition at line 273 of file gnunet-service-fs_cp.c.

Referenced by flush_respect(), GSF_connected_peer_done_(), GSF_connected_peer_init_(), and GSF_peer_connect_handler().

◆ fr_task

struct GNUNET_SCHEDULER_Task* fr_task
static

Task used to flush respect values to disk.

Definition at line 278 of file gnunet-service-fs_cp.c.

Referenced by cron_flush_respect(), GSF_connected_peer_done_(), and GSF_connected_peer_init_().