/** * nativenet.h implementation * * Copyright (C) 2013 Ludwig Ortmann * * This file is subject to the terms and conditions of the GNU Lesser General * Public License. See the file LICENSE in the top level directory for more * details. * * @ingroup native_cpu * @ingroup net * @{ * @file * @author Ludwig Ortmann */ #include #include #include #include #include #include #include #include #define ENABLE_DEBUG (0) #include "debug.h" #include "transceiver.h" #include "msg.h" #include "tap.h" #include "nativenet.h" #include "nativenet_internal.h" #include "cpu.h" struct nativenet_callback_s { void (*func)(void); }; static struct nativenet_callback_s _nativenet_callbacks[255]; struct rx_buffer_s _nativenet_rx_buffer[RX_BUF_SIZE]; static volatile uint8_t rx_buffer_next; uint8_t _native_net_chan; uint16_t _native_net_pan; uint8_t _native_net_monitor; static int _native_net_tpid; radio_address_t _native_net_addr; uint64_t _native_net_addr_long; /************************************************************************/ /* nativenet.h **********************************************************/ /************************************************************************/ void nativenet_init(int transceiver_pid) { DEBUG("nativenet_init(transceiver_pid=%d)\n", transceiver_pid); rx_buffer_next = 0; _native_net_pan = 0; _native_net_chan = 0; _native_net_monitor = 0; _native_net_tpid = transceiver_pid; } void nativenet_powerdown() { return; } void nativenet_set_monitor(uint8_t mode) { DEBUG("nativenet_set_monitor(mode=%d)\n", mode); _native_net_monitor = mode; } int16_t nativenet_set_channel(uint8_t channel) { _native_net_chan = channel; return _native_net_chan; } int16_t nativenet_get_channel() { return _native_net_chan; } uint16_t nativenet_set_pan(uint16_t pan) { _native_net_pan = pan; return _native_net_pan; } uint16_t nativenet_get_pan() { return _native_net_pan; } radio_address_t nativenet_set_address(radio_address_t address) { DEBUG("nativenet_set_address(address=%d)\n", address); _native_net_addr = address; return _native_net_addr; } radio_address_t nativenet_get_address() { DEBUG("nativenet_get_address -> address = %d\n", _native_net_addr); return _native_net_addr; } uint64_t nativenet_get_address_long(void) { DEBUG("nativenet_get_address_long -> address = %" PRIx64 "\n", _native_net_addr_long); return _native_net_addr_long; } uint64_t nativenet_set_address_long(uint64_t address) { DEBUG("nativenet_set_address_long(address=%" PRIx64 ")\n", address); warnx("nativenet_set_address_long: this does not actually change the interfaces address"); _native_net_addr_long = address; return _native_net_addr_long; } int8_t nativenet_send(radio_packet_t *packet) { packet->src = _native_net_addr; DEBUG("nativenet_send: Sending packet of length %" PRIu16 " from %" PRIu16 " to %" PRIu16 "\n", packet->length, packet->src, packet->dst); return send_buf(packet); } void nativenet_switch_to_rx() { return; } /************************************************************************/ /* nativenet_internal.h *************************************************/ /************************************************************************/ int _nativenet_register_cb(int event, void (*func)(void)) { if (event > NNEV_MAXEV) { DEBUG("_nativenet_register_cb: event > NNEV_MAXEV\n"); return -1; } _nativenet_callbacks[event].func = func; return 0; } int _nativenet_unregister_cb(int event) { if (event > NNEV_MAXEV) { DEBUG("_nativenet_unregister_cb: event > NNEV_MAXEV\n"); return -1; } _nativenet_callbacks[event].func = NULL; return 0; } void do_cb(int event) { if (event > NNEV_MAXEV) { DEBUG("do_cb: event > NNEV_MAXEV\n"); return; } if (_nativenet_callbacks[event].func != NULL) { _nativenet_callbacks[event].func(); } } void _nativenet_handle_packet(radio_packet_t *packet) { radio_address_t dst_addr = packet->dst; /* address filter / monitor mode */ if (_native_net_monitor == 1) { DEBUG("_nativenet_handle_packet: monitoring, not filtering address \n"); } else { /* own addr check */ if (dst_addr == _native_net_addr) { DEBUG("_nativenet_handle_packet: accept packet, addressed to us\n"); } else if (dst_addr == 0) { DEBUG("_nativenet_handle_packet: accept packet, broadcast\n"); } else { DEBUG("_nativenet_handle_packet: discard packet addressed to someone else\n"); return; } } /* copy packet to rx buffer */ DEBUG("\n\t\trx_buffer_next: %i\n\n", rx_buffer_next); memcpy(&_nativenet_rx_buffer[rx_buffer_next].data, packet->data, packet->length); memcpy(&_nativenet_rx_buffer[rx_buffer_next].packet, packet, sizeof(radio_packet_t)); _nativenet_rx_buffer[rx_buffer_next].packet.data = (uint8_t *) &_nativenet_rx_buffer[rx_buffer_next].data; /* notify transceiver thread if any */ if (_native_net_tpid) { DEBUG("_nativenet_handle_packet: notifying transceiver thread!\n"); msg_t m; m.type = (uint16_t) RCV_PKT_NATIVE; m.content.value = rx_buffer_next; msg_send_int(&m, _native_net_tpid); } else { DEBUG("_nativenet_handle_packet: no one to notify =(\n"); } /* shift to next buffer element */ if (++rx_buffer_next == RX_BUF_SIZE) { rx_buffer_next = 0; } } /** @} */