/* This file is part of GNUnet. Copyright (C) 2011 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 . SPDX-License-Identifier: AGPL3.0-or-later */ /** * @file ats/gnunet-service-ats.c * @brief ats service * @author Matthias Wachs * @author Christian Grothoff */ #include "platform.h" #include "gnunet_util_lib.h" #include "gnunet-service-ats.h" #include "gnunet-service-ats_addresses.h" #include "gnunet-service-ats_connectivity.h" #include "gnunet-service-ats_normalization.h" #include "gnunet-service-ats_performance.h" #include "gnunet-service-ats_preferences.h" #include "gnunet-service-ats_scheduling.h" #include "gnunet-service-ats_reservations.h" #include "gnunet-service-ats_plugins.h" #include "ats.h" /** * Handle for statistics. */ struct GNUNET_STATISTICS_Handle *GSA_stats; /** * We have received a `struct ClientStartMessage` from a client. Find * out which type of client it is and notify the respective subsystem. * * @param cls handle to the client * @param msg the start message */ static void handle_ats_start(void *cls, const struct ClientStartMessage *msg) { struct GNUNET_SERVICE_Client *client = cls; enum StartFlag flag; flag = ntohl(msg->start_flag); GNUNET_log(GNUNET_ERROR_TYPE_DEBUG, "Received ATS_START (%d) message\n", (int)flag); switch (flag) { case START_FLAG_SCHEDULING: if (GNUNET_OK != GAS_scheduling_add_client(client)) { GNUNET_SERVICE_client_drop(client); return; } break; case START_FLAG_PERFORMANCE_WITH_PIC: GAS_performance_add_client(client, flag); break; case START_FLAG_PERFORMANCE_NO_PIC: GAS_performance_add_client(client, flag); break; case START_FLAG_CONNECTION_SUGGESTION: /* This client won't receive messages from us, no need to 'add' */ break; default: GNUNET_break(0); GNUNET_SERVICE_client_drop(client); return; } GNUNET_SERVICE_client_continue(client); } /** * Handle 'reservation request' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_reservation_request(void *cls, const struct ReservationRequestMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_reservation_request(client, message); GNUNET_SERVICE_client_continue(client); } /** * Check 'preference feedback' message is well-formed * * @param cls client that sent the request * @param message the request message * @return #GNUNET_OK if @a message is well-formed */ static int check_feedback(void *cls, const struct FeedbackPreferenceMessage *message) { uint16_t msize; uint32_t nump; GNUNET_log(GNUNET_ERROR_TYPE_DEBUG, "Received PREFERENCE_FEEDBACK message\n"); msize = ntohs(message->header.size); nump = ntohl(message->num_feedback); if (msize != sizeof(struct FeedbackPreferenceMessage) + nump * sizeof(struct PreferenceInformation)) { GNUNET_break(0); return GNUNET_SYSERR; } return GNUNET_OK; } /** * Handle 'preference feedback' messages from clients. * * @param cls client that sent the request * @param msg the request message */ static void handle_feedback(void *cls, const struct FeedbackPreferenceMessage *msg) { struct GNUNET_SERVICE_Client *client = cls; const struct PreferenceInformation *pi; uint32_t nump; nump = ntohl(msg->num_feedback); if (GNUNET_NO == GNUNET_CONTAINER_multipeermap_contains(GSA_addresses, &msg->peer)) { GNUNET_log(GNUNET_ERROR_TYPE_WARNING, "Received PREFERENCE FEEDBACK for unknown peer `%s'\n", GNUNET_i2s(&msg->peer)); GNUNET_SERVICE_client_continue(client); return; } GNUNET_STATISTICS_update(GSA_stats, "# preference feedbacks requests processed", 1, GNUNET_NO); pi = (const struct PreferenceInformation *)&msg[1]; for (uint32_t i = 0; i < nump; i++) { GNUNET_log(GNUNET_ERROR_TYPE_DEBUG, "Received PREFERENCE FEEDBACK for peer `%s'\n", GNUNET_i2s(&msg->peer)); GAS_plugin_notify_feedback(client, &msg->peer, GNUNET_TIME_relative_ntoh(msg->scope), (enum GNUNET_ATS_PreferenceKind)ntohl(pi[i].preference_kind), pi[i].preference_value); } GNUNET_SERVICE_client_continue(client); } /** * Handle 'request address list' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_request_address_list(void *cls, const struct AddressListRequestMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_request_address_list(client, message); GNUNET_SERVICE_client_continue(client); } /** * Handle 'request address' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_request_address(void *cls, const struct RequestAddressMessage * message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_request_address(client, message); GNUNET_SERVICE_client_continue(client); } /** * Cancel 'request address' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_request_address_cancel(void *cls, const struct RequestAddressMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_request_address_cancel(client, message); GNUNET_SERVICE_client_continue(client); } /** * Handle 'address add' messages from clients. * * @param cls client that sent the request * @param m the request message */ static int check_address_add(void *cls, const struct AddressAddMessage *m) { const char *address; const char *plugin_name; uint16_t address_length; uint16_t plugin_name_length; uint16_t size; size = ntohs(m->header.size); address_length = ntohs(m->address_length); plugin_name_length = ntohs(m->plugin_name_length); address = (const char *)&m[1]; if (plugin_name_length != 0) plugin_name = &address[address_length]; else plugin_name = ""; if ((address_length + plugin_name_length + sizeof(struct AddressAddMessage) != size) || ((plugin_name_length > 0) && (plugin_name[plugin_name_length - 1] != '\0'))) { GNUNET_break(0); return GNUNET_SYSERR; } return GNUNET_OK; } /** * Handle 'address add' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_address_add(void *cls, const struct AddressAddMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_address_add(message); GNUNET_SERVICE_client_continue(client); } /** * Handle 'address update' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_address_update(void *cls, const struct AddressUpdateMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_address_update(message); GNUNET_SERVICE_client_continue(client); } /** * Handle 'address destroyed' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_address_destroyed(void *cls, const struct AddressDestroyedMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_address_destroyed(message); GNUNET_SERVICE_client_continue(client); } /** * Check that 'change preference' message is well-formed. * * @param cls client that sent the request * @param message the request message * @return #GNUNET_OK if @a message is well-formed */ static int check_preference_change(void *cls, const struct ChangePreferenceMessage *message) { uint16_t msize; uint32_t nump; msize = ntohs(message->header.size); nump = ntohl(message->num_preferences); if ((msize != sizeof(struct ChangePreferenceMessage) + nump * sizeof(struct PreferenceInformation)) || (UINT16_MAX / sizeof(struct PreferenceInformation) < nump)) { GNUNET_break(0); return GNUNET_SYSERR; } return GNUNET_OK; } /** * Handle 'change preference' messages from clients. * * @param cls client that sent the request * @param message the request message */ static void handle_preference_change(void *cls, const struct ChangePreferenceMessage *message) { struct GNUNET_SERVICE_Client *client = cls; GAS_handle_preference_change(client, message); GNUNET_SERVICE_client_continue(client); } /** * A client connected to us. Setup the local client * record. * * @param cls unused * @param client handle of the client * @param mq message queue to talk to @a client * @return @a client */ static void * client_connect_cb(void *cls, struct GNUNET_SERVICE_Client *client, struct GNUNET_MQ_Handle *mq) { return client; } /** * A client disconnected from us. Tear down the local client * record. * * @param cls unused * @param client handle of the client * @param app_ctx */ static void client_disconnect_cb(void *cls, struct GNUNET_SERVICE_Client *client, void *app_ctx) { if (NULL == client) return; GAS_scheduling_remove_client(client); GAS_connectivity_remove_client(client); GAS_preference_client_disconnect(client); } /** * Task run during shutdown. * * @param cls unused */ static void cleanup_task(void *cls) { GNUNET_log(GNUNET_ERROR_TYPE_DEBUG, "ATS shutdown initiated\n"); GAS_connectivity_done(); GAS_addresses_done(); GAS_plugin_done(); GAS_normalization_stop(); GAS_performance_done(); GAS_preference_done(); GAS_reservations_done(); if (NULL != GSA_stats) { GNUNET_STATISTICS_destroy(GSA_stats, GNUNET_NO); GSA_stats = NULL; } } /** * Process template requests. * * @param cls closure * @param cfg configuration to use * @param service the initialized service */ static void run(void *cls, const struct GNUNET_CONFIGURATION_Handle *cfg, struct GNUNET_SERVICE_Handle *service) { GSA_stats = GNUNET_STATISTICS_create("ats", cfg); GAS_reservations_init(); GAS_connectivity_init(); GAS_preference_init(); GAS_normalization_start(); GAS_addresses_init(); if (GNUNET_OK != GAS_plugin_init(cfg)) { GNUNET_break(0); GAS_addresses_done(); GAS_normalization_stop(); GAS_reservations_done(); GAS_connectivity_done(); GAS_preference_done(); if (NULL != GSA_stats) { GNUNET_STATISTICS_destroy(GSA_stats, GNUNET_NO); GSA_stats = NULL; } return; } GAS_performance_init(); GNUNET_SCHEDULER_add_shutdown(&cleanup_task, NULL); } /** * Define "main" method using service macro. */ GNUNET_SERVICE_MAIN ("ats", GNUNET_SERVICE_OPTION_NONE, &run, &client_connect_cb, &client_disconnect_cb, NULL, GNUNET_MQ_hd_fixed_size(ats_start, GNUNET_MESSAGE_TYPE_ATS_START, struct ClientStartMessage, NULL), GNUNET_MQ_hd_fixed_size(request_address, GNUNET_MESSAGE_TYPE_ATS_REQUEST_ADDRESS, struct RequestAddressMessage, NULL), GNUNET_MQ_hd_fixed_size(request_address_cancel, GNUNET_MESSAGE_TYPE_ATS_REQUEST_ADDRESS_CANCEL, struct RequestAddressMessage, NULL), GNUNET_MQ_hd_fixed_size(request_address_list, GNUNET_MESSAGE_TYPE_ATS_ADDRESSLIST_REQUEST, struct AddressListRequestMessage, NULL), GNUNET_MQ_hd_var_size(address_add, GNUNET_MESSAGE_TYPE_ATS_ADDRESS_ADD, struct AddressAddMessage, NULL), GNUNET_MQ_hd_fixed_size(address_update, GNUNET_MESSAGE_TYPE_ATS_ADDRESS_UPDATE, struct AddressUpdateMessage, NULL), GNUNET_MQ_hd_fixed_size(address_destroyed, GNUNET_MESSAGE_TYPE_ATS_ADDRESS_DESTROYED, struct AddressDestroyedMessage, NULL), GNUNET_MQ_hd_fixed_size(reservation_request, GNUNET_MESSAGE_TYPE_ATS_RESERVATION_REQUEST, struct ReservationRequestMessage, NULL), GNUNET_MQ_hd_var_size(preference_change, GNUNET_MESSAGE_TYPE_ATS_PREFERENCE_CHANGE, struct ChangePreferenceMessage, NULL), GNUNET_MQ_hd_var_size(feedback, GNUNET_MESSAGE_TYPE_ATS_PREFERENCE_FEEDBACK, struct FeedbackPreferenceMessage, NULL), GNUNET_MQ_handler_end()); /* end of gnunet-service-ats.c */