123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393 |
- /*
- This file is part of GNUnet.
- Copyright (C) 2010-2013, 2016 GNUnet e.V.
- GNUnet is free software: you can redistribute it and/or modify it
- under the terms of the GNU Affero General Public License as published
- by the Free Software Foundation, either version 3 of the License,
- or (at your option) any later version.
- GNUnet is distributed in the hope that it will be useful, but
- WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- Affero General Public License for more details.
- You should have received a copy of the GNU Affero General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
- SPDX-License-Identifier: AGPL3.0-or-later
- */
- /**
- * @file ats-tests/ats-testing-traffic.c
- * @brief ats benchmark: traffic generator
- * @author Christian Grothoff
- * @author Matthias Wachs
- */
- #include "platform.h"
- #include "gnunet_util_lib.h"
- #include "ats-testing.h"
- static struct TrafficGenerator *tg_head;
- static struct TrafficGenerator *tg_tail;
- extern struct GNUNET_ATS_TEST_Topology *top;
- static struct GNUNET_TIME_Relative
- get_delay (struct TrafficGenerator *tg)
- {
- struct GNUNET_TIME_Relative delay;
- struct GNUNET_TIME_Relative time_delta;
- long long int cur_rate;
- long long int delta_rate;
- delay.rel_value_us = 0;
- /* Calculate the current transmission rate based on the type of traffic */
- switch (tg->type)
- {
- case GNUNET_ATS_TEST_TG_CONSTANT:
- if (UINT32_MAX == tg->base_rate)
- return GNUNET_TIME_UNIT_ZERO;
- cur_rate = tg->base_rate;
- break;
- case GNUNET_ATS_TEST_TG_LINEAR:
- time_delta = GNUNET_TIME_absolute_get_duration (tg->time_start);
- /* Calculate point of time in the current period */
- time_delta.rel_value_us = time_delta.rel_value_us
- % tg->duration_period.rel_value_us;
- delta_rate = ((double) time_delta.rel_value_us
- / tg->duration_period.rel_value_us)
- * (tg->max_rate - tg->base_rate);
- if ((tg->max_rate < tg->base_rate) && ((tg->max_rate - tg->base_rate) >
- tg->base_rate))
- {
- /* This will cause an underflow */
- GNUNET_break (0);
- }
- cur_rate = tg->base_rate + delta_rate;
- break;
- case GNUNET_ATS_TEST_TG_RANDOM:
- cur_rate = tg->base_rate + GNUNET_CRYPTO_random_u32 (
- GNUNET_CRYPTO_QUALITY_WEAK,
- tg->max_rate
- - tg->base_rate);
- break;
- case GNUNET_ATS_TEST_TG_SINUS:
- time_delta = GNUNET_TIME_absolute_get_duration (tg->time_start);
- /* Calculate point of time in the current period */
- time_delta.rel_value_us = time_delta.rel_value_us
- % tg->duration_period.rel_value_us;
- if ((tg->max_rate - tg->base_rate) > tg->base_rate)
- {
- /* This will cause an underflow for second half of sinus period,
- * will be detected in general when experiments are loaded */
- GNUNET_break (0);
- }
- delta_rate = (tg->max_rate - tg->base_rate)
- * sin ((2 * M_PI)
- / ((double) tg->duration_period.rel_value_us)
- * time_delta.rel_value_us);
- cur_rate = tg->base_rate + delta_rate;
- break;
- default:
- return delay;
- break;
- }
- if (cur_rate < 0)
- {
- cur_rate = 1;
- }
- /* Calculate the delay for the next message based on the current delay */
- delay.rel_value_us = GNUNET_TIME_UNIT_SECONDS.rel_value_us
- * TEST_MESSAGE_SIZE / cur_rate;
- GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
- "Current rate is %lld, calculated delay is %llu\n",
- cur_rate,
- (unsigned long long) delay.rel_value_us);
- return delay;
- }
- static void
- update_ping_data (void *cls)
- {
- struct BenchmarkPartner *p = cls;
- struct GNUNET_TIME_Relative delay;
- p->messages_sent++;
- p->bytes_sent += TEST_MESSAGE_SIZE;
- p->me->total_messages_sent++;
- p->me->total_bytes_sent += TEST_MESSAGE_SIZE;
- if (NULL == p->tg)
- {
- GNUNET_break (0);
- return;
- }
- delay = get_delay (p->tg);
- GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
- "Delay for next transmission %s\n",
- GNUNET_STRINGS_relative_time_to_string (delay,
- GNUNET_YES));
- p->tg->next_ping_transmission
- = GNUNET_TIME_absolute_add (GNUNET_TIME_absolute_get (),
- delay);
- }
- static void
- comm_schedule_send (void *cls)
- {
- struct BenchmarkPartner *p = cls;
- struct TestMessage *msg;
- struct GNUNET_MQ_Envelope *env;
- p->tg->send_task = NULL;
- p->last_message_sent = GNUNET_TIME_absolute_get ();
- env = GNUNET_MQ_msg (msg,
- TEST_MESSAGE_TYPE_PING);
- memset (msg->padding,
- 'a',
- sizeof(msg->padding));
- GNUNET_MQ_notify_sent (env,
- &update_ping_data,
- p);
- GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
- "Master [%u]: Sending PING to [%u]\n",
- p->me->no,
- p->dest->no);
- GNUNET_MQ_send (p->mq,
- env);
- }
- static void
- update_pong_data (void *cls)
- {
- struct BenchmarkPartner *p = cls;
- p->messages_sent++;
- p->bytes_sent += TEST_MESSAGE_SIZE;
- p->me->total_messages_sent++;
- p->me->total_bytes_sent += TEST_MESSAGE_SIZE;
- }
- void
- GNUNET_ATS_TEST_traffic_handle_ping (struct BenchmarkPartner *p)
- {
- struct TestMessage *msg;
- struct GNUNET_MQ_Envelope *env;
- GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
- "Slave [%u]: Received PING from [%u], sending PONG\n",
- p->me->no,
- p->dest->no);
- p->messages_received++;
- p->bytes_received += TEST_MESSAGE_SIZE;
- p->me->total_messages_received++;
- p->me->total_bytes_received += TEST_MESSAGE_SIZE;
- env = GNUNET_MQ_msg (msg,
- TEST_MESSAGE_TYPE_PING);
- memset (msg->padding,
- 'a',
- sizeof(msg->padding));
- GNUNET_MQ_notify_sent (env,
- &update_pong_data,
- p);
- GNUNET_MQ_send (p->mq,
- env);
- }
- void
- GNUNET_ATS_TEST_traffic_handle_pong (struct BenchmarkPartner *p)
- {
- struct GNUNET_TIME_Relative left;
- GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
- "Master [%u]: Received PONG from [%u], next message\n",
- p->me->no,
- p->dest->no);
- p->messages_received++;
- p->bytes_received += TEST_MESSAGE_SIZE;
- p->me->total_messages_received++;
- p->me->total_bytes_received += TEST_MESSAGE_SIZE;
- p->total_app_rtt += GNUNET_TIME_absolute_get_difference (p->last_message_sent,
- GNUNET_TIME_absolute_get ())
- .rel_value_us;
- /* Schedule next send event */
- if (NULL == p->tg)
- return;
- left = GNUNET_TIME_absolute_get_remaining (p->tg->next_ping_transmission);
- if (UINT32_MAX == p->tg->base_rate)
- {
- p->tg->send_task = GNUNET_SCHEDULER_add_now (&comm_schedule_send, p);
- }
- else if (0 == left.rel_value_us)
- {
- p->tg->send_task = GNUNET_SCHEDULER_add_now (&comm_schedule_send, p);
- }
- else
- {
- /* Enforce minimum transmission rate 1 msg / sec */
- if (GNUNET_TIME_UNIT_SECONDS.rel_value_us == (left =
- GNUNET_TIME_relative_min (
- left,
- GNUNET_TIME_UNIT_SECONDS))
- .rel_value_us)
- GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
- "Enforcing minimum send rate between master [%u] and slave [%u]\n",
- p->me->no, p->dest->no);
- p->tg->send_task = GNUNET_SCHEDULER_add_delayed (left,
- &comm_schedule_send, p);
- }
- }
- /**
- * Generate between the source master and the partner and send traffic with a
- * maximum rate.
- *
- * @param src traffic source
- * @param dest traffic partner
- * @param type type of traffic to generate
- * @param base_rate traffic base rate to send data with
- * @param max_rate traffic maximum rate to send data with
- * @param period duration of a period of traffic generation (~ 1/frequency)
- * @param duration how long to generate traffic
- * @return the traffic generator
- */
- struct TrafficGenerator *
- GNUNET_ATS_TEST_generate_traffic_start (struct BenchmarkPeer *src,
- struct BenchmarkPartner *dest,
- enum GeneratorType type,
- unsigned int base_rate,
- unsigned int max_rate,
- struct GNUNET_TIME_Relative period,
- struct GNUNET_TIME_Relative duration)
- {
- struct TrafficGenerator *tg;
- if (NULL != dest->tg)
- {
- GNUNET_break (0);
- return NULL;
- }
- tg = GNUNET_new (struct TrafficGenerator);
- GNUNET_CONTAINER_DLL_insert (tg_head,
- tg_tail,
- tg);
- tg->type = type;
- tg->src = src;
- tg->dest = dest;
- tg->base_rate = base_rate;
- tg->max_rate = max_rate;
- tg->duration_period = period;
- tg->time_start = GNUNET_TIME_absolute_get ();
- tg->next_ping_transmission = GNUNET_TIME_UNIT_FOREVER_ABS;
- switch (type)
- {
- case GNUNET_ATS_TEST_TG_CONSTANT:
- GNUNET_log (GNUNET_ERROR_TYPE_INFO,
- "Setting up constant traffic generator master[%u] `%s' and slave [%u] `%s' max %u Bips\n",
- dest->me->no,
- GNUNET_i2s (&dest->me->id),
- dest->dest->no,
- GNUNET_i2s (&dest->dest->id),
- base_rate);
- break;
- case GNUNET_ATS_TEST_TG_LINEAR:
- GNUNET_log (GNUNET_ERROR_TYPE_INFO,
- "Setting up linear traffic generator master[%u] `%s' and slave [%u] `%s' min %u Bips max %u Bips\n",
- dest->me->no,
- GNUNET_i2s (&dest->me->id),
- dest->dest->no,
- GNUNET_i2s (&dest->dest->id),
- base_rate,
- max_rate);
- break;
- case GNUNET_ATS_TEST_TG_SINUS:
- GNUNET_log (GNUNET_ERROR_TYPE_INFO,
- "Setting up sinus traffic generator master[%u] `%s' and slave [%u] `%s' baserate %u Bips, amplitude %u Bps\n",
- dest->me->no,
- GNUNET_i2s (&dest->me->id),
- dest->dest->no,
- GNUNET_i2s (&dest->dest->id),
- base_rate,
- max_rate);
- break;
- case GNUNET_ATS_TEST_TG_RANDOM:
- GNUNET_log (GNUNET_ERROR_TYPE_INFO,
- "Setting up random traffic generator master[%u] `%s' and slave [%u] `%s' min %u Bips max %u Bps\n",
- dest->me->no,
- GNUNET_i2s (&dest->me->id),
- dest->dest->no,
- GNUNET_i2s (&dest->dest->id),
- base_rate,
- max_rate);
- break;
- default:
- break;
- }
- dest->tg = tg;
- tg->send_task
- = GNUNET_SCHEDULER_add_now (&comm_schedule_send,
- dest);
- return tg;
- }
- void
- GNUNET_ATS_TEST_generate_traffic_stop (struct TrafficGenerator *tg)
- {
- GNUNET_CONTAINER_DLL_remove (tg_head,
- tg_tail,
- tg);
- tg->dest->tg = NULL;
- if (NULL != tg->send_task)
- {
- GNUNET_SCHEDULER_cancel (tg->send_task);
- tg->send_task = NULL;
- }
- GNUNET_free (tg);
- }
- /**
- * Stop all traffic generators
- */
- void
- GNUNET_ATS_TEST_generate_traffic_stop_all ()
- {
- struct TrafficGenerator *cur;
- struct TrafficGenerator *next;
- next = tg_head;
- for (cur = next; NULL != cur; cur = next)
- {
- next = cur->next;
- GNUNET_ATS_TEST_generate_traffic_stop (cur);
- }
- }
- /* end of file ats-testing-traffic.c */
|