aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMatthias Wachs <wachs@net.in.tum.de>2014-02-07 16:03:59 +0000
committerMatthias Wachs <wachs@net.in.tum.de>2014-02-07 16:03:59 +0000
commitc60d7753d1a7f8ff3ea97eeeaad5085cefb69bda (patch)
treedb34c162140a22667105222ff05535fb17fb85b1
parente0454193d828a6a96ea662f53666be38f8369a8f (diff)
downloadgnunet-c60d7753d1a7f8ff3ea97eeeaad5085cefb69bda.tar.gz
gnunet-c60d7753d1a7f8ff3ea97eeeaad5085cefb69bda.zip
moved eval tool to ats
-rw-r--r--src/ats/Makefile.am15
-rw-r--r--src/ats/gnunet-ats-solver-eval.c1536
-rw-r--r--src/ats/gnunet-ats-solver-eval.h930
3 files changed, 2481 insertions, 0 deletions
diff --git a/src/ats/Makefile.am b/src/ats/Makefile.am
index 63a781abf..42401932c 100644
--- a/src/ats/Makefile.am
+++ b/src/ats/Makefile.am
@@ -17,6 +17,9 @@ if USE_COVERAGE
17 AM_CFLAGS = -fprofile-arcs -ftest-coverage 17 AM_CFLAGS = -fprofile-arcs -ftest-coverage
18endif 18endif
19 19
20noinst_PROGRAMS = \
21 gnunet-ats-solver-eval
22
20if HAVE_LIBGLPK 23if HAVE_LIBGLPK
21 GN_LIBGLPK = -lglpk 24 GN_LIBGLPK = -lglpk
22 GN_MLP_LIB = libgnunet_plugin_ats_mlp.la 25 GN_MLP_LIB = libgnunet_plugin_ats_mlp.la
@@ -37,6 +40,18 @@ plugin_LTLIBRARIES = \
37 $(GN_MLP_LIB) \ 40 $(GN_MLP_LIB) \
38 libgnunet_plugin_ats_ril.la 41 libgnunet_plugin_ats_ril.la
39 42
43gnunet_ats_solver_eval_SOURCES = \
44 gnunet-ats-solver-eval.c gnunet-service-ats_normalization.c
45gnunet_ats_solver_eval_LDADD = \
46 $(top_builddir)/src/util/libgnunetutil.la \
47 $(top_builddir)/src/ats/libgnunetats.la \
48 $(top_builddir)/src/statistics/libgnunetstatistics.la \
49 $(LTLIBINTL)
50gnunet_ats_solver_eval_DEPENDENCIES = \
51 $(top_builddir)/src/util/libgnunetutil.la \
52 $(top_builddir)/src/ats/libgnunetats.la \
53 $(top_builddir)/src/statistics/libgnunetstatistics.la
54
40libgnunetats_la_SOURCES = \ 55libgnunetats_la_SOURCES = \
41 ats_api_scheduling.c \ 56 ats_api_scheduling.c \
42 ats_api_performance.c 57 ats_api_performance.c
diff --git a/src/ats/gnunet-ats-solver-eval.c b/src/ats/gnunet-ats-solver-eval.c
new file mode 100644
index 000000000..adf7177ba
--- /dev/null
+++ b/src/ats/gnunet-ats-solver-eval.c
@@ -0,0 +1,1536 @@
1/*
2 This file is part of GNUnet.
3 (C) 2010-2013 Christian Grothoff (and other contributing authors)
4
5 GNUnet is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; either version 3, or (at your
8 option) any later version.
9
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with GNUnet; see the file COPYING. If not, write to the
17 Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 Boston, MA 02111-1307, USA.
19 */
20/**
21 * @file ats-tests/ats-testing-experiment.c
22 * @brief ats benchmark: controlled experiment execution
23 * @author Christian Grothoff
24 * @author Matthias Wachs
25 */
26#include "platform.h"
27#include "gnunet_util_lib.h"
28#include "gnunet-ats-solver-eval.h"
29
30
31#define BIG_M_STRING "unlimited"
32
33static struct Experiment *e;
34
35static struct GNUNET_ATS_TESTING_SolverHandle *sh;
36
37/**
38 * cmd option -e: experiment file
39 */
40static char *opt_exp_file;
41
42static char *opt_solver;
43
44/**
45 * cmd option -l: enable logging
46 */
47static int opt_log;
48
49/**
50 * cmd option -p: enable plots
51 */
52static int opt_plot;
53
54/**
55 * cmd option -v: verbose logs
56 */
57static int opt_verbose;
58
59static int res;
60
61static void
62end_now ();
63
64/**
65 * Experiments
66 */
67
68const char *
69print_op (enum OperationType op)
70{
71 switch (op) {
72 case SOLVER_OP_ADD_ADDRESS:
73 return "ADD_ADDRESS";
74 case SOLVER_OP_DEL_ADDRESS:
75 return "DEL_ADDRESS";
76 case SOLVER_OP_START_SET_PREFERENCE:
77 return "START_SET_PREFERENCE";
78 case SOLVER_OP_STOP_SET_PREFERENCE:
79 return "STOP_STOP_PREFERENCE";
80 case SOLVER_OP_START_SET_PROPERTY:
81 return "START_SET_PROPERTY";
82 case SOLVER_OP_STOP_SET_PROPERTY:
83 return "STOP_SET_PROPERTY";
84 default:
85 break;
86 }
87 return "";
88}
89
90
91static struct Experiment *
92create_experiment ()
93{
94 struct Experiment *e;
95 e = GNUNET_new (struct Experiment);
96 e->name = NULL;
97 e->num_masters = 0;
98 e->num_slaves = 0;
99 e->start = NULL;
100 e->total_duration = GNUNET_TIME_UNIT_ZERO;
101 return e;
102}
103
104static void
105free_experiment (struct Experiment *e)
106{
107 struct Episode *cur;
108 struct Episode *next;
109 struct GNUNET_ATS_TEST_Operation *cur_o;
110 struct GNUNET_ATS_TEST_Operation *next_o;
111
112 next = e->start;
113 for (cur = next; NULL != cur; cur = next)
114 {
115 next = cur->next;
116
117 next_o = cur->head;
118 for (cur_o = next_o; NULL != cur_o; cur_o = next_o)
119 {
120 next_o = cur_o->next;
121 GNUNET_free (cur_o);
122 }
123 GNUNET_free (cur);
124 }
125
126 GNUNET_free_non_null (e->name);
127 GNUNET_free_non_null (e->cfg_file);
128 GNUNET_free (e);
129}
130
131
132static int
133load_op_add_address (struct GNUNET_ATS_TEST_Operation *o,
134 int op_counter,
135 char *sec_name,
136 const struct GNUNET_CONFIGURATION_Handle *cfg)
137{
138 char *op_name;
139
140 /* peer id */
141 GNUNET_asprintf(&op_name, "op-%u-peer-id", op_counter);
142 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
143 sec_name, op_name, &o->peer_id))
144 {
145 fprintf (stderr, "Missing peer-id in operation %u `%s' in episode `%s'\n",
146 op_counter, "ADD_ADDRESS", op_name);
147 GNUNET_free (op_name);
148 return GNUNET_SYSERR;
149 }
150 GNUNET_free (op_name);
151
152 /* address id */
153 GNUNET_asprintf(&op_name, "op-%u-address-id", op_counter);
154 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
155 sec_name, op_name, &o->address_id))
156 {
157 fprintf (stderr, "Missing address-id in operation %u `%s' in episode `%s'\n",
158 op_counter, "ADD_ADDRESS", op_name);
159 GNUNET_free (op_name);
160 return GNUNET_SYSERR;
161 }
162 GNUNET_free (op_name);
163
164 /* plugin */
165 GNUNET_asprintf(&op_name, "op-%u-plugin", op_counter);
166 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string (cfg,
167 sec_name, op_name, &o->plugin))
168 {
169 fprintf (stderr, "Missing plugin in operation %u `%s' in episode `%s'\n",
170 op_counter, "ADD_ADDRESS", op_name);
171 GNUNET_free (op_name);
172 return GNUNET_SYSERR;
173 }
174 GNUNET_free (op_name);
175
176 /* address */
177 GNUNET_asprintf(&op_name, "op-%u-address", op_counter);
178 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string (cfg,
179 sec_name, op_name, &o->address))
180 {
181 fprintf (stderr, "Missing address in operation %u `%s' in episode `%s'\n",
182 op_counter, "ADD_ADDRESS", op_name);
183 GNUNET_free (op_name);
184 return GNUNET_SYSERR;
185 }
186 GNUNET_free (op_name);
187
188 /* session */
189 GNUNET_asprintf(&op_name, "op-%u-address-session", op_counter);
190 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
191 sec_name, op_name, &o->address_session))
192 {
193 fprintf (stderr, "Missing address-session in operation %u `%s' in episode `%s'\n",
194 op_counter, "ADD_ADDRESS", op_name);
195 GNUNET_free (op_name);
196 return GNUNET_SYSERR;
197 }
198 GNUNET_free (op_name);
199
200 /* network */
201 GNUNET_asprintf(&op_name, "op-%u-address-network", op_counter);
202 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
203 sec_name, op_name, &o->address_network))
204 {
205 fprintf (stderr, "Missing address-network in operation %u `%s' in episode `%s'\n",
206 op_counter, "ADD_ADDRESS", op_name);
207 GNUNET_free (op_name);
208 return GNUNET_SYSERR;
209 }
210 GNUNET_free (op_name);
211
212 fprintf (stderr,
213 "Found operation %s: [%llu:%llu] address `%s' plugin `%s' \n",
214 "ADD_ADDRESS", o->peer_id, o->address_id, o->address, o->plugin);
215
216 return GNUNET_OK;
217}
218
219static int
220load_op_del_address (struct GNUNET_ATS_TEST_Operation *o,
221 int op_counter,
222 char *sec_name,
223 const struct GNUNET_CONFIGURATION_Handle *cfg)
224{
225 char *op_name;
226
227 /* peer id */
228 GNUNET_asprintf(&op_name, "op-%u-peer-id", op_counter);
229 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
230 sec_name, op_name, &o->peer_id))
231 {
232 fprintf (stderr, "Missing peer-id in operation %u `%s' in episode `%s'\n",
233 op_counter, "DEL_ADDRESS", op_name);
234 GNUNET_free (op_name);
235 return GNUNET_SYSERR;
236 }
237 GNUNET_free (op_name);
238
239 /* address id */
240 GNUNET_asprintf(&op_name, "op-%u-address-id", op_counter);
241 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
242 sec_name, op_name, &o->address_id))
243 {
244 fprintf (stderr, "Missing address-id in operation %u `%s' in episode `%s'\n",
245 op_counter, "DEL_ADDRESS", op_name);
246 GNUNET_free (op_name);
247 return GNUNET_SYSERR;
248 }
249 GNUNET_free (op_name);
250
251 /* plugin */
252 GNUNET_asprintf(&op_name, "op-%u-plugin", op_counter);
253 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string (cfg,
254 sec_name, op_name, &o->plugin))
255 {
256 fprintf (stderr, "Missing plugin in operation %u `%s' in episode `%s'\n",
257 op_counter, "DEL_ADDRESS", op_name);
258 GNUNET_free (op_name);
259 return GNUNET_SYSERR;
260 }
261 GNUNET_free (op_name);
262
263 /* address */
264 GNUNET_asprintf(&op_name, "op-%u-address", op_counter);
265 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string (cfg,
266 sec_name, op_name, &o->address))
267 {
268 fprintf (stderr, "Missing address in operation %u `%s' in episode `%s'\n",
269 op_counter, "DEL_ADDRESS", op_name);
270 GNUNET_free (op_name);
271 return GNUNET_SYSERR;
272 }
273 GNUNET_free (op_name);
274
275 /* session */
276 GNUNET_asprintf(&op_name, "op-%u-address-session", op_counter);
277 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
278 sec_name, op_name, &o->address_session))
279 {
280 fprintf (stderr, "Missing address-session in operation %u `%s' in episode `%s'\n",
281 op_counter, "DEL_ADDRESS", op_name);
282 GNUNET_free (op_name);
283 return GNUNET_SYSERR;
284 }
285 GNUNET_free (op_name);
286
287 /* network */
288 GNUNET_asprintf(&op_name, "op-%u-address-network", op_counter);
289 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
290 sec_name, op_name, &o->address_network))
291 {
292 fprintf (stderr, "Missing address-network in operation %u `%s' in episode `%s'\n",
293 op_counter, "DEL_ADDRESS", op_name);
294 GNUNET_free (op_name);
295 return GNUNET_SYSERR;
296 }
297 GNUNET_free (op_name);
298
299 fprintf (stderr,
300 "Found operation %s: [%llu:%llu] address `%s' plugin `%s' \n",
301 "DEL_ADDRESS", o->peer_id, o->address_id, o->address, o->plugin);
302
303 return GNUNET_OK;
304}
305
306static int
307load_episode (struct Experiment *e, struct Episode *cur,
308 struct GNUNET_CONFIGURATION_Handle *cfg)
309{
310 struct GNUNET_ATS_TEST_Operation *o;
311 char *sec_name;
312 char *op_name;
313 char *op;
314 char *type;
315 char *pref;
316 int op_counter = 0;
317 fprintf (stderr, "Parsing episode %u\n",cur->id);
318 GNUNET_asprintf(&sec_name, "episode-%u", cur->id);
319
320 while (1)
321 {
322 /* Load operation */
323 GNUNET_asprintf(&op_name, "op-%u-operation", op_counter);
324 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string(cfg,
325 sec_name, op_name, &op))
326 {
327 GNUNET_free (op_name);
328 break;
329 }
330 o = GNUNET_new (struct GNUNET_ATS_TEST_Operation);
331 /* operations = set_rate, start_send, stop_send, set_preference */
332 if (0 == strcmp (op, "address_add"))
333 {
334 o->type = SOLVER_OP_ADD_ADDRESS;
335 if (GNUNET_SYSERR == load_op_add_address (o, op_counter, sec_name, cfg))
336 {
337 GNUNET_free (o);
338 GNUNET_free (op);
339 GNUNET_free (op_name);
340 return GNUNET_SYSERR;
341 }
342 }
343 else if (0 == strcmp (op, "address_del"))
344 {
345 o->type = SOLVER_OP_DEL_ADDRESS;
346 if (GNUNET_SYSERR == load_op_del_address (o, op_counter, sec_name, cfg))
347 {
348 GNUNET_free (o);
349 GNUNET_free (op);
350 GNUNET_free (op_name);
351 return GNUNET_SYSERR;
352 }
353 }
354 else if (0 == strcmp (op, "start_set_property"))
355 {
356 o->type = SOLVER_OP_START_SET_PROPERTY;
357 }
358 else if (0 == strcmp (op, "stop_set_property"))
359 {
360 o->type = SOLVER_OP_STOP_SET_PROPERTY;
361 }
362 else if (0 == strcmp (op, "start_set_preference"))
363 {
364 o->type = SOLVER_OP_START_SET_PREFERENCE;
365 }
366 else if (0 == strcmp (op, "stop_set_preference"))
367 {
368 o->type = SOLVER_OP_STOP_SET_PREFERENCE;
369 }
370 else
371 {
372 fprintf (stderr, "Invalid operation %u `%s' in episode %u\n",
373 op_counter, op, cur->id);
374 GNUNET_free (op);
375 GNUNET_free (op_name);
376 return GNUNET_SYSERR;
377 }
378 GNUNET_free (op);
379 GNUNET_free (op_name);
380
381 GNUNET_CONTAINER_DLL_insert (cur->head,cur->tail, o);
382 op_counter++;
383 }
384 GNUNET_free (sec_name);
385
386
387#if 0
388 /* Get source */
389 GNUNET_asprintf(&op_name, "op-%u-src", op_counter);
390 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
391 sec_name, op_name, &o->src_id))
392 {
393 fprintf (stderr, "Missing src in operation %u `%s' in episode %u\n",
394 op_counter, op, cur->id);
395 GNUNET_free (op);
396 GNUNET_free (op_name);
397 return GNUNET_SYSERR;
398 }
399 if (o->src_id > (e->num_masters - 1))
400 {
401 fprintf (stderr, "Invalid src %llu in operation %u `%s' in episode %u\n",
402 o->src_id, op_counter, op, cur->id);
403 GNUNET_free (op);
404 GNUNET_free (op_name);
405 return GNUNET_SYSERR;
406 }
407 GNUNET_free (op_name);
408
409 /* Get destination */
410 GNUNET_asprintf(&op_name, "op-%u-dest", op_counter);
411 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
412 sec_name, op_name, &o->dest_id))
413 {
414 fprintf (stderr, "Missing src in operation %u `%s' in episode %u\n",
415 op_counter, op, cur->id);
416 GNUNET_free (op);
417 GNUNET_free (op_name);
418 return GNUNET_SYSERR;
419 }
420 if (o->dest_id > (e->num_slaves - 1))
421 {
422 fprintf (stderr, "Invalid destination %llu in operation %u `%s' in episode %u\n",
423 o->dest_id, op_counter, op, cur->id);
424 GNUNET_free (op);
425 GNUNET_free (op_name);
426 return GNUNET_SYSERR;
427 }
428 GNUNET_free (op_name);
429
430 GNUNET_asprintf(&op_name, "op-%u-type", op_counter);
431 if ( (GNUNET_SYSERR != GNUNET_CONFIGURATION_get_value_string(cfg,
432 sec_name, op_name, &type)) &&
433 ((STOP_SEND != o->type) || (STOP_PREFERENCE != o->type)))
434 {
435 /* Load arguments for set_rate, start_send, set_preference */
436 if (0 == strcmp (type, "constant"))
437 {
438 o->tg_type = GNUNET_ATS_TEST_TG_CONSTANT;
439 }
440 else if (0 == strcmp (type, "linear"))
441 {
442 o->tg_type = GNUNET_ATS_TEST_TG_LINEAR;
443 }
444 else if (0 == strcmp (type, "sinus"))
445 {
446 o->tg_type = GNUNET_ATS_TEST_TG_SINUS;
447 }
448 else if (0 == strcmp (type, "random"))
449 {
450 o->tg_type = GNUNET_ATS_TEST_TG_RANDOM;
451 }
452 else
453 {
454 fprintf (stderr, "Invalid type %u `%s' in episode %u\n",
455 op_counter, op, cur->id);
456 GNUNET_free (type);
457 GNUNET_free (op);
458 GNUNET_free (op_name);
459 return GNUNET_SYSERR;
460 }
461 GNUNET_free (op_name);
462
463 /* Get base rate */
464 GNUNET_asprintf(&op_name, "op-%u-base-rate", op_counter);
465 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
466 sec_name, op_name, &o->base_rate))
467 {
468 fprintf (stderr, "Missing base rate in operation %u `%s' in episode %u\n",
469 op_counter, op, cur->id);
470 GNUNET_free (type);
471 GNUNET_free (op);
472 GNUNET_free (op_name);
473 return GNUNET_SYSERR;
474 }
475 GNUNET_free (op_name);
476
477 /* Get max rate */
478 GNUNET_asprintf(&op_name, "op-%u-max-rate", op_counter);
479 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number (cfg,
480 sec_name, op_name, &o->max_rate))
481 {
482 if ((GNUNET_ATS_TEST_TG_LINEAR == o->tg_type) ||
483 (GNUNET_ATS_TEST_TG_RANDOM == o->tg_type) ||
484 (GNUNET_ATS_TEST_TG_SINUS == o->tg_type))
485 {
486 fprintf (stderr, "Missing max rate in operation %u `%s' in episode %u\n",
487 op_counter, op, cur->id);
488 GNUNET_free (type);
489 GNUNET_free (op_name);
490 GNUNET_free (op);
491 return GNUNET_SYSERR;
492 }
493 }
494 GNUNET_free (op_name);
495
496 /* Get period */
497 GNUNET_asprintf(&op_name, "op-%u-period", op_counter);
498 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_time (cfg,
499 sec_name, op_name, &o->period))
500 {
501 o->period = cur->duration;
502 }
503 GNUNET_free (op_name);
504
505 if (START_PREFERENCE == o->type)
506 {
507 /* Get frequency */
508 GNUNET_asprintf(&op_name, "op-%u-frequency", op_counter);
509 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_time (cfg,
510 sec_name, op_name, &o->frequency))
511 {
512 fprintf (stderr, "Missing frequency in operation %u `%s' in episode %u\n",
513 op_counter, op, cur->id);
514 GNUNET_free (type);
515 GNUNET_free (op_name);
516 GNUNET_free (op);
517 return GNUNET_SYSERR;
518 }
519 GNUNET_free (op_name);
520
521 /* Get preference */
522 GNUNET_asprintf(&op_name, "op-%u-pref", op_counter);
523 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string (cfg,
524 sec_name, op_name, &pref))
525 {
526 fprintf (stderr, "Missing preference in operation %u `%s' in episode %u\n",
527 op_counter, op, cur->id);
528 GNUNET_free (type);
529 GNUNET_free (op_name);
530 GNUNET_free (op);
531 GNUNET_free_non_null (pref);
532 return GNUNET_SYSERR;
533 }
534
535 if (0 == strcmp(pref, "bandwidth"))
536 o->pref_type = GNUNET_ATS_PREFERENCE_BANDWIDTH;
537 else if (0 == strcmp(pref, "latency"))
538 o->pref_type = GNUNET_ATS_PREFERENCE_LATENCY;
539 else
540 {
541 fprintf (stderr, "Invalid preference in operation %u `%s' in episode %u\n",
542 op_counter, op, cur->id);
543 GNUNET_free (type);
544 GNUNET_free (op_name);
545 GNUNET_free (op);
546 GNUNET_free (pref);
547 GNUNET_free_non_null (pref);
548 return GNUNET_SYSERR;
549 }
550 GNUNET_free (pref);
551 GNUNET_free (op_name);
552 }
553 }
554
555 /* Safety checks */
556 if ((GNUNET_ATS_TEST_TG_LINEAR == o->tg_type) ||
557 (GNUNET_ATS_TEST_TG_SINUS == o->tg_type))
558 {
559 if ((o->max_rate - o->base_rate) > o->base_rate)
560 {
561 /* This will cause an underflow */
562 GNUNET_break (0);
563 }
564 fprintf (stderr, "Selected max rate and base rate cannot be used for desired traffic form!\n");
565 }
566
567 if ((START_SEND == o->type) || (START_PREFERENCE == o->type))
568 fprintf (stderr, "Found operation %u in episode %u: %s [%llu]->[%llu] == %s, %llu -> %llu in %s\n",
569 op_counter, cur->id, print_op (o->type), o->src_id,
570 o->dest_id, (NULL != type) ? type : "",
571 o->base_rate, o->max_rate,
572 GNUNET_STRINGS_relative_time_to_string (o->period, GNUNET_YES));
573 else
574 fprintf (stderr, "Found operation %u in episode %u: %s [%llu]->[%llu]\n",
575 op_counter, cur->id, print_op (o->type), o->src_id, o->dest_id);
576
577 GNUNET_free_non_null (type);
578 GNUNET_free (op);
579#endif
580
581 return GNUNET_OK;
582}
583
584static int
585load_episodes (struct Experiment *e, struct GNUNET_CONFIGURATION_Handle *cfg)
586{
587 int e_counter = 0;
588 char *sec_name;
589 struct GNUNET_TIME_Relative e_duration;
590 struct Episode *cur;
591 struct Episode *last;
592
593 e_counter = 0;
594 last = NULL;
595 while (1)
596 {
597 GNUNET_asprintf(&sec_name, "episode-%u", e_counter);
598 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_time(cfg,
599 sec_name, "duration", &e_duration))
600 {
601 GNUNET_free (sec_name);
602 break;
603 }
604
605 cur = GNUNET_new (struct Episode);
606 cur->duration = e_duration;
607 cur->id = e_counter;
608
609 if (GNUNET_OK != load_episode (e, cur, cfg))
610 {
611 GNUNET_free (sec_name);
612 GNUNET_free (cur);
613 return GNUNET_SYSERR;
614 }
615
616 fprintf (stderr, "Found episode %u with duration %s \n",
617 e_counter,
618 GNUNET_STRINGS_relative_time_to_string(cur->duration, GNUNET_YES));
619
620 /* Update experiment */
621 e->num_episodes ++;
622 e->total_duration = GNUNET_TIME_relative_add(e->total_duration, cur->duration);
623 /* Put in linked list */
624 if (NULL == last)
625 e->start = cur;
626 else
627 last->next = cur;
628
629 GNUNET_free (sec_name);
630 e_counter ++;
631 last = cur;
632 }
633 return e_counter;
634}
635
636static void
637timeout_experiment (void *cls, const struct GNUNET_SCHEDULER_TaskContext* tc)
638{
639 struct Experiment *e = cls;
640 e->experiment_timeout_task = GNUNET_SCHEDULER_NO_TASK;
641 fprintf (stderr, "Experiment timeout!\n");
642
643 if (GNUNET_SCHEDULER_NO_TASK != e->episode_timeout_task)
644 {
645 GNUNET_SCHEDULER_cancel (e->episode_timeout_task);
646 e->episode_timeout_task = GNUNET_SCHEDULER_NO_TASK;
647 }
648
649 e->e_done_cb (e, GNUNET_TIME_absolute_get_duration(e->start_time),
650 GNUNET_SYSERR);
651}
652
653static void
654enforce_add_address (struct GNUNET_ATS_TEST_Operation *op)
655{
656 /*
657 struct BenchmarkPeer *peer;
658 struct BenchmarkPartner *partner;
659
660 peer = GNUNET_ATS_TEST_get_peer (op->src_id);
661 if (NULL == peer)
662 {
663 GNUNET_break (0);
664 return;
665 }
666
667 partner = GNUNET_ATS_TEST_get_partner (op->src_id, op->dest_id);
668 if (NULL == partner)
669 {
670 GNUNET_break (0);
671 return;
672 }
673
674 fprintf (stderr, "Found master %llu slave %llu\n",op->src_id, op->dest_id);
675
676 if (NULL != partner->tg)
677 {
678 fprintf (stderr, "Stopping traffic between master %llu slave %llu\n",op->src_id, op->dest_id);
679 GNUNET_ATS_TEST_generate_traffic_stop(partner->tg);
680 partner->tg = NULL;
681 }
682
683 partner->tg = GNUNET_ATS_TEST_generate_traffic_start(peer, partner,
684 op->tg_type, op->base_rate, op->max_rate, op->period,
685 GNUNET_TIME_UNIT_FOREVER_REL);
686 */
687}
688
689static void
690enforce_del_address (struct GNUNET_ATS_TEST_Operation *op)
691{
692 /*
693 struct BenchmarkPartner *p;
694 p = GNUNET_ATS_TEST_get_partner (op->src_id, op->dest_id);
695 if (NULL == p)
696 {
697 GNUNET_break (0);
698 return;
699 }
700
701 fprintf (stderr, "Found master %llu slave %llu\n",op->src_id, op->dest_id);
702
703 if (NULL != p->tg)
704 {
705 fprintf (stderr, "Stopping traffic between master %llu slave %llu\n",
706 op->src_id, op->dest_id);
707 GNUNET_ATS_TEST_generate_traffic_stop(p->tg);
708 p->tg = NULL;
709 }
710 */
711}
712
713static void
714enforce_start_property (struct GNUNET_ATS_TEST_Operation *op)
715{
716
717}
718
719static void
720enforce_stop_property (struct GNUNET_ATS_TEST_Operation *op)
721{
722
723}
724
725static void
726enforce_start_preference (struct GNUNET_ATS_TEST_Operation *op)
727{
728 /*
729 struct BenchmarkPeer *peer;
730 struct BenchmarkPartner *partner;
731
732 peer = GNUNET_ATS_TEST_get_peer (op->src_id);
733 if (NULL == peer)
734 {
735 GNUNET_break (0);
736 return;
737 }
738
739 partner = GNUNET_ATS_TEST_get_partner (op->src_id, op->dest_id);
740 if (NULL == partner)
741 {
742 GNUNET_break (0);
743 return;
744 }
745
746 fprintf (stderr, "Found master %llu slave %llu\n",op->src_id, op->dest_id);
747
748 if (NULL != partner->pg)
749 {
750 fprintf (stderr, "Stopping traffic between master %llu slave %llu\n",
751 op->src_id, op->dest_id);
752 GNUNET_ATS_TEST_generate_preferences_stop(partner->pg);
753 partner->pg = NULL;
754 }
755
756 partner->pg = GNUNET_ATS_TEST_generate_preferences_start(peer, partner,
757 op->tg_type, op->base_rate, op->max_rate, op->period, op->frequency,
758 op->pref_type);
759 */
760}
761
762static void
763enforce_stop_preference (struct GNUNET_ATS_TEST_Operation *op)
764{
765 /*
766 struct BenchmarkPartner *p;
767 p = GNUNET_ATS_TEST_get_partner (op->src_id, op->dest_id);
768 if (NULL == p)
769 {
770 GNUNET_break (0);
771 return;
772 }
773
774 fprintf (stderr, "Found master %llu slave %llu\n",op->src_id, op->dest_id);
775
776 if (NULL != p->pg)
777 {
778 fprintf (stderr, "Stopping preference between master %llu slave %llu\n",
779 op->src_id, op->dest_id);
780 GNUNET_ATS_TEST_generate_preferences_stop (p->pg);
781 p->pg = NULL;
782 }
783 */
784}
785
786static void enforce_episode (struct Episode *ep)
787{
788 struct GNUNET_ATS_TEST_Operation *cur;
789 for (cur = ep->head; NULL != cur; cur = cur->next)
790 {
791 switch (cur->type) {
792 case SOLVER_OP_ADD_ADDRESS:
793 fprintf (stderr, "Enforcing operation: %s [%llu:%llu]\n",
794 print_op (cur->type), cur->peer_id, cur->address_id);
795 enforce_add_address (cur);
796 break;
797 case SOLVER_OP_DEL_ADDRESS:
798 fprintf (stderr, "Enforcing operation: %s [%llu:%llu]\n",
799 print_op (cur->type), cur->peer_id, cur->address_id);
800 enforce_del_address (cur);
801 break;
802 case SOLVER_OP_START_SET_PROPERTY:
803 fprintf (stderr, "Enforcing operation: %s [%llu:%llu] == %llu\n",
804 print_op (cur->type), cur->peer_id, cur->address_id, cur->base_rate);
805 enforce_start_property (cur);
806 break;
807 case SOLVER_OP_STOP_SET_PROPERTY:
808 fprintf (stderr, "Enforcing operation: %s [%llu:%llu] == %llu\n",
809 print_op (cur->type), cur->peer_id, cur->address_id, cur->base_rate);
810 enforce_stop_property (cur);
811 break;
812 case SOLVER_OP_START_SET_PREFERENCE:
813 fprintf (stderr, "Enforcing operation: %s [%llu:%llu] == %llu\n",
814 print_op (cur->type), cur->peer_id, cur->address_id, cur->base_rate);
815 enforce_start_preference (cur);
816 break;
817 case SOLVER_OP_STOP_SET_PREFERENCE:
818 fprintf (stderr, "Enforcing operation: %s [%llu:%llu] == %llu\n",
819 print_op (cur->type), cur->peer_id, cur->address_id, cur->base_rate);
820 enforce_stop_preference (cur);
821 break;
822 default:
823 break;
824 }
825 }
826}
827
828static void
829timeout_episode (void *cls, const struct GNUNET_SCHEDULER_TaskContext* tc)
830{
831 struct Experiment *e = cls;
832 e->episode_timeout_task = GNUNET_SCHEDULER_NO_TASK;
833 if (NULL != e->ep_done_cb)
834 e->ep_done_cb (e->cur);
835
836 /* Scheduling next */
837 e->cur = e->cur->next;
838 if (NULL == e->cur)
839 {
840 /* done */
841 fprintf (stderr, "Last episode done!\n");
842 if (GNUNET_SCHEDULER_NO_TASK != e->experiment_timeout_task)
843 {
844 GNUNET_SCHEDULER_cancel (e->experiment_timeout_task);
845 e->experiment_timeout_task = GNUNET_SCHEDULER_NO_TASK;
846 }
847 e->e_done_cb (e, GNUNET_TIME_absolute_get_duration(e->start_time), GNUNET_OK);
848 return;
849 }
850
851 fprintf (stderr, "Running episode %u with timeout %s\n",
852 e->cur->id,
853 GNUNET_STRINGS_relative_time_to_string(e->cur->duration, GNUNET_YES));
854 e->episode_timeout_task = GNUNET_SCHEDULER_add_delayed (e->cur->duration,
855 &timeout_episode, e);
856 enforce_episode(e->cur);
857
858
859}
860
861
862void
863GNUNET_ATS_solvers_experimentation_run (struct Experiment *e,
864 GNUNET_ATS_TESTING_EpisodeDoneCallback ep_done_cb,
865 GNUNET_ATS_TESTING_ExperimentDoneCallback e_done_cb)
866{
867 fprintf (stderr, "Running experiment `%s' with timeout %s\n", e->name,
868 GNUNET_STRINGS_relative_time_to_string(e->max_duration, GNUNET_YES));
869 e->e_done_cb = e_done_cb;
870 e->ep_done_cb = ep_done_cb;
871 e->start_time = GNUNET_TIME_absolute_get();
872
873 /* Start total time out */
874 e->experiment_timeout_task = GNUNET_SCHEDULER_add_delayed (e->max_duration,
875 &timeout_experiment, e);
876
877
878 /* Start */
879 if (NULL == e->start)
880 {
881 GNUNET_break (0);
882 return;
883 }
884 e->cur = e->start;
885 fprintf (stderr, "Running episode %u with timeout %s\n",
886 e->cur->id,
887 GNUNET_STRINGS_relative_time_to_string(e->cur->duration, GNUNET_YES));
888 e->episode_timeout_task = GNUNET_SCHEDULER_add_delayed (e->cur->duration,
889 &timeout_episode, e);
890 enforce_episode(e->cur);
891
892}
893
894
895struct Experiment *
896GNUNET_ATS_solvers_experimentation_load (char *filename)
897{
898 struct Experiment *e;
899 struct GNUNET_CONFIGURATION_Handle *cfg;
900 e = NULL;
901
902 cfg = GNUNET_CONFIGURATION_create();
903 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_load (cfg, filename))
904 {
905 fprintf (stderr, "Failed to load `%s'\n", filename);
906 GNUNET_CONFIGURATION_destroy (cfg);
907 return NULL;
908 }
909
910 e = create_experiment ();
911
912 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_string(cfg, "experiment",
913 "name", &e->name))
914 {
915 fprintf (stderr, "Invalid %s", "name");
916 free_experiment (e);
917 return NULL;
918 }
919 else
920 fprintf (stderr, "Experiment name: `%s'\n", e->name);
921
922 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_filename (cfg, "experiment",
923 "cfg_file", &e->cfg_file))
924 {
925 fprintf (stderr, "Invalid %s", "cfg_file");
926 free_experiment (e);
927 return NULL;
928 }
929 else
930 {
931 fprintf (stderr, "Experiment name: `%s'\n", e->cfg_file);
932 e->cfg = GNUNET_CONFIGURATION_create();
933 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_load (e->cfg, e->cfg_file))
934 {
935 fprintf (stderr, "Invalid configuration %s", "cfg_file");
936 free_experiment (e);
937 return NULL;
938 }
939
940 }
941
942 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number(cfg, "experiment",
943 "masters", &e->num_masters))
944 {
945 fprintf (stderr, "Invalid %s", "masters");
946 free_experiment (e);
947 return NULL;
948 }
949 else
950 fprintf (stderr, "Experiment masters: `%llu'\n",
951 e->num_masters);
952
953 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_number(cfg, "experiment",
954 "slaves", &e->num_slaves))
955 {
956 fprintf (stderr, "Invalid %s", "slaves");
957 free_experiment (e);
958 return NULL;
959 }
960 else
961 fprintf (stderr, "Experiment slaves: `%llu'\n",
962 e->num_slaves);
963
964 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_time(cfg, "experiment",
965 "log_freq", &e->log_freq))
966 {
967 fprintf (stderr, "Invalid %s", "log_freq");
968 free_experiment (e);
969 return NULL;
970 }
971 else
972 fprintf (stderr, "Experiment logging frequency: `%s'\n",
973 GNUNET_STRINGS_relative_time_to_string (e->log_freq, GNUNET_YES));
974
975 if (GNUNET_SYSERR == GNUNET_CONFIGURATION_get_value_time(cfg, "experiment",
976 "max_duration", &e->max_duration))
977 {
978 fprintf (stderr, "Invalid %s", "max_duration");
979 free_experiment (e);
980 return NULL;
981 }
982 else
983 fprintf (stderr, "Experiment duration: `%s'\n",
984 GNUNET_STRINGS_relative_time_to_string (e->max_duration, GNUNET_YES));
985
986 load_episodes (e, cfg);
987 fprintf (stderr, "Loaded %u episodes with total duration %s\n",
988 e->num_episodes,
989 GNUNET_STRINGS_relative_time_to_string (e->total_duration, GNUNET_YES));
990
991 GNUNET_CONFIGURATION_destroy (cfg);
992 return e;
993}
994
995void
996GNUNET_ATS_solvers_experimentation_stop (struct Experiment *e)
997{
998 if (GNUNET_SCHEDULER_NO_TASK != e->experiment_timeout_task)
999 {
1000 GNUNET_SCHEDULER_cancel (e->experiment_timeout_task);
1001 e->experiment_timeout_task = GNUNET_SCHEDULER_NO_TASK;
1002 }
1003 if (GNUNET_SCHEDULER_NO_TASK != e->episode_timeout_task)
1004 {
1005 GNUNET_SCHEDULER_cancel (e->episode_timeout_task);
1006 e->episode_timeout_task = GNUNET_SCHEDULER_NO_TASK;
1007 }
1008 if (NULL != e->cfg)
1009 {
1010 GNUNET_CONFIGURATION_destroy(e->cfg);
1011 e->cfg = NULL;
1012 }
1013 free_experiment (e);
1014}
1015
1016/**
1017 * Logging
1018 */
1019
1020
1021/**
1022 * Solver
1023 */
1024
1025struct GNUNET_ATS_TESTING_SolverHandle
1026{
1027 char * plugin;
1028 struct GNUNET_ATS_PluginEnvironment env;
1029 void *solver;
1030 struct GNUNET_CONTAINER_MultiPeerMap *addresses;
1031};
1032
1033enum GNUNET_ATS_Solvers
1034{
1035 GNUNET_ATS_SOLVER_PROPORTIONAL,
1036 GNUNET_ATS_SOLVER_MLP,
1037 GNUNET_ATS_SOLVER_RIL,
1038};
1039
1040void
1041GNUNET_ATS_solvers_solver_stop (struct GNUNET_ATS_TESTING_SolverHandle *sh)
1042{
1043 GNUNET_PLUGIN_unload (sh->plugin, sh->solver);
1044 GNUNET_CONTAINER_multipeermap_destroy(sh->addresses);
1045 GNUNET_free (sh->plugin);
1046 GNUNET_free (sh);
1047}
1048
1049/**
1050 * Load quotas for networks from configuration
1051 *
1052 * @param cfg configuration handle
1053 * @param out_dest where to write outbound quotas
1054 * @param in_dest where to write inbound quotas
1055 * @param dest_length length of inbound and outbound arrays
1056 * @return number of networks loaded
1057 */
1058unsigned int
1059GNUNET_ATS_solvers_load_quotas (const struct GNUNET_CONFIGURATION_Handle *cfg,
1060 unsigned long long *out_dest,
1061 unsigned long long *in_dest,
1062 int dest_length)
1063{
1064 char *network_str[GNUNET_ATS_NetworkTypeCount] = GNUNET_ATS_NetworkTypeString;
1065 char * entry_in = NULL;
1066 char * entry_out = NULL;
1067 char * quota_out_str;
1068 char * quota_in_str;
1069 int c;
1070 int res;
1071
1072 for (c = 0; (c < GNUNET_ATS_NetworkTypeCount) && (c < dest_length); c++)
1073 {
1074 in_dest[c] = 0;
1075 out_dest[c] = 0;
1076 GNUNET_asprintf (&entry_out, "%s_QUOTA_OUT", network_str[c]);
1077 GNUNET_asprintf (&entry_in, "%s_QUOTA_IN", network_str[c]);
1078
1079 /* quota out */
1080 if (GNUNET_OK == GNUNET_CONFIGURATION_get_value_string(cfg, "ats", entry_out, &quota_out_str))
1081 {
1082 res = GNUNET_NO;
1083 if (0 == strcmp(quota_out_str, BIG_M_STRING))
1084 {
1085 out_dest[c] = GNUNET_ATS_MaxBandwidth;
1086 res = GNUNET_YES;
1087 }
1088 if ((GNUNET_NO == res) && (GNUNET_OK == GNUNET_STRINGS_fancy_size_to_bytes (quota_out_str, &out_dest[c])))
1089 res = GNUNET_YES;
1090 if ((GNUNET_NO == res) && (GNUNET_OK == GNUNET_CONFIGURATION_get_value_number (cfg, "ats", entry_out, &out_dest[c])))
1091 res = GNUNET_YES;
1092
1093 if (GNUNET_NO == res)
1094 {
1095 GNUNET_log (GNUNET_ERROR_TYPE_ERROR, _("Could not load quota for network `%s': `%s', assigning default bandwidth %llu\n"),
1096 network_str[c], quota_out_str, GNUNET_ATS_DefaultBandwidth);
1097 out_dest[c] = GNUNET_ATS_DefaultBandwidth;
1098 }
1099 else
1100 {
1101 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, _("Outbound quota configure for network `%s' is %llu\n"),
1102 network_str[c], out_dest[c]);
1103 }
1104 GNUNET_free (quota_out_str);
1105 }
1106 else
1107 {
1108 GNUNET_log (GNUNET_ERROR_TYPE_WARNING, _("No outbound quota configured for network `%s', assigning default bandwidth %llu\n"),
1109 network_str[c], GNUNET_ATS_DefaultBandwidth);
1110 out_dest[c] = GNUNET_ATS_DefaultBandwidth;
1111 }
1112
1113 /* quota in */
1114 if (GNUNET_OK == GNUNET_CONFIGURATION_get_value_string(cfg, "ats", entry_in, &quota_in_str))
1115 {
1116 res = GNUNET_NO;
1117 if (0 == strcmp(quota_in_str, BIG_M_STRING))
1118 {
1119 in_dest[c] = GNUNET_ATS_MaxBandwidth;
1120 res = GNUNET_YES;
1121 }
1122 if ((GNUNET_NO == res) && (GNUNET_OK == GNUNET_STRINGS_fancy_size_to_bytes (quota_in_str, &in_dest[c])))
1123 res = GNUNET_YES;
1124 if ((GNUNET_NO == res) && (GNUNET_OK == GNUNET_CONFIGURATION_get_value_number (cfg, "ats", entry_in, &in_dest[c])))
1125 res = GNUNET_YES;
1126
1127 if (GNUNET_NO == res)
1128 {
1129 GNUNET_log (GNUNET_ERROR_TYPE_ERROR, _("Could not load quota for network `%s': `%s', assigning default bandwidth %llu\n"),
1130 network_str[c], quota_in_str, GNUNET_ATS_DefaultBandwidth);
1131 in_dest[c] = GNUNET_ATS_DefaultBandwidth;
1132 }
1133 else
1134 {
1135 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, _("Inbound quota configured for network `%s' is %llu\n"),
1136 network_str[c], in_dest[c]);
1137 }
1138 GNUNET_free (quota_in_str);
1139 }
1140 else
1141 {
1142 GNUNET_log (GNUNET_ERROR_TYPE_WARNING, _("No outbound quota configure for network `%s', assigning default bandwidth %llu\n"),
1143 network_str[c], GNUNET_ATS_DefaultBandwidth);
1144 out_dest[c] = GNUNET_ATS_DefaultBandwidth;
1145 }
1146 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Loaded quota for network `%s' (in/out): %llu %llu\n", network_str[c], in_dest[c], out_dest[c]);
1147 GNUNET_free (entry_out);
1148 GNUNET_free (entry_in);
1149 }
1150 return GNUNET_ATS_NetworkTypeCount;
1151}
1152
1153/**
1154 * Information callback for the solver
1155 *
1156 * @param op the solver operation
1157 * @param stat status of the solver operation
1158 * @param add additional solver information
1159 */
1160static void
1161solver_info_cb (void *cls,
1162 enum GAS_Solver_Operation op,
1163 enum GAS_Solver_Status stat,
1164 enum GAS_Solver_Additional_Information add)
1165{
1166 char *add_info;
1167 switch (add) {
1168 case GAS_INFO_NONE:
1169 add_info = "GAS_INFO_NONE";
1170 break;
1171 case GAS_INFO_FULL:
1172 add_info = "GAS_INFO_MLP_FULL";
1173 break;
1174 case GAS_INFO_UPDATED:
1175 add_info = "GAS_INFO_MLP_UPDATED";
1176 break;
1177 case GAS_INFO_PROP_ALL:
1178 add_info = "GAS_INFO_PROP_ALL";
1179 break;
1180 case GAS_INFO_PROP_SINGLE:
1181 add_info = "GAS_INFO_PROP_SINGLE";
1182 break;
1183 default:
1184 add_info = "INVALID";
1185 break;
1186 }
1187
1188 switch (op)
1189 {
1190 case GAS_OP_SOLVE_START:
1191 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1192 "Solver notifies `%s' with result `%s' `%s'\n", "GAS_OP_SOLVE_START",
1193 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL", add_info);
1194 return;
1195 case GAS_OP_SOLVE_STOP:
1196 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1197 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_STOP",
1198 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL", add_info);
1199 return;
1200
1201 case GAS_OP_SOLVE_SETUP_START:
1202 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1203 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_SETUP_START",
1204 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1205 return;
1206
1207 case GAS_OP_SOLVE_SETUP_STOP:
1208 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1209 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_SETUP_STOP",
1210 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1211 return;
1212
1213 case GAS_OP_SOLVE_MLP_LP_START:
1214 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1215 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_LP_START",
1216 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1217 return;
1218 case GAS_OP_SOLVE_MLP_LP_STOP:
1219 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1220 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_LP_STOP",
1221 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1222 return;
1223
1224 case GAS_OP_SOLVE_MLP_MLP_START:
1225 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1226 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_MLP_START",
1227 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1228 return;
1229 case GAS_OP_SOLVE_MLP_MLP_STOP:
1230 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1231 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_MLP_STOP",
1232 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1233 return;
1234 case GAS_OP_SOLVE_UPDATE_NOTIFICATION_START:
1235 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1236 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_UPDATE_NOTIFICATION_START",
1237 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1238 return;
1239 case GAS_OP_SOLVE_UPDATE_NOTIFICATION_STOP:
1240 GNUNET_log(GNUNET_ERROR_TYPE_INFO,
1241 "Solver notifies `%s' with result `%s'\n", "GAS_OP_SOLVE_UPDATE_NOTIFICATION_STOP",
1242 (GAS_STAT_SUCCESS == stat) ? "SUCCESS" : "FAIL");
1243 return;
1244 default:
1245 break;
1246 }
1247}
1248
1249static void
1250solver_bandwidth_changed_cb (void *cls, struct ATS_Address *address)
1251{
1252 if ( (0 == ntohl (address->assigned_bw_out.value__)) &&
1253 (0 == ntohl (address->assigned_bw_in.value__)) )
1254 return;
1255
1256 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
1257 "Bandwidth changed addresses %s %p to %u Bps out / %u Bps in\n",
1258 GNUNET_i2s (&address->peer),
1259 address,
1260 (unsigned int) ntohl (address->assigned_bw_out.value__),
1261 (unsigned int) ntohl (address->assigned_bw_in.value__));
1262 /*if (GNUNET_YES == ph.bulk_running)
1263 GNUNET_break (0);*/
1264 return;
1265}
1266
1267const double *
1268get_preferences_cb (void *cls, const struct GNUNET_PeerIdentity *id)
1269{
1270 return GAS_normalization_get_preferences_by_peer (id);
1271}
1272
1273
1274const double *
1275get_property_cb (void *cls, const struct ATS_Address *address)
1276{
1277 return GAS_normalization_get_properties ((struct ATS_Address *) address);
1278}
1279
1280static void
1281normalized_property_changed_cb (void *cls, struct ATS_Address *peer,
1282 uint32_t type, double prop_rel)
1283{
1284 /* TODO */
1285}
1286
1287
1288struct GNUNET_ATS_TESTING_SolverHandle *
1289GNUNET_ATS_solvers_solver_start (enum GNUNET_ATS_Solvers type)
1290{
1291 struct GNUNET_ATS_TESTING_SolverHandle *sh;
1292 char * solver_str;
1293 unsigned long long quotas_in[GNUNET_ATS_NetworkTypeCount];
1294 unsigned long long quotas_out[GNUNET_ATS_NetworkTypeCount];
1295
1296 switch (type) {
1297 case GNUNET_ATS_SOLVER_PROPORTIONAL:
1298 solver_str = "proportional";
1299 break;
1300 case GNUNET_ATS_SOLVER_MLP:
1301 solver_str = "mlp";
1302 break;
1303 case GNUNET_ATS_SOLVER_RIL:
1304 solver_str = "ril";
1305 break;
1306 default:
1307 GNUNET_break (0);
1308 return NULL;
1309 break;
1310 }
1311
1312 sh = GNUNET_new (struct GNUNET_ATS_TESTING_SolverHandle);
1313 GNUNET_asprintf (&sh->plugin, "libgnunet_plugin_ats_%s", solver_str);
1314
1315 /* setup environment */
1316 sh->env.cfg = e->cfg;
1317 sh->env.stats = GNUNET_STATISTICS_create ("ats", e->cfg);
1318 sh->env.addresses = GNUNET_CONTAINER_multipeermap_create (128, GNUNET_NO);
1319 sh->env.bandwidth_changed_cb = &solver_bandwidth_changed_cb;
1320 sh->env.get_preferences = &get_preferences_cb;
1321 sh->env.get_property = &get_property_cb;
1322 sh->env.network_count = GNUNET_ATS_NetworkTypeCount;
1323 sh->env.info_cb = &solver_info_cb;
1324 sh->env.info_cb_cls = NULL;
1325
1326 /* start normalization */
1327 GAS_normalization_start (NULL, NULL, &normalized_property_changed_cb, NULL );
1328
1329 /* load quotas */
1330 if (GNUNET_ATS_NetworkTypeCount != GNUNET_ATS_solvers_load_quotas (e->cfg,
1331 quotas_out, quotas_in, GNUNET_ATS_NetworkTypeCount))
1332 {
1333 GNUNET_break(0);
1334 GNUNET_free (sh->plugin);
1335 GNUNET_free (sh);
1336 end_now ();
1337 return NULL;
1338 }
1339
1340 sh->solver = GNUNET_PLUGIN_load (sh->plugin, &sh->env);
1341 if (NULL == sh->solver)
1342 {
1343 fprintf (stderr, "Failed to load solver `%s'\n", sh->plugin);
1344 GNUNET_break(0);
1345 GNUNET_free (sh->plugin);
1346 GNUNET_free (sh);
1347 end_now ();
1348 return NULL;
1349 }
1350
1351 sh->addresses = GNUNET_CONTAINER_multipeermap_create (10, GNUNET_NO);
1352
1353 return sh;
1354}
1355
1356static void
1357done ()
1358{
1359 /* Clean up experiment */
1360 GNUNET_ATS_solvers_experimentation_stop (e);
1361 e = NULL;
1362
1363 /* Shutdown */
1364 end_now();
1365
1366}
1367
1368static void
1369experiment_done_cb (struct Experiment *e, struct GNUNET_TIME_Relative duration,int success)
1370{
1371 if (GNUNET_OK == success)
1372 GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Experiment done successful in %s\n",
1373 GNUNET_STRINGS_relative_time_to_string (duration, GNUNET_YES));
1374 else
1375 GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Experiment failed \n");
1376
1377 /* Stop logging */
1378 // GNUNET_ATS_TEST_logging_stop (l);
1379
1380 /* Stop traffic generation */
1381 // GNUNET_ATS_TEST_generate_traffic_stop_all();
1382
1383 /* Stop all preference generations */
1384 // GNUNET_ATS_TEST_generate_preferences_stop_all ();
1385
1386 /*
1387 evaluate (duration);
1388 if (opt_log)
1389 GNUNET_ATS_TEST_logging_write_to_file(l, opt_exp_file, opt_plot);
1390
1391 if (NULL != l)
1392 {
1393 GNUNET_ATS_TEST_logging_stop (l);
1394 GNUNET_ATS_TEST_logging_clean_up (l);
1395 l = NULL;
1396 }
1397 */
1398 GNUNET_SCHEDULER_add_now (&done, NULL);
1399}
1400
1401static void
1402episode_done_cb (struct Episode *ep)
1403{
1404 GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Episode %u done\n", ep->id);
1405}
1406
1407
1408
1409/**
1410 * Do shutdown
1411 */
1412static void
1413end_now ()
1414{
1415 if (NULL != e)
1416 {
1417 GNUNET_ATS_solvers_experimentation_stop (e);
1418 e = NULL;
1419 }
1420 if (NULL != sh)
1421 {
1422 GNUNET_ATS_solvers_solver_stop (sh);
1423 sh = NULL;
1424 }
1425}
1426
1427static void
1428run (void *cls, char * const *args, const char *cfgfile,
1429 const struct GNUNET_CONFIGURATION_Handle *cfg)
1430{
1431 enum GNUNET_ATS_Solvers solver;
1432
1433 if (NULL == opt_exp_file)
1434 {
1435 fprintf (stderr, "No experiment given ...\n");
1436 res = 1;
1437 end_now ();
1438 return;
1439 }
1440
1441 if (NULL == opt_solver)
1442 {
1443 fprintf (stderr, "No solver given ...\n");
1444 res = 1;
1445 end_now ();
1446 return;
1447 }
1448
1449 if (0 == strcmp(opt_solver, "mlp"))
1450 {
1451 solver = GNUNET_ATS_SOLVER_MLP;
1452 }
1453 else if (0 == strcmp(opt_solver, "proportional"))
1454 {
1455 solver = GNUNET_ATS_SOLVER_PROPORTIONAL;
1456 }
1457 else if (0 == strcmp(opt_solver, "ril"))
1458 {
1459 solver = GNUNET_ATS_SOLVER_RIL;
1460 }
1461 else
1462 {
1463 fprintf (stderr, "No solver given ...");
1464 res = 1;
1465 end_now ();
1466 return;
1467 }
1468
1469 /* load experiment */
1470 e = GNUNET_ATS_solvers_experimentation_load (opt_exp_file);
1471 if (NULL == e)
1472 {
1473 fprintf (stderr, "Failed to load experiment ...\n");
1474 res = 1;
1475 end_now ();
1476 return;
1477 }
1478
1479
1480 /* load solver */
1481 sh = GNUNET_ATS_solvers_solver_start (solver);
1482 if (NULL == sh)
1483 {
1484 fprintf (stderr, "Failed to start solver ...\n");
1485 end_now ();
1486 res = 1;
1487 return;
1488 }
1489
1490 /* start logging */
1491
1492
1493 /* run experiment */
1494 GNUNET_ATS_solvers_experimentation_run (e, episode_done_cb,
1495 experiment_done_cb);
1496
1497 /* WAIT */
1498}
1499
1500
1501/**
1502 * Main function of the benchmark
1503 *
1504 * @param argc argument count
1505 * @param argv argument values
1506 */
1507int
1508main (int argc, char *argv[])
1509{
1510 opt_exp_file = NULL;
1511 opt_solver = NULL;
1512 opt_log = GNUNET_NO;
1513 opt_plot = GNUNET_NO;
1514
1515 res = 0;
1516
1517 static struct GNUNET_GETOPT_CommandLineOption options[] =
1518 {
1519 { 's', "solver", NULL,
1520 gettext_noop ("solver to use"),
1521 1, &GNUNET_GETOPT_set_string, &opt_solver},
1522 { 'e', "experiment", NULL,
1523 gettext_noop ("experiment to use"),
1524 1, &GNUNET_GETOPT_set_string, &opt_exp_file},
1525 { 'e', "experiment", NULL,
1526 gettext_noop ("experiment to use"),
1527 1, &GNUNET_GETOPT_set_one, &opt_verbose},
1528 GNUNET_GETOPT_OPTION_END
1529 };
1530
1531 GNUNET_PROGRAM_run (argc, argv, argv[0], NULL, options, &run, argv[0]);
1532
1533 return res;
1534}
1535/* end of file ats-testing-experiment.c*/
1536
diff --git a/src/ats/gnunet-ats-solver-eval.h b/src/ats/gnunet-ats-solver-eval.h
new file mode 100644
index 000000000..0b30fead7
--- /dev/null
+++ b/src/ats/gnunet-ats-solver-eval.h
@@ -0,0 +1,930 @@
1/*
2 This file is part of GNUnet.
3 (C) 2010-2013 Christian Grothoff (and other contributing authors)
4
5 GNUnet is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; either version 3, or (at your
8 option) any later version.
9
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with GNUnet; see the file COPYING. If not, write to the
17 Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 Boston, MA 02111-1307, USA.
19 */
20/**
21 * @file ats-tests/ats-testing-experiment.c
22 * @brief ats benchmark: controlled experiment execution
23 * @author Christian Grothoff
24 * @author Matthias Wachs
25 */
26#include "platform.h"
27#include "gnunet_util_lib.h"
28#include "gnunet_ats_plugin.h"
29#include "gnunet_ats_service.h"
30#include "gnunet-service-ats_addresses.h"
31#include "gnunet-service-ats_normalization.h"
32#include "test_ats_api_common.h"
33
34enum GeneratorType
35{
36 GNUNET_ATS_TEST_TG_LINEAR,
37 GNUNET_ATS_TEST_TG_CONSTANT,
38 GNUNET_ATS_TEST_TG_RANDOM,
39 GNUNET_ATS_TEST_TG_SINUS
40};
41
42
43enum OperationType
44{
45 SOLVER_OP_ADD_ADDRESS,
46 SOLVER_OP_DEL_ADDRESS,
47 SOLVER_OP_START_SET_PROPERTY,
48 SOLVER_OP_STOP_SET_PROPERTY,
49 SOLVER_OP_START_SET_PREFERENCE,
50 SOLVER_OP_STOP_SET_PREFERENCE
51};
52
53struct Episode;
54
55struct Experiment;
56
57typedef void (*GNUNET_ATS_TESTING_EpisodeDoneCallback) (
58 struct Episode *e);
59
60typedef void (*GNUNET_ATS_TESTING_ExperimentDoneCallback) (struct Experiment *e,
61 struct GNUNET_TIME_Relative duration,int success);
62
63/**
64 * An operation in an experiment
65 */
66struct GNUNET_ATS_TEST_Operation
67{
68 struct GNUNET_ATS_TEST_Operation *next;
69 struct GNUNET_ATS_TEST_Operation *prev;
70
71 long long unsigned int address_id;
72 long long unsigned int peer_id;
73 long long unsigned int address_session;
74 long long unsigned int address_network;
75 char*address;
76 char*plugin;
77
78 long long unsigned int base_rate;
79 long long unsigned int max_rate;
80 struct GNUNET_TIME_Relative period;
81 struct GNUNET_TIME_Relative frequency;
82
83 enum OperationType type;
84 enum GeneratorType tg_type;
85 enum GNUNET_ATS_PreferenceKind pref_type;
86};
87
88struct Episode
89{
90 int id;
91 struct Episode *next;
92 struct GNUNET_TIME_Relative duration;
93
94 struct GNUNET_ATS_TEST_Operation *head;
95 struct GNUNET_ATS_TEST_Operation *tail;
96};
97
98
99struct Experiment
100{
101 char *name;
102 char *cfg_file;
103 unsigned long long int num_masters;
104 unsigned long long int num_slaves;
105 struct GNUNET_TIME_Relative log_freq;
106 struct GNUNET_TIME_Relative max_duration;
107 struct GNUNET_TIME_Relative total_duration;
108 struct GNUNET_TIME_Absolute start_time;
109 unsigned int num_episodes;
110 struct Episode *start;
111
112 struct GNUNET_CONFIGURATION_Handle *cfg;
113
114 GNUNET_SCHEDULER_TaskIdentifier experiment_timeout_task;
115 GNUNET_SCHEDULER_TaskIdentifier episode_timeout_task;
116 struct Episode *cur;
117
118 GNUNET_ATS_TESTING_EpisodeDoneCallback ep_done_cb;
119 GNUNET_ATS_TESTING_ExperimentDoneCallback e_done_cb;
120};
121
122
123/**
124 * A single logging time step for a partner
125 */
126struct PartnerLoggingTimestep
127{
128 /**
129 * Peer
130 */
131 struct BenchmarkPeer *slave;
132
133 /**
134 * Total number of messages this peer has sent
135 */
136 unsigned int total_messages_sent;
137
138 /**
139 * Total number of bytes this peer has sent
140 */
141 unsigned int total_bytes_sent;
142
143 /**
144 * Total number of messages this peer has received
145 */
146 unsigned int total_messages_received;
147
148 /**
149 * Total number of bytes this peer has received
150 */
151 unsigned int total_bytes_received;
152
153 /**
154 * Total outbound throughput for master in Bytes / s
155 */
156 unsigned int throughput_sent;
157
158 /**
159 * Total inbound throughput for master in Bytes / s
160 */
161 unsigned int throughput_recv;
162
163 /**
164 * Accumulated RTT for all messages
165 */
166 unsigned int total_app_rtt;
167
168 /**
169 * Current application level delay
170 */
171 unsigned int app_rtt;
172
173 /* Current ATS properties */
174
175 uint32_t ats_distance;
176
177 uint32_t ats_delay;
178
179 uint32_t bandwidth_in;
180
181 uint32_t bandwidth_out;
182
183 uint32_t ats_utilization_up;
184
185 uint32_t ats_utilization_down;
186
187 uint32_t ats_network_type;
188
189 uint32_t ats_cost_wan;
190
191 uint32_t ats_cost_lan;
192
193 uint32_t ats_cost_wlan;
194
195 double pref_bandwidth;
196 double pref_delay;
197};
198
199
200/**
201 * A single logging time step for a peer
202 */
203struct PeerLoggingTimestep
204{
205 /**
206 * Next in DLL
207 */
208 struct PeerLoggingTimestep *next;
209
210 /**
211 * Prev in DLL
212 */
213 struct PeerLoggingTimestep *prev;
214
215 /**
216 * Logging timestamp
217 */
218 struct GNUNET_TIME_Absolute timestamp;
219
220 /**
221 * Total number of messages this peer has sent
222 */
223 unsigned int total_messages_sent;
224
225 /**
226 * Total number of bytes this peer has sent
227 */
228 unsigned int total_bytes_sent;
229
230 /**
231 * Total number of messages this peer has received
232 */
233 unsigned int total_messages_received;
234
235 /**
236 * Total number of bytes this peer has received
237 */
238 unsigned int total_bytes_received;
239
240 /**
241 * Total outbound throughput for master in Bytes / s
242 */
243 unsigned int total_throughput_send;
244
245 /**
246 * Total inbound throughput for master in Bytes / s
247 */
248 unsigned int total_throughput_recv;
249
250 /**
251 * Logs for slaves
252 */
253 struct PartnerLoggingTimestep *slaves_log;
254};
255
256/**
257 * Entry for a benchmark peer
258 */
259struct LoggingPeer
260{
261 /**
262 * Peer
263 */
264 struct BenchmarkPeer *peer;
265
266 /**
267 * Start time
268 */
269 struct GNUNET_TIME_Absolute start;
270
271 /**
272 * DLL for logging entries: head
273 */
274 struct PeerLoggingTimestep *head;
275
276 /**
277 * DLL for logging entries: tail
278 */
279 struct PeerLoggingTimestep *tail;
280};
281
282
283struct LoggingHandle
284{
285 /**
286 * Logging task
287 */
288 GNUNET_SCHEDULER_TaskIdentifier log_task;
289
290 /**
291 * Reference to perf_ats' masters
292 */
293 int num_masters;
294 int num_slaves;
295 int running;
296 int verbose;
297 char *name;
298 struct GNUNET_TIME_Relative frequency;
299
300 /**
301 * Log structure of length num_peers
302 */
303 struct LoggingPeer *lp;
304};
305
306
307/**
308 * Start logging
309 *
310 * @param log_frequency the logging frequency
311 * @param testname the testname
312 * @param masters the master peers used for benchmarking
313 * @param num_master the number of master peers
314 * @return the logging handle or NULL on error
315 */
316struct LoggingHandle *
317GNUNET_ATS_TEST_logging_start(struct GNUNET_TIME_Relative log_frequency,
318 char *testname, struct BenchmarkPeer *masters, int num_masters, int num_slaves,
319 int verbose);
320
321/**
322 * Stop logging
323 *
324 * @param l the logging handle
325 */
326void
327GNUNET_ATS_TEST_logging_clean_up (struct LoggingHandle *l);
328
329/**
330 * Stop logging
331 *
332 * @param l the logging handle
333 */
334void
335GNUNET_ATS_TEST_logging_stop (struct LoggingHandle *l);
336
337/**
338 * Log all data now
339 *
340 * @param l logging handle to use
341 */
342void
343GNUNET_ATS_TEST_logging_now (struct LoggingHandle *l);
344
345
346/**
347 * Write logging data to file
348 *
349 * @param l logging handle to use
350 * @param test_name name of the current test
351 * @param plots create gnuplots: GNUNET_YES or GNUNET_NO
352 */
353void
354GNUNET_ATS_TEST_logging_write_to_file (struct LoggingHandle *l,
355 char *test_name, int plots);
356
357
358
359
360
361
362/* LEGACY */
363
364#if 0
365#define TEST_ATS_PREFERENCE_DEFAULT 1.0
366
367/**
368 * Message type sent for traffic generation
369 */
370#define TEST_MESSAGE_TYPE_PING 12345
371
372/**
373 * Message type sent as response during traffic generation
374 */
375#define TEST_MESSAGE_TYPE_PONG 12346
376
377/**
378 * Size of test messages
379 */
380#define TEST_MESSAGE_SIZE 100
381
382struct BenchmarkPartner;
383
384struct BenchmarkPeer;
385
386struct GNUNET_ATS_TEST_Topology;
387
388struct TrafficGenerator;
389
390
391
392
393
394/**
395 * Callback to call when topology setup is completed
396 *
397 * @param cls the closure
398 * @param masters array of master peers
399 * @param slaves array of master peers
400 */
401typedef void (*GNUNET_ATS_TEST_TopologySetupDoneCallback) (void *cls,
402 struct BenchmarkPeer *masters,
403 struct BenchmarkPeer *slaves);
404
405/**
406 * Callback called when logging is required for the data contained
407 *
408 * @param cls the closure
409 * @param address an address
410 * @param address_active is address active
411 * @param bandwidth_out bandwidth outbound
412 * @param bandwidth_in bandwidth inbound
413 * @param ats ats information
414 * @param ats_count number of ats inforation
415 */
416typedef void
417(*GNUNET_ATS_TEST_LogRequest) (void *cls,
418 const struct GNUNET_HELLO_Address *address,
419 int address_active,
420 struct GNUNET_BANDWIDTH_Value32NBO bandwidth_out,
421 struct GNUNET_BANDWIDTH_Value32NBO bandwidth_in,
422 const struct GNUNET_ATS_Information *ats,
423 uint32_t ats_count);
424
425/**
426 * Information we track for a peer in the testbed.
427 */
428struct BenchmarkPeer
429{
430 /**
431 * Handle with testbed.
432 */
433 struct GNUNET_TESTBED_Peer *peer;
434
435 /**
436 * Unique identifier
437 */
438 int no;
439
440 /**
441 * Is this peer a measter: GNUNET_YES/GNUNET_NO
442 */
443 int master;
444
445 /**
446 * Peer ID
447 */
448 struct GNUNET_PeerIdentity id;
449
450 /**
451 * Testbed operation to get peer information
452 */
453 struct GNUNET_TESTBED_Operation *peer_id_op;
454
455 /**
456 * Testbed operation to connect to ATS performance service
457 */
458 struct GNUNET_TESTBED_Operation *ats_perf_op;
459
460 /**
461 * Testbed operation to connect to core
462 */
463 struct GNUNET_TESTBED_Operation *comm_op;
464
465 /**
466 * ATS performance handle
467 */
468 struct GNUNET_ATS_PerformanceHandle *ats_perf_handle;
469
470 /**
471 * Masters only:
472 * Testbed connect operations to connect masters to slaves
473 */
474 struct TestbedConnectOperation *core_connect_ops;
475
476 /**
477 * Core handle
478 */
479 struct GNUNET_CORE_Handle *ch;
480
481 /**
482 * Core handle
483 */
484 struct GNUNET_TRANSPORT_Handle *th;
485
486 /**
487 * Masters only:
488 * Peer to set ATS preferences for
489 */
490 struct BenchmarkPeer *pref_partner;
491
492 /**
493 * Masters only
494 * Progress task
495 */
496 GNUNET_SCHEDULER_TaskIdentifier ats_task;
497
498 /**
499 * Masters only
500 * Progress task
501 */
502 double pref_value;
503
504 /**
505 * Array of partners with num_slaves entries (if master) or
506 * num_master entries (if slave)
507 */
508 struct BenchmarkPartner *partners;
509
510 /**
511 * Number of partners
512 */
513 int num_partners;
514
515 /**
516 * Number of core connections
517 */
518 int core_connections;
519
520 /**
521 * Masters only:
522 * Number of connections to slave peers
523 */
524 int core_slave_connections;
525
526 /**
527 * Total number of messages this peer has sent
528 */
529 unsigned int total_messages_sent;
530
531 /**
532 * Total number of bytes this peer has sent
533 */
534 unsigned int total_bytes_sent;
535
536 /**
537 * Total number of messages this peer has received
538 */
539 unsigned int total_messages_received;
540
541 /**
542 * Total number of bytes this peer has received
543 */
544 unsigned int total_bytes_received;
545};
546
547struct TrafficGenerator
548{
549 struct TrafficGenerator *prev;
550 struct TrafficGenerator *next;
551
552 enum GeneratorType type;
553
554 struct BenchmarkPeer *src;
555 struct BenchmarkPartner *dest;
556
557 long int base_rate;
558 long int max_rate;
559 struct GNUNET_TIME_Relative duration_period;
560
561 GNUNET_SCHEDULER_TaskIdentifier send_task;
562 struct GNUNET_TIME_Absolute next_ping_transmission;
563 struct GNUNET_TIME_Absolute time_start;
564};
565
566
567struct PreferenceGenerator
568{
569 struct PreferenceGenerator *prev;
570 struct PreferenceGenerator *next;
571
572 enum GeneratorType type;
573
574 struct BenchmarkPeer *src;
575 struct BenchmarkPartner *dest;
576
577 enum GNUNET_ATS_PreferenceKind kind;
578
579 long int base_value;
580 long int max_value;
581 struct GNUNET_TIME_Relative duration_period;
582 struct GNUNET_TIME_Relative frequency;
583
584 GNUNET_SCHEDULER_TaskIdentifier set_task;
585 struct GNUNET_TIME_Absolute next_ping_transmission;
586 struct GNUNET_TIME_Absolute time_start;
587};
588
589/**
590 * Information about a benchmarking partner
591 */
592struct BenchmarkPartner
593{
594 /**
595 * The peer itself this partner belongs to
596 */
597 struct BenchmarkPeer *me;
598
599 /**
600 * The partner peer
601 */
602 struct BenchmarkPeer *dest;
603
604 /**
605 * Core transmit handles
606 */
607 struct GNUNET_CORE_TransmitHandle *cth;
608
609 /**
610 * Transport transmit handles
611 */
612 struct GNUNET_TRANSPORT_TransmitHandle *tth;
613
614 struct TrafficGenerator *tg;
615 struct PreferenceGenerator *pg;
616
617 /**
618 * Timestamp to calculate communication layer delay
619 */
620 struct GNUNET_TIME_Absolute last_message_sent;
621
622 /**
623 * Accumulated RTT for all messages
624 */
625 unsigned int total_app_rtt;
626
627 /**
628 * Number of messages sent to this partner
629 */
630 unsigned int messages_sent;
631
632 /**
633 * Number of bytes sent to this partner
634 */
635 unsigned int bytes_sent;
636
637 /**
638 * Number of messages received from this partner
639 */
640 unsigned int messages_received;
641
642 /**
643 * Number of bytes received from this partner
644 */
645 unsigned int bytes_received;
646
647 /* Current ATS properties */
648
649 uint32_t ats_distance;
650
651 uint32_t ats_delay;
652
653 uint32_t bandwidth_in;
654
655 uint32_t bandwidth_out;
656
657 uint32_t ats_utilization_up;
658
659 uint32_t ats_utilization_down;
660
661 uint32_t ats_network_type;
662
663 uint32_t ats_cost_wan;
664
665 uint32_t ats_cost_lan;
666
667 uint32_t ats_cost_wlan;
668
669 double pref_bandwidth;
670 double pref_delay;
671};
672
673/**
674 * Overall state of the performance benchmark
675 */
676struct BenchmarkState
677{
678 /**
679 * Are we connected to ATS service of all peers: GNUNET_YES/NO
680 */
681 int connected_ATS_service;
682
683 /**
684 * Are we connected to CORE service of all peers: GNUNET_YES/NO
685 */
686 int connected_COMM_service;
687
688 /**
689 * Are we connected to all peers: GNUNET_YES/NO
690 */
691 int connected_PEERS;
692
693 /**
694 * Are we connected to all slave peers on CORE level: GNUNET_YES/NO
695 */
696 int connected_CORE;
697
698 /**
699 * Are we connected to CORE service of all peers: GNUNET_YES/NO
700 */
701 int benchmarking;
702};
703
704
705struct GNUNET_ATS_TEST_Topology
706{
707 /**
708 * Shutdown task
709 */
710 GNUNET_SCHEDULER_TaskIdentifier shutdown_task;
711
712 /**
713 * Progress task
714 */
715 GNUNET_SCHEDULER_TaskIdentifier progress_task;
716
717 /**
718 * Test result
719 */
720 int result;
721
722 /**Test core (GNUNET_YES) or transport (GNUNET_NO)
723 */
724 int test_core;
725
726 /**
727 * Solver string
728 */
729 char *solver;
730
731 /**
732 * Preference string
733 */
734 char *testname;
735
736 /**
737 * Preference string
738 */
739 char *pref_str;
740
741 /**
742 * ATS preference value
743 */
744 int pref_val;
745
746 /**
747 * Number master peers
748 */
749 unsigned int num_masters;
750
751 /**
752 * Array of master peers
753 */
754 struct BenchmarkPeer *mps;
755
756 /**
757 * Number slave peers
758 */
759 unsigned int num_slaves;
760
761 /**
762 * Array of slave peers
763 */
764 struct BenchmarkPeer *sps;
765
766 /**
767 * Benchmark duration
768 */
769 struct GNUNET_TIME_Relative perf_duration;
770
771 /**
772 * Logging frequency
773 */
774 struct GNUNET_TIME_Relative log_frequency;
775
776 /**
777 * Benchmark state
778 */
779 struct BenchmarkState state;
780
781 struct GNUNET_CORE_MessageHandler *handlers;
782
783 GNUNET_TRANSPORT_ReceiveCallback transport_recv_cb;
784
785 GNUNET_ATS_TEST_TopologySetupDoneCallback done_cb;
786 GNUNET_ATS_AddressInformationCallback ats_perf_cb;
787 void *done_cb_cls;
788};
789
790
791
792/*
793 * Experiment related functions
794 */
795
796
797/**
798 * Execute the specified experiment
799 *
800 * @param e the Experiment
801 * @param ep_done_cb a episode is completed
802 * @param e_done_cb the experiment is completed
803 */
804void
805GNUNET_ATS_TEST_experimentation_run (struct Experiment *e,
806 GNUNET_ATS_TESTING_EpisodeDoneCallback ep_done_cb,
807 GNUNET_ATS_TESTING_ExperimentDoneCallback e_done_cb);
808
809/**
810 * Load an experiment from a file
811 *
812 * @param filename the file
813 * @return the Experiment or NULL on failure
814 */
815struct Experiment *
816GNUNET_ATS_TEST_experimentation_load (char *filename);
817
818
819/**
820 * Stop an experiment
821 *
822 * @param e the experiment
823 */
824void
825GNUNET_ATS_TEST_experimentation_stop (struct Experiment *e);
826
827/*
828 * Traffic related functions
829 */
830
831void
832GNUNET_ATS_TEST_traffic_handle_ping (struct BenchmarkPartner *p);
833
834void
835GNUNET_ATS_TEST_traffic_handle_pong (struct BenchmarkPartner *p);
836
837
838/**
839 * Generate between the source master and the partner and send traffic with a
840 * maximum rate.
841 *
842 * @param src traffic source
843 * @param dest traffic partner
844 * @param type type of traffic to generate
845 * @param base_rate traffic base rate to send data with
846 * @param max_rate traffic maximum rate to send data with
847 * @param period duration of a period of traffic generation (~ 1/frequency)
848 * @param duration how long to generate traffic
849 * @return the traffic generator
850 */
851struct TrafficGenerator *
852GNUNET_ATS_TEST_generate_traffic_start (struct BenchmarkPeer *src,
853 struct BenchmarkPartner *dest,
854 enum GeneratorType type,
855 long int base_rate,
856 long int max_rate,
857 struct GNUNET_TIME_Relative period,
858 struct GNUNET_TIME_Relative duration);
859
860void
861GNUNET_ATS_TEST_generate_traffic_stop (struct TrafficGenerator *tg);
862
863/**
864 * Stop all traffic generators
865 */
866void
867GNUNET_ATS_TEST_generate_traffic_stop_all ();
868
869struct PreferenceGenerator *
870GNUNET_ATS_TEST_generate_preferences_start (struct BenchmarkPeer *src,
871 struct BenchmarkPartner *dest,
872 enum GeneratorType type,
873 long int base_value,
874 long int value_rate,
875 struct GNUNET_TIME_Relative period,
876 struct GNUNET_TIME_Relative frequency,
877 enum GNUNET_ATS_PreferenceKind kind);
878
879void
880GNUNET_ATS_TEST_generate_preferences_stop (struct PreferenceGenerator *pg);
881
882void
883GNUNET_ATS_TEST_generate_preferences_stop_all ();
884
885/*
886 * Logging related functions
887 */
888
889
890
891/*
892 * Topology related functions
893 */
894
895struct BenchmarkPeer *
896GNUNET_ATS_TEST_get_peer (int src);
897
898struct BenchmarkPartner *
899GNUNET_ATS_TEST_get_partner (int src, int dest);
900
901/**
902 * Create a topology for ats testing
903 *
904 * @param name test name
905 * @param cfg_file configuration file to use for the peers
906 * @param num_slaves number of slaves
907 * @param num_masters number of masters
908 * @param test_core connect to CORE service (GNUNET_YES) or transport (GNUNET_NO)
909 * @param done_cb function to call when topology is setup
910 * @param done_cb_cls cls for callback
911 * @param recv_cb callback to call when data are received
912 * @param perf_cb callback to call when performance info are received
913 */
914void
915GNUNET_ATS_TEST_create_topology (char *name, char *cfg_file,
916 unsigned int num_slaves,
917 unsigned int num_masters,
918 int test_core,
919 GNUNET_ATS_TEST_TopologySetupDoneCallback done_cb,
920 void *done_cb_cls,
921 GNUNET_TRANSPORT_ReceiveCallback recv_cb,
922 GNUNET_ATS_TEST_LogRequest ats_perf_cb);
923
924/**
925 * Shutdown topology
926 */
927void
928GNUNET_ATS_TEST_shutdown_topology (void);
929#endif
930/* end of file ats-testing.h */