/* * Copyright (C) 2015 Freie Universität Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @{ * @ingroup net_ng_netapi * @file * @brief This file contains a number of helper functions that provide * some shortcuts for some always repeating code snippets when * servicing the netapi interface * * @author Hauke Petersen * @} */ #include "kernel.h" #include "msg.h" #include "net/ng_netreg.h" #include "net/ng_pktbuf.h" #include "net/ng_netapi.h" /** * @brief Unified function for getting and setting netapi options * * @param[in] pid PID of the targeted thread * @param[in] type specify if option is to be set or get * @param[in] opt option to set or get * @param[in] data data to set or pointer to buffer for reading options * @param[in] data_len size of the given buffer * * @return the value from the received ACK message */ static inline int _get_set(kernel_pid_t pid, uint16_t type, ng_netopt_t opt, uint16_t context, void *data, size_t data_len) { msg_t cmd; msg_t ack; ng_netapi_opt_t o; /* set ńetapi's option struct */ o.opt = opt; o.context = context; o.data = data; o.data_len = data_len; /* set outgoing message's fields */ cmd.type = type; cmd.content.ptr = (void *)&o; /* trigger the netapi */ msg_send_receive(&cmd, &ack, pid); /* return the ACK message's value */ return (int)ack.content.value; } static inline int _snd_rcv(kernel_pid_t pid, uint16_t type, ng_pktsnip_t *pkt) { msg_t msg; /* set the outgoing message's fields */ msg.type = type; msg.content.ptr = (void *)pkt; /* send message */ return msg_send(&msg, pid); } static inline int _snd_rcv_dispatch(ng_nettype_t type, uint32_t demux_ctx, uint16_t cmd, ng_pktsnip_t *pkt) { int numof = ng_netreg_num(type, demux_ctx); ng_netreg_entry_t *sendto; if (numof != 0) { sendto = ng_netreg_lookup(type, demux_ctx); ng_pktbuf_hold(pkt, numof - 1); while (sendto) { _snd_rcv(sendto->pid, cmd, pkt); sendto = ng_netreg_getnext(sendto); } } return numof; } int ng_netapi_send(kernel_pid_t pid, ng_pktsnip_t *pkt) { return _snd_rcv(pid, NG_NETAPI_MSG_TYPE_SND, pkt); } int ng_netapi_dispatch_send(ng_nettype_t type, uint32_t demux_ctx, ng_pktsnip_t *pkt) { return _snd_rcv_dispatch(type, demux_ctx, NG_NETAPI_MSG_TYPE_SND, pkt); } int ng_netapi_receive(kernel_pid_t pid, ng_pktsnip_t *pkt) { return _snd_rcv(pid, NG_NETAPI_MSG_TYPE_RCV, pkt); } int ng_netapi_dispatch_receive(ng_nettype_t type, uint32_t demux_ctx, ng_pktsnip_t *pkt) { return _snd_rcv_dispatch(type, demux_ctx, NG_NETAPI_MSG_TYPE_RCV, pkt); } int ng_netapi_get(kernel_pid_t pid, ng_netopt_t opt, uint16_t context, void *data, size_t data_len) { return _get_set(pid, NG_NETAPI_MSG_TYPE_GET, opt, context, data, data_len); } int ng_netapi_set(kernel_pid_t pid, ng_netopt_t opt, uint16_t context, void *data, size_t data_len) { return _get_set(pid, NG_NETAPI_MSG_TYPE_SET, opt, context, data, data_len); }