From: Christian Grothoff Date: Sun, 29 Jan 2017 16:57:58 +0000 (+0100) Subject: optimize mqm_head scans by avoiding constantly scanning over definitively non-ready... X-Git-Tag: taler-0.2.1~257 X-Git-Url: https://git.librecmc.org/?a=commitdiff_plain;h=dc430a9d23f8fd76f700d6836d4f4748429d70cd;p=oweals%2Fgnunet.git optimize mqm_head scans by avoiding constantly scanning over definitively non-ready entries --- diff --git a/src/cadet/gnunet-service-cadet-new_peer.c b/src/cadet/gnunet-service-cadet-new_peer.c index 97bb1378e..bcc69be53 100644 --- a/src/cadet/gnunet-service-cadet-new_peer.c +++ b/src/cadet/gnunet-service-cadet-new_peer.c @@ -29,7 +29,6 @@ * where we actually need it (i.e. not if we have a direct connection, * or if we already have plenty of good short ones, or maybe even * to take a break if we have some connections and have searched a lot (?)) - * - optimize MQM ready scans (O(n) -> O(1)) */ #include "platform.h" #include "gnunet_util_lib.h" @@ -141,6 +140,11 @@ struct CadetPeer */ struct GCP_MessageQueueManager *mqm_tail; + /** + * Pointer to first "ready" entry in @e mqm_head. + */ + struct GCP_MessageQueueManager *mqm_ready_ptr; + /** * MIN-heap of paths owned by this peer (they also end at this * peer). Ordered by desirability. @@ -295,6 +299,7 @@ destroy_peer (void *cls) notifications. Thus we can assert that mqm_head is empty at this point. */ GNUNET_assert (NULL == cp->mqm_head); + GNUNET_assert (NULL == cp->mqm_ready_ptr); GNUNET_free (cp); } @@ -540,6 +545,10 @@ mqm_execute (struct GCP_MessageQueueManager *mqm) { struct CadetPeer *cp = mqm->cp; + /* Move ready pointer to the next entry that might be ready. */ + if ( (mqm == cp->mqm_ready_ptr) && + (NULL != mqm->next) ) + cp->mqm_ready_ptr = mqm->next; /* Move entry to the end of the DLL, to be fair. */ if (mqm != cp->mqm_tail) { @@ -576,6 +585,29 @@ mqm_execute (struct GCP_MessageQueueManager *mqm) } +/** + * Find the next ready message in the queue (starting + * the search from the `cp->mqm_ready_ptr`) and if possible + * execute the transmission. + * + * @param cp peer to try to send the next ready message to + */ +static void +send_next_ready (struct CadetPeer *cp) +{ + struct GCP_MessageQueueManager *mqm; + + if (0 == cp->mqm_ready_counter) + return; + while ( (NULL != (mqm = cp->mqm_ready_ptr)) && + (NULL == mqm->env) ) + cp->mqm_ready_ptr = mqm->next; + if (NULL == mqm) + return; /* nothing to do */ + mqm_execute (mqm); +} + + /** * Function called when CORE took one of the messages from * a message queue manager and transmitted it. @@ -590,17 +622,7 @@ mqm_send_done (void *cls) LOG (GNUNET_ERROR_TYPE_DEBUG, "Sending to peer %s completed\n", GCP_2s (cp)); - if (0 == cp->mqm_ready_counter) - return; /* nothing to do */ - for (struct GCP_MessageQueueManager *mqm = cp->mqm_head; - NULL != mqm; - mqm = mqm->next) - { - if (NULL == mqm->env) - continue; - mqm_execute (mqm); - return; - } + send_next_ready (cp); } @@ -617,6 +639,7 @@ GCP_send (struct GCP_MessageQueueManager *mqm, { struct CadetPeer *cp = mqm->cp; + GNUNET_assert (NULL != env); LOG (GNUNET_ERROR_TYPE_DEBUG, "Queueing message to peer %s in MQM %p\n", GCP_2s (cp), @@ -628,9 +651,13 @@ GCP_send (struct GCP_MessageQueueManager *mqm, cp); mqm->env = env; cp->mqm_ready_counter++; + if (mqm != cp->mqm_ready_ptr) + cp->mqm_ready_ptr = cp->mqm_head; + if (1 == cp->mqm_ready_counter) + cp->mqm_ready_ptr = mqm; if (0 != GNUNET_MQ_get_length (cp->core_mq)) return; - mqm_execute (mqm); + send_next_ready (cp); } @@ -1285,6 +1312,8 @@ GCP_request_mq_cancel (struct GCP_MessageQueueManager *mqm, else GNUNET_MQ_discard (last_env); } + if (cp->mqm_ready_ptr == mqm) + cp->mqm_ready_ptr = mqm->next; GNUNET_CONTAINER_DLL_remove (cp->mqm_head, cp->mqm_tail, mqm);