diff --git a/Makefile.dep b/Makefile.dep index 153c14b66dce24bc47cc2eb7bccf15a0e95c34b7..9aa67841c8edfae6159bd6bd4306222b1ea688c2 100644 --- a/Makefile.dep +++ b/Makefile.dep @@ -370,6 +370,7 @@ ifneq (,$(filter emb6,$(USEMODULE))) USEMODULE += emb6_ipv6_multicast USEMODULE += emb6_llsec USEMODULE += emb6_mac + USEMODULE += emb6_netdev2 USEMODULE += emb6_rpl USEMODULE += emb6_sicslowpan USEMODULE += emb6_utils diff --git a/pkg/emb6/Makefile.include b/pkg/emb6/Makefile.include index ef653e1a6d7f65522966c8f995cfc9578df81e7b..e1236c8b8d127b46378b07784a114ff77ec1d364 100644 --- a/pkg/emb6/Makefile.include +++ b/pkg/emb6/Makefile.include @@ -42,6 +42,10 @@ ifneq (,$(filter emb6_mac,$(USEMODULE))) INCLUDES += -I$(EMB6_DIR)/emb6/inc/mac endif +ifneq (,$(filter emb6_netdev2,$(USEMODULE))) + DIRS += $(EMB6_CONTRIB)/netdev2 +endif + ifneq (,$(filter emb6_rpl,$(USEMODULE))) DIRS += $(EMB6_DIR)/emb6/src/net/rpl INCLUDES += -I$(EMB6_DIR)/emb6/inc/net/rpl diff --git a/pkg/emb6/contrib/board_conf.c b/pkg/emb6/contrib/board_conf.c index 4096341f3deceb5d695932b58f14ab689e720613..1ba6463a688bb854119a8d39dcfe3abfde589395 100644 --- a/pkg/emb6/contrib/board_conf.c +++ b/pkg/emb6/contrib/board_conf.c @@ -13,6 +13,8 @@ * @author Martine Lenders <mlenders@inf.fu-berlin.de> */ +#include "emb6/netdev2.h" + #include "etimer.h" #include "board_conf.h" @@ -22,6 +24,7 @@ uint8_t board_conf(s_ns_t *ps_nStack) { if (ps_nStack != NULL) { + ps_nStack->inif = &emb6_netdev2_driver; etimer_init(); return ps_nStack->inif->init(ps_nStack); } diff --git a/pkg/emb6/contrib/netdev2/Makefile b/pkg/emb6/contrib/netdev2/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..bdc2e75de0764773d0f519ff415e212db5c40e89 --- /dev/null +++ b/pkg/emb6/contrib/netdev2/Makefile @@ -0,0 +1,3 @@ +MODULE = emb6_netdev2 + +include $(RIOTBASE)/Makefile.base diff --git a/pkg/emb6/contrib/netdev2/emb6_netdev2.c b/pkg/emb6/contrib/netdev2/emb6_netdev2.c new file mode 100644 index 0000000000000000000000000000000000000000..039db6f7627b88806e7390cd75909919ab528f45 --- /dev/null +++ b/pkg/emb6/contrib/netdev2/emb6_netdev2.c @@ -0,0 +1,207 @@ +/* + * Copyright (C) 2016 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. + */ + +/** + * @{ + * + * @file + * @author Martine Lenders <mlenders@inf.fu-berlin.de> + */ + +#include <assert.h> +#include <stdint.h> +#include <sys/uio.h> + +#include "msg.h" +#include "net/netdev2.h" + +#include "evproc.h" +#include "emb6.h" +#include "linkaddr.h" +#include "packetbuf.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +extern uip_lladdr_t uip_lladdr; + +static netdev2_t *_dev = NULL; +static s_nsLowMac_t *_lowmac = NULL; +static int8_t _rssi_base_value = -100; +static uint8_t _last_rssi; + +static int8_t _netdev2_init(s_ns_t *p_ns); +static int8_t _netdev2_send(const void *pr_payload, uint8_t c_len); +static int8_t _netdev2_on(void); +static int8_t _netdev2_off(void); +static void _netdev2_set_txpower(int8_t power); +static int8_t _netdev2_get_txpower(void); +static void _netdev2_set_sensitivity(int8_t sens); +static int8_t _netdev2_get_sensitivity(void); +static int8_t _netdev2_get_rssi(void); +static void _netdev2_set_promisc(uint8_t c_on_off); + +const s_nsIf_t emb6_netdev2_driver = { + .name = "netdev2", + .init = &_netdev2_init, + .send = &_netdev2_send, + .on = &_netdev2_on, + .off = &_netdev2_off, + .set_txpower = &_netdev2_set_txpower, + .get_txpower = &_netdev2_get_txpower, + .set_sensitivity = &_netdev2_set_sensitivity, + .get_sensitivity = &_netdev2_get_sensitivity, + .get_rssi = &_netdev2_get_rssi, + .ant_div = NULL, + .ant_rf_switch = NULL, + .set_promisc = &_netdev2_set_promisc, +}; + +static void _get_recv_pkt(void) +{ + char *dataptr; + struct netdev2_radio_rx_info rx_info; + int8_t len; + + packetbuf_clear(); + + dataptr = packetbuf_dataptr(); + len = _dev->driver->recv(_dev, dataptr, PACKETBUF_SIZE, &rx_info); + _last_rssi = rx_info.rssi; + + if ((len > 0) && (_lowmac != NULL)) { + packetbuf_set_datalen(len); + _lowmac->input(); + } +} + +static void _event_cb(netdev2_t *dev, netdev2_event_t event, void *arg) +{ + (void)arg; + if (event == NETDEV2_EVENT_ISR) { + /* EVENT_TYPE_PCK_LL is supposed to be used by drivers, so use it + * (though NETDEV2_EVENT_ISR technically doesn't only signify + * incoming packets) */ + evproc_putEvent(E_EVPROC_HEAD, EVENT_TYPE_PCK_LL, NULL); + } + else { + switch (event) { + case NETDEV2_EVENT_RX_COMPLETE: { + _get_recv_pkt(); + } + break; + default: + break; + } + } +} + +static void _emb6_netdev2_callback(c_event_t c_event, p_data_t p_data) +{ + (void)p_data; + if (c_event == EVENT_TYPE_PCK_LL) { + _dev->driver->isr(_dev); + } +} + +int emb6_netdev2_setup(netdev2_t *dev) +{ + if (_dev == NULL) { + _dev = dev; + return 0; + } + return -1; +} + +static int8_t _netdev2_init(s_ns_t *p_ns) +{ + if ((_dev != NULL) && (p_ns != NULL) && (p_ns->lmac != NULL)) { + _dev->event_callback = _event_cb; + _dev->driver->get(_dev, NETOPT_ADDRESS_LONG, &mac_phy_config.mac_address, + sizeof(mac_phy_config.mac_address)); + memcpy(&uip_lladdr, mac_phy_config.mac_address, + sizeof(mac_phy_config.mac_address)); + _dev->driver->get(_dev, NETOPT_NID, &mac_phy_config.pan_id, + sizeof(mac_phy_config.pan_id)); + linkaddr_set_node_addr((linkaddr_t *)&uip_lladdr); + _lowmac = p_ns->lmac; + evproc_regCallback(EVENT_TYPE_PCK_LL, _emb6_netdev2_callback); + return 1; + } + return 0; +} + +static int8_t _netdev2_send(const void *pr_payload, uint8_t c_len) +{ + if (_dev != NULL) { + const struct iovec vector = { + .iov_base = (void *)pr_payload, + .iov_len = c_len + }; + if (_dev->driver->send(_dev, &vector, 1) < 0) { + DEBUG("Error on send\n"); + return RADIO_TX_ERR; + } + DEBUG("Packet of length %u was transmitted\n", (unsigned)c_len); + return RADIO_TX_OK; + } + DEBUG("Device was not initialized\n"); + return RADIO_TX_ERR; +} + +static int8_t _netdev2_on(void) +{ + /* TODO: turn netdev2 on */ + return 1; +} + +static int8_t _netdev2_off(void) +{ + /* TODO: turn netdev2 off */ + return 1; +} + +static void _netdev2_set_txpower(int8_t power) +{ + int16_t pwr = power; + + _dev->driver->set(_dev, NETOPT_TX_POWER, &pwr, sizeof(pwr)); +} + +static int8_t _netdev2_get_txpower(void) +{ + int16_t power = 0; + + _dev->driver->get(_dev, NETOPT_TX_POWER, &power, sizeof(power)); + return (int8_t)power; +} + +static void _netdev2_set_sensitivity(int8_t sens) +{ + /* TODO: set sensitivity */ +} + +static int8_t _netdev2_get_sensitivity(void) +{ + /* TODO: get sensitivity */ + return 0; +} + +static int8_t _netdev2_get_rssi(void) +{ + return (int8_t)(_rssi_base_value + 1.03 * _last_rssi); +} + +static void _netdev2_set_promisc(uint8_t c_on_off) +{ + netopt_enable_t en = (c_on_off) ? NETOPT_ENABLE : NETOPT_DISABLE; + + _dev->driver->set(_dev, NETOPT_PROMISCUOUSMODE, &en, sizeof(en)); +} + +/** @} */ diff --git a/pkg/emb6/include/emb6/netdev2.h b/pkg/emb6/include/emb6/netdev2.h new file mode 100644 index 0000000000000000000000000000000000000000..3076603ffcd7b66fad4ac7734934649a06597798 --- /dev/null +++ b/pkg/emb6/include/emb6/netdev2.h @@ -0,0 +1,56 @@ +/* + * Copyright (C) 2016 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. + */ + +/** + * @defgroup pkgemb6_netdev2 netdev2 wrapper for emb6 + * @ingroup pkg_emb6 + * @brief + * @{ + * + * @file + * @brief + * + * @author Martine Lenders <mlenders@inf.fu-berlin.de> + */ +#ifndef EMB6_NETDEV2_H_ +#define EMB6_NETDEV2_H_ + +#include "net/netdev2.h" + +#include "emb6.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief The emb6 network interface. + * + * @note emb6 only supports one network interface. + * + * This variable is used by @ref board_conf() to set the interface for the + * stack. + */ +extern const s_nsIf_t emb6_netdev2_driver; + +/** + * @brief Setup a network device as the emb6 interface. + * + * @param[in] dev The network device for the interface + * + * @return 0 on success. + * @return <= 0 on error. + */ +int emb6_netdev2_setup(netdev2_t *dev); + +#ifdef __cplusplus +} +#endif + +#endif /* EMB6_NETDEV2_H_ */ +/** @} */