Ignore:
File:
1 edited

Legend:

Unmodified
Added
Removed
  • uspace/srv/net/tl/udp/udp.c

    rc5b59ce r14f1db0  
    4242#include <ipc/ipc.h>
    4343#include <ipc/services.h>
    44 #include <errno.h>
    45 #include <err.h>
    46 
     44
     45#include <net_err.h>
    4746#include <net_messages.h>
    4847#include <net_modules.h>
     
    6160#include <net_interface.h>
    6261#include <socket_codes.h>
     62#include <socket_errno.h>
    6363#include <socket_core.h>
    6464#include <socket_messages.h>
     
    7272#include "udp_module.h"
    7373
    74 /** UDP module name. */
     74/** UDP module name.
     75 */
    7576#define NAME    "UDP protocol"
    7677
    77 /** Default UDP checksum computing. */
     78/** Default UDP checksum computing.
     79 */
    7880#define NET_DEFAULT_UDP_CHECKSUM_COMPUTING      true
    7981
    80 /** Default UDP autobind when sending via unbound sockets. */
     82/** Default UDP autobind when sending via unbound sockets.
     83 */
    8184#define NET_DEFAULT_UDP_AUTOBINDING     true
    8285
    83 /** Maximum UDP fragment size. */
    84 #define MAX_UDP_FRAGMENT_SIZE           65535
    85 
    86 /** Free ports pool start. */
    87 #define UDP_FREE_PORTS_START            1025
    88 
    89 /** Free ports pool end. */
     86/** Maximum UDP fragment size.
     87 */
     88#define MAX_UDP_FRAGMENT_SIZE   65535
     89
     90/** Free ports pool start.
     91 */
     92#define UDP_FREE_PORTS_START    1025
     93
     94/** Free ports pool end.
     95 */
    9096#define UDP_FREE_PORTS_END              65535
    9197
    9298/** Processes the received UDP packet queue.
    93  *
    9499 *  Is used as an entry point from the underlying IP module.
    95100 *  Locks the global lock and calls udp_process_packet() function.
    96  *
    97101 *  @param[in] device_id The receiving device identifier.
    98102 *  @param[in,out] packet The received packet queue.
    99  *  @param receiver     The target service. Ignored parameter.
    100  *  @param[in] error    The packet error reporting service. Prefixes the
    101  *                      received packet.
    102  *  @returns            EOK on success.
    103  *  @returns            Other error codes as defined for the
    104  *                      udp_process_packet() function.
    105  */
    106 int
    107 udp_received_msg(device_id_t device_id, packet_t packet, services_t receiver,
    108     services_t error);
     103 *  @param receiver The target service. Ignored parameter.
     104 *  @param[in] error The packet error reporting service. Prefixes the received packet.
     105 *  @returns EOK on success.
     106 *  @returns Other error codes as defined for the udp_process_packet() function.
     107 */
     108int udp_received_msg(device_id_t device_id, packet_t packet, services_t receiver, services_t error);
    109109
    110110/** Processes the received UDP packet queue.
    111  *
    112111 *  Notifies the destination socket application.
    113  *  Releases the packet on error or sends an ICMP error notification.
    114  *
     112 *  Releases the packet on error or sends an ICMP error notification..
    115113 *  @param[in] device_id The receiving device identifier.
    116114 *  @param[in,out] packet The received packet queue.
    117  *  @param[in] error    The packet error reporting service. Prefixes the
    118  *                      received packet.
    119  *  @returns            EOK on success.
    120  *  @returns            EINVAL if the packet is not valid.
    121  *  @returns            EINVAL if the stored packet address is not the
    122  *                      an_addr_t.
    123  *  @returns            EINVAL if the packet does not contain any data.
    124  *  @returns            NO_DATA if the packet content is shorter than the user
    125  *                      datagram header.
    126  *  @returns            ENOMEM if there is not enough memory left.
    127  *  @returns            EADDRNOTAVAIL if the destination socket does not exist.
    128  *  @returns            Other error codes as defined for the
    129  *                      ip_client_process_packet() function.
    130  */
    131 int
    132 udp_process_packet(device_id_t device_id, packet_t packet, services_t error);
     115 *  @param[in] error The packet error reporting service. Prefixes the received packet.
     116 *  @returns EOK on success.
     117 *  @returns EINVAL if the packet is not valid.
     118 *  @returns EINVAL if the stored packet address is not the an_addr_t.
     119 *  @returns EINVAL if the packet does not contain any data.
     120 *  @returns NO_DATA if the packet content is shorter than the user datagram header.
     121 *  @returns ENOMEM if there is not enough memory left.
     122 *  @returns EADDRNOTAVAIL if the destination socket does not exist.
     123 *  @returns Other error codes as defined for the ip_client_process_packet() function.
     124 */
     125int udp_process_packet(device_id_t device_id, packet_t packet, services_t error);
    133126
    134127/** Releases the packet and returns the result.
    135  *
    136  *  @param[in] packet   The packet queue to be released.
    137  *  @param[in] result   The result to be returned.
    138  *  @return             The result parameter.
     128 *  @param[in] packet The packet queue to be released.
     129 *  @param[in] result The result to be returned.
     130 *  @return The result parameter.
    139131 */
    140132int udp_release_and_return(packet_t packet, int result);
     
    145137
    146138/** Processes the socket client messages.
    147  *
    148139 *  Runs until the client module disconnects.
    149  *
    150  *  @param[in] callid   The message identifier.
    151  *  @param[in] call     The message parameters.
    152  *  @returns            EOK on success.
    153  *  @see                socket.h
     140 *  @param[in] callid The message identifier.
     141 *  @param[in] call The message parameters.
     142 *  @returns EOK on success.
     143 *  @see socket.h
    154144 */
    155145int udp_process_client_messages(ipc_callid_t callid, ipc_call_t call);
    156146
    157147/** Sends data from the socket to the remote address.
    158  *
    159148 *  Binds the socket to a free port if not already connected/bound.
    160149 *  Handles the NET_SOCKET_SENDTO message.
    161150 *  Supports AF_INET and AF_INET6 address families.
    162  *
    163151 *  @param[in,out] local_sockets The application local sockets.
    164152 *  @param[in] socket_id Socket identifier.
    165  *  @param[in] addr     The destination address.
    166  *  @param[in] addrlen  The address length.
     153 *  @param[in] addr The destination address.
     154 *  @param[in] addrlen The address length.
    167155 *  @param[in] fragments The number of data fragments.
    168156 *  @param[out] data_fragment_size The data fragment size in bytes.
    169  *  @param[in] flags    Various send flags.
    170  *  @returns            EOK on success.
    171  *  @returns            EAFNOTSUPPORT if the address family is not supported.
    172  *  @returns            ENOTSOCK if the socket is not found.
    173  *  @returns            EINVAL if the address is invalid.
    174  *  @returns            ENOTCONN if the sending socket is not and cannot be
    175  *                      bound.
    176  *  @returns            ENOMEM if there is not enough memory left.
    177  *  @returns            Other error codes as defined for the
    178  *                      socket_read_packet_data() function.
    179  *  @returns            Other error codes as defined for the
    180  *                      ip_client_prepare_packet() function.
    181  *  @returns            Other error codes as defined for the ip_send_msg()
    182  *                      function.
    183  */
    184 int
    185 udp_sendto_message(socket_cores_ref local_sockets, int socket_id,
    186     const struct sockaddr * addr, socklen_t addrlen, int fragments,
    187     size_t * data_fragment_size, int flags);
     157 *  @param[in] flags Various send flags.
     158 *  @returns EOK on success.
     159 *  @returns EAFNOTSUPPORT if the address family is not supported.
     160 *  @returns ENOTSOCK if the socket is not found.
     161 *  @returns EINVAL if the address is invalid.
     162 *  @returns ENOTCONN if the sending socket is not and cannot be bound.
     163 *  @returns ENOMEM if there is not enough memory left.
     164 *  @returns Other error codes as defined for the socket_read_packet_data() function.
     165 *  @returns Other error codes as defined for the ip_client_prepare_packet() function.
     166 *  @returns Other error codes as defined for the ip_send_msg() function.
     167 */
     168int udp_sendto_message(socket_cores_ref local_sockets, int socket_id, const struct sockaddr * addr, socklen_t addrlen, int fragments, size_t * data_fragment_size, int flags);
    188169
    189170/** Receives data to the socket.
    190  *
    191171 *  Handles the NET_SOCKET_RECVFROM message.
    192172 *  Replies the source address as well.
    193  *
    194173 *  @param[in] local_sockets The application local sockets.
    195174 *  @param[in] socket_id Socket identifier.
    196  *  @param[in] flags    Various receive flags.
    197  *  @param[out] addrlen The source address length.
    198  *  @returns            The number of bytes received.
    199  *  @returns            ENOTSOCK if the socket is not found.
    200  *  @returns            NO_DATA if there are no received packets or data.
    201  *  @returns            ENOMEM if there is not enough memory left.
    202  *  @returns            EINVAL if the received address is not an IP address.
    203  *  @returns            Other error codes as defined for the packet_translate()
    204  *                      function.
    205  *  @returns            Other error codes as defined for the data_reply()
    206  *                      function.
    207  */
    208 int
    209 udp_recvfrom_message(socket_cores_ref local_sockets, int socket_id, int flags,
    210     size_t * addrlen);
     175 *  @param[in] flags Various receive flags.
     176 *  @param[out] addrlen The source address length.
     177 *  @returns The number of bytes received.
     178 *  @returns ENOTSOCK if the socket is not found.
     179 *  @returns NO_DATA if there are no received packets or data.
     180 *  @returns ENOMEM if there is not enough memory left.
     181 *  @returns EINVAL if the received address is not an IP address.
     182 *  @returns Other error codes as defined for the packet_translate() function.
     183 *  @returns Other error codes as defined for the data_reply() function.
     184 */
     185int udp_recvfrom_message(socket_cores_ref local_sockets, int socket_id, int flags, size_t * addrlen);
    211186
    212187/*@}*/
     
    214189/** UDP global data.
    215190 */
    216 udp_globals_t udp_globals;
    217 
    218 int udp_initialize(async_client_conn_t client_connection)
    219 {
     191udp_globals_t   udp_globals;
     192
     193int udp_initialize(async_client_conn_t client_connection){
    220194        ERROR_DECLARE;
    221195
    222         measured_string_t names[] = {
    223                 {
    224                         str_dup("UDP_CHECKSUM_COMPUTING"),
    225                         22
    226                 },
    227                 {
    228                         str_dup("UDP_AUTOBINDING"),
    229                         15
    230                 }
    231         };
     196        measured_string_t names[] = {{str_dup("UDP_CHECKSUM_COMPUTING"), 22}, {str_dup("UDP_AUTOBINDING"), 15}};
    232197        measured_string_ref configuration;
    233198        size_t count = sizeof(names) / sizeof(measured_string_t);
     
    236201        fibril_rwlock_initialize(&udp_globals.lock);
    237202        fibril_rwlock_write_lock(&udp_globals.lock);
    238 
    239         udp_globals.icmp_phone = icmp_connect_module(SERVICE_ICMP,
    240             ICMP_CONNECT_TIMEOUT);
    241         udp_globals.ip_phone = ip_bind_service(SERVICE_IP, IPPROTO_UDP,
    242             SERVICE_UDP, client_connection, udp_received_msg);
    243         if (udp_globals.ip_phone < 0)
     203        udp_globals.icmp_phone = icmp_connect_module(SERVICE_ICMP, ICMP_CONNECT_TIMEOUT);
     204        udp_globals.ip_phone = ip_bind_service(SERVICE_IP, IPPROTO_UDP, SERVICE_UDP, client_connection, udp_received_msg);
     205        if(udp_globals.ip_phone < 0){
    244206                return udp_globals.ip_phone;
    245 
     207        }
    246208        // read default packet dimensions
    247         ERROR_PROPAGATE(ip_packet_size_req(udp_globals.ip_phone, -1,
    248             &udp_globals.packet_dimension));
     209        ERROR_PROPAGATE(ip_packet_size_req(udp_globals.ip_phone, -1, &udp_globals.packet_dimension));
    249210        ERROR_PROPAGATE(socket_ports_initialize(&udp_globals.sockets));
    250         if (ERROR_OCCURRED(packet_dimensions_initialize(
    251             &udp_globals.dimensions))) {
     211        if(ERROR_OCCURRED(packet_dimensions_initialize(&udp_globals.dimensions))){
    252212                socket_ports_destroy(&udp_globals.sockets);
    253213                return ERROR_CODE;
     
    256216        udp_globals.packet_dimension.content -= sizeof(udp_header_t);
    257217        udp_globals.last_used_port = UDP_FREE_PORTS_START - 1;
    258 
    259218        // get configuration
    260219        udp_globals.checksum_computing = NET_DEFAULT_UDP_CHECKSUM_COMPUTING;
    261220        udp_globals.autobinding = NET_DEFAULT_UDP_AUTOBINDING;
    262221        configuration = &names[0];
    263         ERROR_PROPAGATE(net_get_conf_req(udp_globals.net_phone, &configuration,
    264             count, &data));
    265         if (configuration) {
    266                 if (configuration[0].value)
    267                         udp_globals.checksum_computing =
    268                             (configuration[0].value[0] == 'y');
    269                
    270                 if (configuration[1].value)
    271                         udp_globals.autobinding =
    272                             (configuration[1].value[0] == 'y');
    273 
     222        ERROR_PROPAGATE(net_get_conf_req(udp_globals.net_phone, &configuration, count, &data));
     223        if(configuration){
     224                if(configuration[0].value){
     225                        udp_globals.checksum_computing = (configuration[0].value[0] == 'y');
     226                }
     227                if(configuration[1].value){
     228                        udp_globals.autobinding = (configuration[1].value[0] == 'y');
     229                }
    274230                net_free_settings(configuration, data);
    275231        }
    276 
    277232        fibril_rwlock_write_unlock(&udp_globals.lock);
    278233        return EOK;
    279234}
    280235
    281 int
    282 udp_received_msg(device_id_t device_id, packet_t packet, services_t receiver,
    283     services_t error)
    284 {
     236int udp_received_msg(device_id_t device_id, packet_t packet, services_t receiver, services_t error){
    285237        int result;
    286238
    287239        fibril_rwlock_write_lock(&udp_globals.lock);
    288240        result = udp_process_packet(device_id, packet, error);
    289         if (result != EOK)
     241        if(result != EOK){
    290242                fibril_rwlock_write_unlock(&udp_globals.lock);
     243        }
    291244
    292245        return result;
    293246}
    294247
    295 int udp_process_packet(device_id_t device_id, packet_t packet, services_t error)
    296 {
     248int udp_process_packet(device_id_t device_id, packet_t packet, services_t error){
    297249        ERROR_DECLARE;
    298250
     
    310262        icmp_code_t code;
    311263        void *ip_header;
    312         struct sockaddr *src;
    313         struct sockaddr *dest;
     264        struct sockaddr * src;
     265        struct sockaddr * dest;
    314266        packet_dimension_ref packet_dimension;
    315267
    316         if (error) {
    317                 switch (error) {
    318                 case SERVICE_ICMP:
    319                         // ignore error
    320                         // length = icmp_client_header_length(packet);
    321                         // process error
    322                         result = icmp_client_process_packet(packet, &type,
    323                             &code, NULL, NULL);
    324                         if (result < 0)
    325                                 return udp_release_and_return(packet, result);
    326                         length = (size_t) result;
    327                         if (ERROR_OCCURRED(packet_trim(packet, length, 0)))
    328                                 return udp_release_and_return(packet,
    329                                     ERROR_CODE);
    330                         break;
    331                 default:
    332                         return udp_release_and_return(packet, ENOTSUP);
    333                 }
    334         }
    335 
     268        if(error){
     269                switch(error){
     270                        case SERVICE_ICMP:
     271                                // ignore error
     272                                // length = icmp_client_header_length(packet);
     273                                // process error
     274                                result = icmp_client_process_packet(packet, &type, &code, NULL, NULL);
     275                                if(result < 0){
     276                                        return udp_release_and_return(packet, result);
     277                                }
     278                                length = (size_t) result;
     279                                if(ERROR_OCCURRED(packet_trim(packet, length, 0))){
     280                                        return udp_release_and_return(packet, ERROR_CODE);
     281                                }
     282                                break;
     283                        default:
     284                                return udp_release_and_return(packet, ENOTSUP);
     285                }
     286        }
    336287        // TODO process received ipopts?
    337288        result = ip_client_process_packet(packet, NULL, NULL, NULL, NULL, NULL);
    338         if (result < 0)
     289        if(result < 0){
    339290                return udp_release_and_return(packet, result);
     291        }
    340292        offset = (size_t) result;
    341293
    342294        length = packet_get_data_length(packet);
    343         if (length <= 0)
     295        if(length <= 0){
    344296                return udp_release_and_return(packet, EINVAL);
    345         if (length < UDP_HEADER_SIZE + offset)
     297        }
     298        if(length < UDP_HEADER_SIZE + offset){
    346299                return udp_release_and_return(packet, NO_DATA);
     300        }
    347301
    348302        // trim all but UDP header
    349         if (ERROR_OCCURRED(packet_trim(packet, offset, 0)))
     303        if(ERROR_OCCURRED(packet_trim(packet, offset, 0))){
    350304                return udp_release_and_return(packet, ERROR_CODE);
     305        }
    351306
    352307        // get udp header
    353308        header = (udp_header_ref) packet_get_data(packet);
    354         if (!header)
     309        if(! header){
    355310                return udp_release_and_return(packet, NO_DATA);
    356 
     311        }
    357312        // find the destination socket
    358         socket = socket_port_find(&udp_globals.sockets,
    359         ntohs(header->destination_port), SOCKET_MAP_KEY_LISTENING, 0);
    360         if (!socket) {
    361                 if (tl_prepare_icmp_packet(udp_globals.net_phone,
    362                     udp_globals.icmp_phone, packet, error) == EOK) {
    363                         icmp_destination_unreachable_msg(udp_globals.icmp_phone,
    364                             ICMP_PORT_UNREACH, 0, packet);
     313        socket = socket_port_find(&udp_globals.sockets, ntohs(header->destination_port), SOCKET_MAP_KEY_LISTENING, 0);
     314        if(! socket){
     315                if(tl_prepare_icmp_packet(udp_globals.net_phone, udp_globals.icmp_phone, packet, error) == EOK){
     316                        icmp_destination_unreachable_msg(udp_globals.icmp_phone, ICMP_PORT_UNREACH, 0, packet);
    365317                }
    366318                return EADDRNOTAVAIL;
     
    371323        fragments = 0;
    372324        total_length = ntohs(header->total_length);
    373 
    374325        // compute header checksum if set
    375         if (header->checksum && (!error)) {
    376                 result = packet_get_addr(packet, (uint8_t **) &src,
    377                     (uint8_t **) &dest);
    378                 if( result <= 0)
     326        if(header->checksum && (! error)){
     327                result = packet_get_addr(packet, (uint8_t **) &src, (uint8_t **) &dest);
     328                if(result <= 0){
    379329                        return udp_release_and_return(packet, result);
    380 
    381                 if (ERROR_OCCURRED(ip_client_get_pseudo_header(IPPROTO_UDP,
    382                     src, result, dest, result, total_length, &ip_header,
    383                     &length))) {
     330                }
     331                if(ERROR_OCCURRED(ip_client_get_pseudo_header(IPPROTO_UDP, src, result, dest, result, total_length, &ip_header, &length))){
    384332                        return udp_release_and_return(packet, ERROR_CODE);
    385                 } else {
     333                }else{
    386334                        checksum = compute_checksum(0, ip_header, length);
    387                         // the udp header checksum will be added with the first
    388                         // fragment later
     335                        // the udp header checksum will be added with the first fragment later
    389336                        free(ip_header);
    390337                }
    391         } else {
     338        }else{
    392339                header->checksum = 0;
    393340                checksum = 0;
    394341        }
    395342
    396         do {
     343        do{
    397344                ++ fragments;
    398345                length = packet_get_data_length(next_packet);
    399                 if (length <= 0)
     346                if(length <= 0){
    400347                        return udp_release_and_return(packet, NO_DATA);
    401 
    402                 if (total_length < length) {
    403                         if (ERROR_OCCURRED(packet_trim(next_packet, 0,
    404                             length - total_length))) {
    405                                 return udp_release_and_return(packet,
    406                                     ERROR_CODE);
     348                }
     349                if(total_length < length){
     350                        if(ERROR_OCCURRED(packet_trim(next_packet, 0, length - total_length))){
     351                                return udp_release_and_return(packet, ERROR_CODE);
    407352                        }
    408 
    409353                        // add partial checksum if set
    410                         if (header->checksum) {
    411                                 checksum = compute_checksum(checksum,
    412                                     packet_get_data(packet),
    413                                     packet_get_data_length(packet));
     354                        if(header->checksum){
     355                                checksum = compute_checksum(checksum, packet_get_data(packet), packet_get_data_length(packet));
    414356                        }
    415 
    416357                        // relese the rest of the packet fragments
    417358                        tmp_packet = pq_next(next_packet);
    418                         while (tmp_packet) {
     359                        while(tmp_packet){
    419360                                next_packet = pq_detach(tmp_packet);
    420                                 pq_release_remote(udp_globals.net_phone,
    421                                     packet_get_id(tmp_packet));
     361                                pq_release_remote(udp_globals.net_phone, packet_get_id(tmp_packet));
    422362                                tmp_packet = next_packet;
    423363                        }
    424 
    425364                        // exit the loop
    426365                        break;
    427366                }
    428367                total_length -= length;
    429 
    430368                // add partial checksum if set
    431                 if (header->checksum) {
    432                         checksum = compute_checksum(checksum,
    433                             packet_get_data(packet),
    434                             packet_get_data_length(packet));
    435                 }
    436 
    437         } while ((next_packet = pq_next(next_packet)) && (total_length > 0));
     369                if(header->checksum){
     370                        checksum = compute_checksum(checksum, packet_get_data(packet), packet_get_data_length(packet));
     371                }
     372        }while((next_packet = pq_next(next_packet)) && (total_length > 0));
    438373
    439374        // check checksum
    440         if (header->checksum) {
    441                 if (flip_checksum(compact_checksum(checksum)) !=
    442                     IP_CHECKSUM_ZERO) {
    443                         if (tl_prepare_icmp_packet(udp_globals.net_phone,
    444                             udp_globals.icmp_phone, packet, error) == EOK) {
     375        if(header->checksum){
     376                if(flip_checksum(compact_checksum(checksum)) != IP_CHECKSUM_ZERO){
     377                        if(tl_prepare_icmp_packet(udp_globals.net_phone, udp_globals.icmp_phone, packet, error) == EOK){
    445378                                // checksum error ICMP
    446                                 icmp_parameter_problem_msg(
    447                                     udp_globals.icmp_phone, ICMP_PARAM_POINTER,
    448                                     ((size_t) ((void *) &header->checksum)) -
    449                                     ((size_t) ((void *) header)), packet);
     379                                icmp_parameter_problem_msg(udp_globals.icmp_phone, ICMP_PARAM_POINTER, ((size_t) ((void *) &header->checksum)) - ((size_t) ((void *) header)), packet);
    450380                        }
    451381                        return EINVAL;
     
    454384
    455385        // queue the received packet
    456         if (ERROR_OCCURRED(dyn_fifo_push(&socket->received,
    457             packet_get_id(packet), SOCKET_MAX_RECEIVED_SIZE)) ||
    458             ERROR_OCCURRED(tl_get_ip_packet_dimension(udp_globals.ip_phone,
    459             &udp_globals.dimensions, device_id, &packet_dimension))) {
     386        if(ERROR_OCCURRED(dyn_fifo_push(&socket->received, packet_get_id(packet), SOCKET_MAX_RECEIVED_SIZE))
     387            || ERROR_OCCURRED(tl_get_ip_packet_dimension(udp_globals.ip_phone, &udp_globals.dimensions, device_id, &packet_dimension))){
    460388                return udp_release_and_return(packet, ERROR_CODE);
    461389        }
     
    463391        // notify the destination socket
    464392        fibril_rwlock_write_unlock(&udp_globals.lock);
    465         async_msg_5(socket->phone, NET_SOCKET_RECEIVED,
    466             (ipcarg_t) socket->socket_id, packet_dimension->content, 0, 0,
    467             (ipcarg_t) fragments);
    468 
     393        async_msg_5(socket->phone, NET_SOCKET_RECEIVED, (ipcarg_t) socket->socket_id, packet_dimension->content, 0, 0, (ipcarg_t) fragments);
    469394        return EOK;
    470395}
    471396
    472 int
    473 udp_message_standalone(ipc_callid_t callid, ipc_call_t * call,
    474     ipc_call_t * answer, int * answer_count)
    475 {
     397int udp_message_standalone(ipc_callid_t callid, ipc_call_t * call, ipc_call_t * answer, int * answer_count){
    476398        ERROR_DECLARE;
    477399
     
    479401
    480402        *answer_count = 0;
    481 
    482         switch (IPC_GET_METHOD(*call)) {
    483         case NET_TL_RECEIVED:
    484                 if (!ERROR_OCCURRED(packet_translate_remote(
    485                     udp_globals.net_phone, &packet, IPC_GET_PACKET(call)))) {
    486                         ERROR_CODE = udp_received_msg(IPC_GET_DEVICE(call),
    487                             packet, SERVICE_UDP, IPC_GET_ERROR(call));
    488                 }
    489                 return ERROR_CODE;
    490        
    491         case IPC_M_CONNECT_TO_ME:
    492                 return udp_process_client_messages(callid, * call);
    493         }
    494 
     403        switch(IPC_GET_METHOD(*call)){
     404                case NET_TL_RECEIVED:
     405                        if(! ERROR_OCCURRED(packet_translate_remote(udp_globals.net_phone, &packet, IPC_GET_PACKET(call)))){
     406                                ERROR_CODE = udp_received_msg(IPC_GET_DEVICE(call), packet, SERVICE_UDP, IPC_GET_ERROR(call));
     407                        }
     408                        return ERROR_CODE;
     409                case IPC_M_CONNECT_TO_ME:
     410                        return udp_process_client_messages(callid, * call);
     411        }
    495412        return ENOTSUP;
    496413}
    497414
    498 int udp_process_client_messages(ipc_callid_t callid, ipc_call_t call)
    499 {
     415int udp_process_client_messages(ipc_callid_t callid, ipc_call_t call){
    500416        int res;
    501417        bool keep_on_going = true;
    502418        socket_cores_t local_sockets;
    503419        int app_phone = IPC_GET_PHONE(&call);
    504         struct sockaddr *addr;
     420        struct sockaddr * addr;
    505421        int socket_id;
    506422        size_t addrlen;
     
    517433        answer_count = 0;
    518434
    519         // The client connection is only in one fibril and therefore no
    520         // additional locks are needed.
     435        // The client connection is only in one fibril and therefore no additional locks are needed.
    521436
    522437        socket_cores_initialize(&local_sockets);
    523438
    524         while (keep_on_going) {
     439        while(keep_on_going){
    525440
    526441                // answer the call
     
    534449
    535450                // process the call
    536                 switch (IPC_GET_METHOD(call)) {
    537                 case IPC_M_PHONE_HUNGUP:
    538                         keep_on_going = false;
    539                         res = EHANGUP;
    540                         break;
    541 
    542                 case NET_SOCKET:
    543                         socket_id = SOCKET_GET_SOCKET_ID(call);
    544                         res = socket_create(&local_sockets, app_phone, NULL,
    545                             &socket_id);
    546                         SOCKET_SET_SOCKET_ID(answer, socket_id);
    547 
    548                         if (res != EOK)
     451                switch(IPC_GET_METHOD(call)){
     452                        case IPC_M_PHONE_HUNGUP:
     453                                keep_on_going = false;
     454                                res = EHANGUP;
    549455                                break;
    550                        
    551                         if (tl_get_ip_packet_dimension(udp_globals.ip_phone,
    552                             &udp_globals.dimensions, DEVICE_INVALID_ID,
    553                             &packet_dimension) == EOK) {
    554                                 SOCKET_SET_DATA_FRAGMENT_SIZE(answer,
    555                                     packet_dimension->content);
    556                         }
    557 
    558 //                      SOCKET_SET_DATA_FRAGMENT_SIZE(answer,
    559 //                          MAX_UDP_FRAGMENT_SIZE);
    560                         SOCKET_SET_HEADER_SIZE(answer, UDP_HEADER_SIZE);
    561                         answer_count = 3;
    562                         break;
    563 
    564                 case NET_SOCKET_BIND:
    565                         res = data_receive((void **) &addr, &addrlen);
    566                         if (res != EOK)
     456                        case NET_SOCKET:
     457                                socket_id = SOCKET_GET_SOCKET_ID(call);
     458                                res = socket_create(&local_sockets, app_phone, NULL, &socket_id);
     459                                SOCKET_SET_SOCKET_ID(answer, socket_id);
     460
     461                                if(res == EOK){
     462                                        if (tl_get_ip_packet_dimension(udp_globals.ip_phone, &udp_globals.dimensions, DEVICE_INVALID_ID, &packet_dimension) == EOK){
     463                                                SOCKET_SET_DATA_FRAGMENT_SIZE(answer, packet_dimension->content);
     464                                        }
     465//                                      SOCKET_SET_DATA_FRAGMENT_SIZE(answer, MAX_UDP_FRAGMENT_SIZE);
     466                                        SOCKET_SET_HEADER_SIZE(answer, UDP_HEADER_SIZE);
     467                                        answer_count = 3;
     468                                }
    567469                                break;
    568                         fibril_rwlock_write_lock(&udp_globals.lock);
    569                         res = socket_bind(&local_sockets, &udp_globals.sockets,
    570                             SOCKET_GET_SOCKET_ID(call), addr, addrlen,
    571                             UDP_FREE_PORTS_START, UDP_FREE_PORTS_END,
    572                             udp_globals.last_used_port);
    573                         fibril_rwlock_write_unlock(&udp_globals.lock);
    574                         free(addr);
    575                         break;
    576 
    577                 case NET_SOCKET_SENDTO:
    578                         res = data_receive((void **) &addr, &addrlen);
    579                         if (res != EOK)
     470                        case NET_SOCKET_BIND:
     471                                res = data_receive((void **) &addr, &addrlen);
     472                                if(res == EOK){
     473                                        fibril_rwlock_write_lock(&udp_globals.lock);
     474                                        res = socket_bind(&local_sockets, &udp_globals.sockets, SOCKET_GET_SOCKET_ID(call), addr, addrlen, UDP_FREE_PORTS_START, UDP_FREE_PORTS_END, udp_globals.last_used_port);
     475                                        fibril_rwlock_write_unlock(&udp_globals.lock);
     476                                        free(addr);
     477                                }
    580478                                break;
    581 
    582                         fibril_rwlock_write_lock(&udp_globals.lock);
    583                         res = udp_sendto_message(&local_sockets,
    584                             SOCKET_GET_SOCKET_ID(call), addr, addrlen,
    585                             SOCKET_GET_DATA_FRAGMENTS(call), &size,
    586                             SOCKET_GET_FLAGS(call));
    587                         SOCKET_SET_DATA_FRAGMENT_SIZE(answer, size);
    588 
    589                         if (res != EOK)
     479                        case NET_SOCKET_SENDTO:
     480                                res = data_receive((void **) &addr, &addrlen);
     481                                if(res == EOK){
     482                                        fibril_rwlock_write_lock(&udp_globals.lock);
     483                                        res = udp_sendto_message(&local_sockets, SOCKET_GET_SOCKET_ID(call), addr, addrlen, SOCKET_GET_DATA_FRAGMENTS(call), &size, SOCKET_GET_FLAGS(call));
     484                                        SOCKET_SET_DATA_FRAGMENT_SIZE(answer, size);
     485                                        if(res != EOK){
     486                                                fibril_rwlock_write_unlock(&udp_globals.lock);
     487                                        }else{
     488                                                answer_count = 2;
     489                                        }
     490                                        free(addr);
     491                                }
     492                                break;
     493                        case NET_SOCKET_RECVFROM:
     494                                fibril_rwlock_write_lock(&udp_globals.lock);
     495                                res = udp_recvfrom_message(&local_sockets, SOCKET_GET_SOCKET_ID(call), SOCKET_GET_FLAGS(call), &addrlen);
    590496                                fibril_rwlock_write_unlock(&udp_globals.lock);
    591                         else
    592                                 answer_count = 2;
    593                        
    594                         free(addr);
    595                         break;
    596 
    597                 case NET_SOCKET_RECVFROM:
    598                         fibril_rwlock_write_lock(&udp_globals.lock);
    599                         res = udp_recvfrom_message(&local_sockets,
    600                              SOCKET_GET_SOCKET_ID(call), SOCKET_GET_FLAGS(call),
    601                              &addrlen);
    602                         fibril_rwlock_write_unlock(&udp_globals.lock);
    603 
    604                         if (res <= 0)
     497                                if(res > 0){
     498                                        SOCKET_SET_READ_DATA_LENGTH(answer, res);
     499                                        SOCKET_SET_ADDRESS_LENGTH(answer, addrlen);
     500                                        answer_count = 3;
     501                                        res = EOK;
     502                                }
    605503                                break;
    606 
    607                         SOCKET_SET_READ_DATA_LENGTH(answer, res);
    608                         SOCKET_SET_ADDRESS_LENGTH(answer, addrlen);
    609                         answer_count = 3;
    610                         res = EOK;
    611                         break;
    612                        
    613                 case NET_SOCKET_CLOSE:
    614                         fibril_rwlock_write_lock(&udp_globals.lock);
    615                         res = socket_destroy(udp_globals.net_phone,
    616                             SOCKET_GET_SOCKET_ID(call), &local_sockets,
    617                             &udp_globals.sockets, NULL);
    618                         fibril_rwlock_write_unlock(&udp_globals.lock);
    619                         break;
    620 
    621                 case NET_SOCKET_GETSOCKOPT:
    622                 case NET_SOCKET_SETSOCKOPT:
    623                 default:
    624                         res = ENOTSUP;
    625                         break;
     504                        case NET_SOCKET_CLOSE:
     505                                fibril_rwlock_write_lock(&udp_globals.lock);
     506                                res = socket_destroy(udp_globals.net_phone, SOCKET_GET_SOCKET_ID(call), &local_sockets, &udp_globals.sockets, NULL);
     507                                fibril_rwlock_write_unlock(&udp_globals.lock);
     508                                break;
     509                        case NET_SOCKET_GETSOCKOPT:
     510                        case NET_SOCKET_SETSOCKOPT:
     511                        default:
     512                                res = ENOTSUP;
     513                                break;
    626514                }
    627515        }
     
    631519
    632520        // release all local sockets
    633         socket_cores_release(udp_globals.net_phone, &local_sockets,
    634             &udp_globals.sockets, NULL);
     521        socket_cores_release(udp_globals.net_phone, &local_sockets, &udp_globals.sockets, NULL);
    635522
    636523        return res;
    637524}
    638525
    639 int
    640 udp_sendto_message(socket_cores_ref local_sockets, int socket_id,
    641     const struct sockaddr *addr, socklen_t addrlen, int fragments,
    642     size_t *data_fragment_size, int flags)
    643 {
     526int udp_sendto_message(socket_cores_ref local_sockets, int socket_id, const struct sockaddr * addr, socklen_t addrlen, int fragments, size_t * data_fragment_size, int flags){
    644527        ERROR_DECLARE;
    645528
     
    661544
    662545        socket = socket_cores_find(local_sockets, socket_id);
    663         if (!socket)
     546        if(! socket){
    664547                return ENOTSOCK;
    665 
    666         if ((socket->port <= 0) && udp_globals.autobinding) {
     548        }
     549
     550        if((socket->port <= 0) && udp_globals.autobinding){
    667551                // bind the socket to a random free port if not bound
    668 //              do {
     552//              do{
    669553                        // try to find a free port
    670554//                      fibril_rwlock_read_unlock(&udp_globals.lock);
    671555//                      fibril_rwlock_write_lock(&udp_globals.lock);
    672556                        // might be changed in the meantime
    673 //                      if (socket->port <= 0) {
    674                                 if (ERROR_OCCURRED(socket_bind_free_port(
    675                                     &udp_globals.sockets, socket,
    676                                     UDP_FREE_PORTS_START, UDP_FREE_PORTS_END,
    677                                     udp_globals.last_used_port))) {
    678 //                                      fibril_rwlock_write_unlock(
    679 //                                          &udp_globals.lock);
    680 //                                      fibril_rwlock_read_lock(
    681 //                                          &udp_globals.lock);
     557//                      if(socket->port <= 0){
     558                                if(ERROR_OCCURRED(socket_bind_free_port(&udp_globals.sockets, socket, UDP_FREE_PORTS_START, UDP_FREE_PORTS_END, udp_globals.last_used_port))){
     559//                                      fibril_rwlock_write_unlock(&udp_globals.lock);
     560//                                      fibril_rwlock_read_lock(&udp_globals.lock);
    682561                                        return ERROR_CODE;
    683562                                }
    684                                 // set the next port as the search starting port
    685                                 // number
     563                                // set the next port as the search starting port number
    686564                                udp_globals.last_used_port = socket->port;
    687565//                      }
     
    689567//                      fibril_rwlock_read_lock(&udp_globals.lock);
    690568                        // might be changed in the meantime
    691 //              } while (socket->port <= 0);
    692         }
    693 
    694         if (udp_globals.checksum_computing) {
    695                 if (ERROR_OCCURRED(ip_get_route_req(udp_globals.ip_phone,
    696                     IPPROTO_UDP, addr, addrlen, &device_id, &ip_header,
    697                     &headerlen))) {
     569//              }while(socket->port <= 0);
     570        }
     571
     572        if(udp_globals.checksum_computing){
     573                if(ERROR_OCCURRED(ip_get_route_req(udp_globals.ip_phone, IPPROTO_UDP, addr, addrlen, &device_id, &ip_header, &headerlen))){
    698574                        return udp_release_and_return(packet, ERROR_CODE);
    699575                }
    700576                // get the device packet dimension
    701 //              ERROR_PROPAGATE(tl_get_ip_packet_dimension(udp_globals.ip_phone,
    702 //                  &udp_globals.dimensions, device_id, &packet_dimension));
    703         }
    704 //      } else {
     577//              ERROR_PROPAGATE(tl_get_ip_packet_dimension(udp_globals.ip_phone, &udp_globals.dimensions, device_id, &packet_dimension));
     578        }
     579//      }else{
    705580                // do not ask all the time
    706                 ERROR_PROPAGATE(ip_packet_size_req(udp_globals.ip_phone, -1,
    707                     &udp_globals.packet_dimension));
     581                ERROR_PROPAGATE(ip_packet_size_req(udp_globals.ip_phone, -1, &udp_globals.packet_dimension));
    708582                packet_dimension = &udp_globals.packet_dimension;
    709583//      }
    710584
    711585        // read the first packet fragment
    712         result = tl_socket_read_packet_data(udp_globals.net_phone, &packet,
    713             UDP_HEADER_SIZE, packet_dimension, addr, addrlen);
    714         if (result < 0)
     586        result = tl_socket_read_packet_data(udp_globals.net_phone, &packet, UDP_HEADER_SIZE, packet_dimension, addr, addrlen);
     587        if(result < 0){
    715588                return result;
    716 
     589        }
    717590        total_length = (size_t) result;
    718         if (udp_globals.checksum_computing)
    719                 checksum = compute_checksum(0, packet_get_data(packet),
    720                     packet_get_data_length(packet));
    721         else
     591        if(udp_globals.checksum_computing){
     592                checksum = compute_checksum(0, packet_get_data(packet), packet_get_data_length(packet));
     593        }else{
    722594                checksum = 0;
    723 
     595        }
    724596        // prefix the udp header
    725597        header = PACKET_PREFIX(packet, udp_header_t);
    726         if(! header)
     598        if(! header){
    727599                return udp_release_and_return(packet, ENOMEM);
    728 
     600        }
    729601        bzero(header, sizeof(*header));
    730602        // read the rest of the packet fragments
    731         for (index = 1; index < fragments; ++ index) {
    732                 result = tl_socket_read_packet_data(udp_globals.net_phone,
    733                     &next_packet, 0, packet_dimension, addr, addrlen);
    734                 if (result < 0)
     603        for(index = 1; index < fragments; ++ index){
     604                result = tl_socket_read_packet_data(udp_globals.net_phone, &next_packet, 0, packet_dimension, addr, addrlen);
     605                if(result < 0){
    735606                        return udp_release_and_return(packet, result);
    736 
    737                 if (ERROR_OCCURRED(pq_add(&packet, next_packet, index, 0)))
     607                }
     608                if(ERROR_OCCURRED(pq_add(&packet, next_packet, index, 0))){
    738609                        return udp_release_and_return(packet, ERROR_CODE);
    739 
     610                }
    740611                total_length += (size_t) result;
    741                 if (udp_globals.checksum_computing) {
    742                         checksum = compute_checksum(checksum,
    743                             packet_get_data(next_packet),
    744                             packet_get_data_length(next_packet));
    745                 }
    746         }
    747 
     612                if(udp_globals.checksum_computing){
     613                        checksum = compute_checksum(checksum, packet_get_data(next_packet), packet_get_data_length(next_packet));
     614                }
     615        }
    748616        // set the udp header
    749617        header->source_port = htons((socket->port > 0) ? socket->port : 0);
     
    751619        header->total_length = htons(total_length + sizeof(*header));
    752620        header->checksum = 0;
    753         if (udp_globals.checksum_computing) {
     621        if(udp_globals.checksum_computing){
    754622                // update the pseudo header
    755                 if (ERROR_OCCURRED(ip_client_set_pseudo_header_data_length(
    756                     ip_header, headerlen, total_length + UDP_HEADER_SIZE))) {
     623                if(ERROR_OCCURRED(ip_client_set_pseudo_header_data_length(ip_header, headerlen, total_length + UDP_HEADER_SIZE))){
    757624                        free(ip_header);
    758625                        return udp_release_and_return(packet, ERROR_CODE);
    759626                }
    760 
    761627                // finish the checksum computation
    762628                checksum = compute_checksum(checksum, ip_header, headerlen);
    763                 checksum = compute_checksum(checksum, (uint8_t *) header,
    764                     sizeof(*header));
    765                 header->checksum =
    766                     htons(flip_checksum(compact_checksum(checksum)));
     629                checksum = compute_checksum(checksum, (uint8_t *) header, sizeof(*header));
     630                header->checksum = htons(flip_checksum(compact_checksum(checksum)));
    767631                free(ip_header);
    768         } else {
     632        }else{
    769633                device_id = DEVICE_INVALID_ID;
    770634        }
    771 
    772635        // prepare the first packet fragment
    773         if (ERROR_OCCURRED(ip_client_prepare_packet(packet, IPPROTO_UDP, 0, 0,
    774             0, 0))) {
     636        if(ERROR_OCCURRED(ip_client_prepare_packet(packet, IPPROTO_UDP, 0, 0, 0, 0))){
    775637                return udp_release_and_return(packet, ERROR_CODE);
    776638        }
    777 
    778639        // send the packet
    779640        fibril_rwlock_write_unlock(&udp_globals.lock);
    780641        ip_send_msg(udp_globals.ip_phone, device_id, packet, SERVICE_UDP, 0);
    781 
    782642        return EOK;
    783643}
    784644
    785 int
    786 udp_recvfrom_message(socket_cores_ref local_sockets, int socket_id, int flags,
    787     size_t *addrlen)
    788 {
     645int udp_recvfrom_message(socket_cores_ref local_sockets, int socket_id, int flags, size_t * addrlen){
    789646        ERROR_DECLARE;
    790647
     
    793650        packet_t packet;
    794651        udp_header_ref header;
    795         struct sockaddr *addr;
     652        struct sockaddr * addr;
    796653        size_t length;
    797         uint8_t *data;
     654        uint8_t * data;
    798655        int result;
    799656
    800657        // find the socket
    801658        socket = socket_cores_find(local_sockets, socket_id);
    802         if (!socket)
     659        if(! socket){
    803660                return ENOTSOCK;
    804 
     661        }
    805662        // get the next received packet
    806663        packet_id = dyn_fifo_value(&socket->received);
    807         if (packet_id < 0)
     664        if(packet_id < 0){
    808665                return NO_DATA;
    809 
    810         ERROR_PROPAGATE(packet_translate_remote(udp_globals.net_phone, &packet,
    811             packet_id));
    812 
     666        }
     667        ERROR_PROPAGATE(packet_translate_remote(udp_globals.net_phone, &packet, packet_id));
    813668        // get udp header
    814669        data = packet_get_data(packet);
    815         if (!data) {
     670        if(! data){
    816671                pq_release_remote(udp_globals.net_phone, packet_id);
    817672                return NO_DATA;
     
    821676        // set the source address port
    822677        result = packet_get_addr(packet, (uint8_t **) &addr, NULL);
    823         if (ERROR_OCCURRED(tl_set_address_port(addr, result,
    824             ntohs(header->source_port)))) {
     678        if(ERROR_OCCURRED(tl_set_address_port(addr, result, ntohs(header->source_port)))){
    825679                pq_release_remote(udp_globals.net_phone, packet_id);
    826680                return ERROR_CODE;
    827681        }
    828682        *addrlen = (size_t) result;
    829 
    830683        // send the source address
    831684        ERROR_PROPAGATE(data_reply(addr, * addrlen));
     
    840693        dyn_fifo_pop(&socket->received);
    841694        pq_release_remote(udp_globals.net_phone, packet_get_id(packet));
    842 
    843695        // return the total length
    844696        return (int) length;
    845697}
    846698
    847 int udp_release_and_return(packet_t packet, int result)
    848 {
     699int udp_release_and_return(packet_t packet, int result){
    849700        pq_release_remote(udp_globals.net_phone, packet_get_id(packet));
    850701        return result;
     
    853704/** Default thread for new connections.
    854705 *
    855  *  @param[in] iid      The initial message identifier.
    856  *  @param[in] icall    The initial message call structure.
     706 *  @param[in] iid The initial message identifier.
     707 *  @param[in] icall The initial message call structure.
    857708 *
    858709 */
     
    865716        ipc_answer_0(iid, EOK);
    866717       
    867         while (true) {
     718        while(true) {
    868719                ipc_call_t answer;
    869720                int answer_count;
     
    880731                    &answer_count);
    881732               
    882                 /*
    883                  * End if said to either by the message or the processing result
    884                  */
    885                 if ((IPC_GET_METHOD(call) == IPC_M_PHONE_HUNGUP) ||
    886                     (res == EHANGUP))
     733                /* End if said to either by the message or the processing result */
     734                if ((IPC_GET_METHOD(call) == IPC_M_PHONE_HUNGUP) || (res == EHANGUP))
    887735                        return;
    888736               
     
    894742/** Starts the module.
    895743 *
    896  *  @param argc         The count of the command line arguments. Ignored
    897  *                      parameter.
    898  *  @param argv         The command line parameters. Ignored parameter.
     744 *  @param argc The count of the command line arguments. Ignored parameter.
     745 *  @param argv The command line parameters. Ignored parameter.
    899746 *
    900  *  @returns            EOK on success.
    901  *  @returns            Other error codes as defined for each specific module
    902  *                      start function.
     747 *  @returns EOK on success.
     748 *  @returns Other error codes as defined for each specific module start function.
     749 *
    903750 */
    904751int main(int argc, char *argv[])
Note: See TracChangeset for help on using the changeset viewer.