/* This file is part of GNUnet. (C) 2001, 2002, 2004, 2005, 2006, 2007, 2009 Christian Grothoff (and other contributing authors) GNUnet is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, 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 General Public License for more details. You should have received a copy of the GNU General Public License along with GNUnet; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /** * @file ats-tool/gnunet-ats.c * @brief ATS command line tool * @author Matthias Wachs */ #include "platform.h" #include "gnunet_util_lib.h" #include "gnunet_ats_service.h" #include "gnunet_transport_service.h" #define TIMEOUT GNUNET_TIME_relative_multiply(GNUNET_TIME_UNIT_SECONDS, 5) /** * Final status code. */ static int ret; static int results; static int resolve_addresses_numeric; static int monitor; /** * For which peer should we change preference values? */ static char *pid_str; static char *type_str; static unsigned int value; /** * Print verbose ATS information */ static int verbose; /** * List only addresses currently used (active) */ static int list_used; /** * List all addresses */ static int list_all; static struct GNUNET_ATS_PerformanceHandle *ph; static struct GNUNET_CONFIGURATION_Handle *cfg; GNUNET_SCHEDULER_TaskIdentifier end_task; struct PendingResolutions { struct PendingResolutions *next; struct PendingResolutions *prev; struct GNUNET_HELLO_Address *address; struct GNUNET_BANDWIDTH_Value32NBO bandwidth_out; struct GNUNET_BANDWIDTH_Value32NBO bandwidth_in; struct GNUNET_ATS_Information *ats; uint32_t ats_count; struct GNUNET_TRANSPORT_AddressToStringContext * tats_ctx; }; struct PendingResolutions *head; struct PendingResolutions *tail; void transport_addr_to_str_cb (void *cls, const char *address) { struct PendingResolutions * pr = cls; char *ats_str; char *ats_tmp; char *ats_prop_arr[GNUNET_ATS_PropertyCount] = GNUNET_ATS_PropertyStrings; char *ats_net_arr[GNUNET_ATS_NetworkTypeCount] = GNUNET_ATS_NetworkTypeString; char *ats_prop_value; unsigned int c; uint32_t ats_type; uint32_t ats_value; if (NULL != address) { ats_str = GNUNET_strdup(""); if (verbose) { for (c = 0; c < pr->ats_count; c++) { ats_tmp = ats_str; ats_type = ntohl(pr->ats[c].type); ats_value = ntohl(pr->ats[c].value); if (ats_type > GNUNET_ATS_PropertyCount) { GNUNET_break (0); continue; } switch (ats_type) { case GNUNET_ATS_NETWORK_TYPE: if (ats_value > GNUNET_ATS_NetworkTypeCount) { GNUNET_break (0); continue; } GNUNET_asprintf (&ats_prop_value, "%s", ats_net_arr[ats_value]); break; default: GNUNET_asprintf (&ats_prop_value, "%u", ats_value); break; } GNUNET_asprintf (&ats_str, "%s%s=%s, ", ats_tmp, ats_prop_arr[ats_type] , ats_prop_value); GNUNET_free (ats_tmp); GNUNET_free (ats_prop_value); } } fprintf (stderr, _("Peer `%s' plugin `%s', address `%s', bw out: %u Bytes/s, bw in %u Bytes/s, %s\n"), GNUNET_i2s (&pr->address->peer), pr->address->transport_name, address, ntohl (pr->bandwidth_out.value__), ntohl (pr->bandwidth_in.value__),ats_str); GNUNET_free (ats_str); } else if (NULL != pr) { /* We're done */ GNUNET_CONTAINER_DLL_remove (head, tail, pr); GNUNET_free (pr->address); GNUNET_free (pr); } } void ats_perf_cb (void *cls, const struct GNUNET_HELLO_Address * address, struct GNUNET_BANDWIDTH_Value32NBO bandwidth_out, struct GNUNET_BANDWIDTH_Value32NBO bandwidth_in, const struct GNUNET_ATS_Information * ats, uint32_t ats_count) { struct PendingResolutions * pr; pr = GNUNET_malloc (sizeof (struct PendingResolutions) + ats_count * sizeof (struct GNUNET_ATS_Information)); pr->ats_count = ats_count; pr->ats = (struct GNUNET_ATS_Information *) &pr[1]; if (ats_count > 0) memcpy (pr->ats, ats, ats_count * sizeof (struct GNUNET_ATS_Information)); pr->address = GNUNET_HELLO_address_copy (address); pr->bandwidth_in = bandwidth_in; pr->bandwidth_out = bandwidth_out; pr->tats_ctx = GNUNET_TRANSPORT_address_to_string(cfg, address, resolve_addresses_numeric, GNUNET_TIME_UNIT_FOREVER_REL, transport_addr_to_str_cb, pr); GNUNET_CONTAINER_DLL_insert (head, tail, pr); results++; } void end (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc) { struct PendingResolutions * pr; struct PendingResolutions * next; unsigned int pending; GNUNET_ATS_performance_done (ph); ph = NULL; pending = 0; next = head; while (NULL != (pr = next)) { next = pr->next; GNUNET_CONTAINER_DLL_remove (head, tail, pr); GNUNET_TRANSPORT_address_to_string_cancel (pr->tats_ctx); GNUNET_free (pr->address); GNUNET_free (pr); pending ++; } if (0 < pending) fprintf (stderr, _("%u address resolutions had a timeout\n"), pending); fprintf (stderr, _("ATS returned results for %u addresses\n"), results); ret = 0; } void testservice_ats (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc) { struct GNUNET_PeerIdentity pid; struct GNUNET_CONFIGURATION_Handle *cfg = cls; unsigned int c; unsigned int type; if (0 != (tc->reason & GNUNET_SCHEDULER_REASON_TIMEOUT)) { FPRINTF (stderr, _("Service `%s' is not running\n"), "ats"); return; } if ((GNUNET_YES == list_all) && (GNUNET_YES ==list_used)) { FPRINTF (stderr, _("Please select one operation : %s or %s \n"),"--used", "--all"); return; } if ((GNUNET_NO == list_all) && (GNUNET_NO == list_used)) list_used = GNUNET_YES; /* set default */ if (GNUNET_YES == list_all) { FPRINTF (stderr, _("NOT YET IMPLEMENTED\n")); return; } results = 0; if (NULL != pid_str) { if (GNUNET_OK != GNUNET_CRYPTO_hash_from_string (pid_str, &pid.hashPubKey)) { FPRINTF (stderr, _("Failed to parse peer identity `%s'\n"), pid_str); return; } if (NULL == type_str) { FPRINTF (stderr, "%s", _("Type required\n")); return; } for (c = 0; c