diff --git a/Makefile.dep b/Makefile.dep index 7aef140320b60bda272d4ce02dad0fc4c3be5788..bb5be6fa067b0448ec965188f825fc08fd07ef22 100644 --- a/Makefile.dep +++ b/Makefile.dep @@ -562,6 +562,11 @@ ifneq (,$(filter random,$(USEMODULE))) endif endif +ifneq (,$(filter openthread_contrib,$(USEMODULE))) + USEMODULE += openthread_contrib_netdev + FEATURES_REQUIRED += cpp +endif + ifneq (,$(filter emcute,$(USEMODULE))) USEMODULE += core_thread_flags USEMODULE += sock_udp diff --git a/drivers/at86rf2xx/at86rf2xx_netdev.c b/drivers/at86rf2xx/at86rf2xx_netdev.c index 6e165ae51592c6a6d6aeb938d375ac91d18f5f8b..a9452c85a7358d4134de1ecf89b184e0dee2acf8 100644 --- a/drivers/at86rf2xx/at86rf2xx_netdev.c +++ b/drivers/at86rf2xx/at86rf2xx_netdev.c @@ -565,11 +565,22 @@ static void _isr(netdev_t *netdev) if (netdev->event_callback && (dev->netdev.flags & AT86RF2XX_OPT_TELL_TX_END)) { switch (trac_status) { +#ifdef MODULE_OPENTHREAD + case AT86RF2XX_TRX_STATE__TRAC_SUCCESS: + netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE); + DEBUG("[at86rf2xx] TX SUCCESS\n"); + break; + case AT86RF2XX_TRX_STATE__TRAC_SUCCESS_DATA_PENDING: + netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE_DATA_PENDING); + DEBUG("[at86rf2xx] TX SUCCESS DATA PENDING\n"); + break; +#else case AT86RF2XX_TRX_STATE__TRAC_SUCCESS: case AT86RF2XX_TRX_STATE__TRAC_SUCCESS_DATA_PENDING: netdev->event_callback(netdev, NETDEV_EVENT_TX_COMPLETE); DEBUG("[at86rf2xx] TX SUCCESS\n"); break; +#endif case AT86RF2XX_TRX_STATE__TRAC_NO_ACK: netdev->event_callback(netdev, NETDEV_EVENT_TX_NOACK); DEBUG("[at86rf2xx] TX NO_ACK\n"); diff --git a/drivers/include/net/netdev.h b/drivers/include/net/netdev.h index 46d9aa7caa63e3a9c083d7b9817c642835544ff6..09a7846b53ad407932cb51fa36de95d3695ccd2a 100644 --- a/drivers/include/net/netdev.h +++ b/drivers/include/net/netdev.h @@ -70,15 +70,16 @@ enum { * upper layer */ typedef enum { - NETDEV_EVENT_ISR, /**< driver needs it's ISR handled */ - NETDEV_EVENT_RX_STARTED, /**< started to receive a packet */ - NETDEV_EVENT_RX_COMPLETE, /**< finished receiving a packet */ - NETDEV_EVENT_TX_STARTED, /**< started to transfer a packet */ - NETDEV_EVENT_TX_COMPLETE, /**< finished transferring packet */ - NETDEV_EVENT_TX_NOACK, /**< ACK requested but not received */ - NETDEV_EVENT_TX_MEDIUM_BUSY, /**< couldn't transfer packet */ - NETDEV_EVENT_LINK_UP, /**< link established */ - NETDEV_EVENT_LINK_DOWN, /**< link gone */ + NETDEV_EVENT_ISR, /**< driver needs it's ISR handled */ + NETDEV_EVENT_RX_STARTED, /**< started to receive a packet */ + NETDEV_EVENT_RX_COMPLETE, /**< finished receiving a packet */ + NETDEV_EVENT_TX_STARTED, /**< started to transfer a packet */ + NETDEV_EVENT_TX_COMPLETE, /**< transfer packet complete */ + NETDEV_EVENT_TX_COMPLETE_DATA_PENDING, /**< transfer packet complete and data pending flag */ + NETDEV_EVENT_TX_NOACK, /**< ACK requested but not received */ + NETDEV_EVENT_TX_MEDIUM_BUSY, /**< couldn't transfer packet */ + NETDEV_EVENT_LINK_UP, /**< link established */ + NETDEV_EVENT_LINK_DOWN, /**< link gone */ /* expand this list if needed */ } netdev_event_t; diff --git a/makefiles/pseudomodules.inc.mk b/makefiles/pseudomodules.inc.mk index d922c62116584ba9c25d3248254289d380bd503e..f94ac7eca09f845414641369e3d5b9f7e234121b 100644 --- a/makefiles/pseudomodules.inc.mk +++ b/makefiles/pseudomodules.inc.mk @@ -41,6 +41,7 @@ PSEUDOMODULES += netstats_ipv6 PSEUDOMODULES += netstats_rpl PSEUDOMODULES += newlib PSEUDOMODULES += newlib_nano +PSEUDOMODULES += openthread PSEUDOMODULES += pktqueue PSEUDOMODULES += posix PSEUDOMODULES += printf_float diff --git a/pkg/openthread/Makefile b/pkg/openthread/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..8d59f1dec345782aea55db1e81dcc91ed4f714c9 --- /dev/null +++ b/pkg/openthread/Makefile @@ -0,0 +1,32 @@ +PKG_NAME=openthread +PKG_URL=https://github.com/openthread/openthread.git +PKG_VERSION=fbfd76a990b81f007957e1bd774e51bce742e53e +PKG_BUILDDIR ?= $(BINDIRBASE)/pkg/$(BOARD)/$(PKG_NAME) + +$(info Compile OpenThread for FTD device) +OPENTHREAD_ARGS+= --enable-cli-app=ftd --enable-application-coap + +$(info $$OPENTHREAD_ARGS is [${OPENTHREAD_ARGS}]) + +.PHONY: all + +OPENTHREAD_COMMON_FLAGS=-fdata-sections -ffunction-sections -Os +all: git-download + cd $(PKG_BUILDDIR) && PREFIX="/" ./bootstrap + cd $(PKG_BUILDDIR) && CPP="$(CPP)" CC="$(CC)" CXX="$(CXX)"\ + OBJC="" OBJCXX="" AR="$(AR)" RANLIB="$(RANLIB)" NM="$(NM)" \ + STRIP="$(STRIP)" \ + CPPFLAGS="$(OPENTHREAD_COMMON_FLAGS) $(CFLAGS_CPU) " \ + CFLAGS="$(OPENTHREAD_COMMON_FLAGS) $(CFLAGS_CPU) " \ + CXXFLAGS="$(OPENTHREAD_COMMON_FLAGS) $(CFLAGS_CPU) -fno-exceptions -fno-rtti " \ + LDFLAGS="$(OPENTHREAD_COMMON_FLAGS) $(CFLAGS_CPU) -nostartfiles -specs=nano.specs \ + -specs=nosys.specs -Wl,--gc-sections -Wl,-Map=map.map " \ + ./configure --disable-docs --host=$(TARGET_ARCH) --target=$(TARGET_ARCH) \ + --prefix=/ --enable-default-logging ${OPENTHREAD_ARGS} + cd $(PKG_BUILDDIR) && DESTDIR=$(PKG_BUILDDIR)/output PREFIX=/ make -j4 --no-print-directory install + + cp $(PKG_BUILDDIR)/output/lib/libmbedcrypto.a ${BINDIR}/libmbedcrypto.a + cp $(PKG_BUILDDIR)/output/lib/libopenthread-ftd.a ${BINDIR}/libopenthread.a + cp $(PKG_BUILDDIR)/output/lib/libopenthread-cli-ftd.a ${BINDIR}/libopenthread-cli.a + sed -ie 's/BASE/_BASE/g' $(PKG_BUILDDIR)/output/include/openthread/types.h +include $(RIOTBASE)/pkg/pkg.mk diff --git a/pkg/openthread/Makefile.include b/pkg/openthread/Makefile.include new file mode 100644 index 0000000000000000000000000000000000000000..b4eebf43654d6e2feb533c16d12f5e2bcd29f4bd --- /dev/null +++ b/pkg/openthread/Makefile.include @@ -0,0 +1,11 @@ +OPENTHREAD_DIR = $(RIOTBASE)/pkg/openthread + +INCLUDES += -I$(OPENTHREAD_DIR)/include \ + -I$(OPENTHREAD_DIR)/include/openthread \ + -I$(BINDIRBASE)/pkg/$(BOARD)/openthread/output/include \ + -I$(BINDIRBASE)/pkg/$(BOARD)/openthread/include/openthread \ + +ifneq (,$(filter openthread_contrib,$(USEMODULE))) + DIRS += $(OPENTHREAD_DIR)/contrib + DIRS += $(OPENTHREAD_DIR)/contrib/netdev +endif diff --git a/pkg/openthread/contrib/Makefile b/pkg/openthread/contrib/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..6533faeb39bd379447a04809aab64df2e7a94daf --- /dev/null +++ b/pkg/openthread/contrib/Makefile @@ -0,0 +1,3 @@ +MODULE := openthread_contrib + +include $(RIOTBASE)/Makefile.base diff --git a/pkg/openthread/contrib/netdev/Makefile b/pkg/openthread/contrib/netdev/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..7f8ff815f52373d8457b2229ebc7ba36cb3895fd --- /dev/null +++ b/pkg/openthread/contrib/netdev/Makefile @@ -0,0 +1,3 @@ +MODULE := openthread_contrib_netdev + +include $(RIOTBASE)/Makefile.base diff --git a/pkg/openthread/contrib/netdev/openthread_netdev.c b/pkg/openthread/contrib/netdev/openthread_netdev.c new file mode 100644 index 0000000000000000000000000000000000000000..6c294cd207d937ca5dececb59d5cd3eab44e7a1e --- /dev/null +++ b/pkg/openthread/contrib/netdev/openthread_netdev.c @@ -0,0 +1,186 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Netdev adoption for OpenThread + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <assert.h> +#include <errno.h> +#include <string.h> +#include "msg.h" +#include "openthread/cli.h" +#include "openthread/instance.h" +#include "openthread/ip6.h" +#include "openthread/platform/alarm.h" +#include "openthread/platform/uart.h" +#include "openthread/tasklet.h" +#include "openthread/thread.h" +#include "random.h" +#include "ot.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +#define OPENTHREAD_QUEUE_LEN (8) +static msg_t _queue[OPENTHREAD_QUEUE_LEN]; + +static kernel_pid_t _pid; +static otInstance *sInstance; + +/** + * @name Default configuration for OpenThread network + * @{ + */ +#ifndef OPENTHREAD_PANID +#define OPENTHREAD_PANID 0x1234 +#endif +#ifndef OPENTHREAD_CHANNEL +#define OPENTHREAD_CHANNEL (26U) +#endif +/** @} */ + +uint8_t ot_call_command(char* command, void *arg, void* answer) { + ot_job_t job; + + job.command = command; + job.arg = arg; + job.answer = answer; + + msg_t msg, reply; + msg.type = OPENTHREAD_JOB_MSG_TYPE_EVENT; + msg.content.ptr = &job; + msg_send_receive(&msg, &reply, openthread_get_pid()); + return (uint8_t)reply.content.value; +} + +/* OpenThread will call this when switching state from empty tasklet to non-empty tasklet. */ +void otTaskletsSignalPending(otInstance *aInstance) { + otTaskletsProcess(aInstance); +} + +static void *_openthread_event_loop(void *arg) { + _pid = thread_getpid(); + + /* enable OpenThread UART */ + otPlatUartEnable(); + + /* init OpenThread */ + sInstance = otInstanceInit(); + + msg_init_queue(_queue, OPENTHREAD_QUEUE_LEN); + netdev_t *dev; + msg_t msg, reply; + + otCliUartInit(sInstance); + +#if OPENTHREAD_ENABLE_DIAG + diagInit(sInstance); +#endif + + /* Init default parameters */ + otPanId panid = OPENTHREAD_PANID; + uint8_t channel = OPENTHREAD_CHANNEL; + otLinkSetPanId(sInstance, panid); + otLinkSetChannel(sInstance, channel); + /* Bring up the IPv6 interface */ + otIp6SetEnabled(sInstance, true); + /* Start Thread protocol operation */ + otThreadSetEnabled(sInstance, true); + + uint8_t *buf; + ot_job_t *job; + while (1) { + msg_receive(&msg); + switch (msg.type) { + case OPENTHREAD_XTIMER_MSG_TYPE_EVENT: + /* Tell OpenThread a time event was received */ + otPlatAlarmFired(sInstance); + break; + case OPENTHREAD_NETDEV_MSG_TYPE_EVENT: + /* Received an event from driver */ + dev = msg.content.ptr; + dev->driver->isr(dev); + break; + case OPENTHREAD_SERIAL_MSG_TYPE_EVENT: + /* Tell OpenThread about the reception of a CLI command */ + buf = msg.content.ptr; + otPlatUartReceived(buf, strlen((char *) buf)); + break; + case OPENTHREAD_JOB_MSG_TYPE_EVENT: + job = msg.content.ptr; + reply.content.value = ot_exec_command(sInstance, job->command, job->arg, job->answer); + msg_reply(&msg, &reply); + break; + } + } + + return NULL; +} + +static void _event_cb(netdev_t *dev, netdev_event_t event) { + switch (event) { + case NETDEV_EVENT_ISR: + { + msg_t msg; + assert(_pid != KERNEL_PID_UNDEF); + + msg.type = OPENTHREAD_NETDEV_MSG_TYPE_EVENT; + msg.content.ptr = dev; + + if (msg_send(&msg, _pid) <= 0) { + DEBUG("openthread_netdev: possibly lost interrupt.\n"); + } + break; + } + + case NETDEV_EVENT_RX_COMPLETE: + DEBUG("openthread_netdev: Reception of a packet\n"); + recv_pkt(sInstance, dev); + break; + case NETDEV_EVENT_TX_COMPLETE: + case NETDEV_EVENT_TX_NOACK: + case NETDEV_EVENT_TX_MEDIUM_BUSY: + DEBUG("openthread_netdev: Transmission of a packet\n"); + send_pkt(sInstance, dev, event); + break; + default: + break; + } +} + +/* get OpenThread thread pid */ +kernel_pid_t openthread_get_pid(void) { + return _pid; +} + +/* starts OpenThread thread */ +int openthread_netdev_init(char *stack, int stacksize, char priority, + const char *name, netdev_t *netdev) { + netdev->driver->init(netdev); + netdev->event_callback = _event_cb; + + netopt_enable_t enable = NETOPT_ENABLE; + netdev->driver->set(netdev, NETOPT_TX_END_IRQ, &enable, sizeof(enable)); + + _pid = thread_create(stack, stacksize, + priority, THREAD_CREATE_STACKTEST, + _openthread_event_loop, NULL, name); + + if (_pid <= 0) { + return -EINVAL; + } + + return _pid; +} diff --git a/pkg/openthread/contrib/openthread.c b/pkg/openthread/contrib/openthread.c new file mode 100644 index 0000000000000000000000000000000000000000..882bb9bd7ae69a4a059cfc02084e200079fc2a15 --- /dev/null +++ b/pkg/openthread/contrib/openthread.c @@ -0,0 +1,80 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread main functions + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <assert.h> + +#include "openthread/platform/alarm.h" +#include "openthread/platform/uart.h" +#include "ot.h" +#include "random.h" +#include "thread.h" +#include "xtimer.h" + +#ifdef MODULE_AT86RF2XX +#include "at86rf2xx.h" +#include "at86rf2xx_params.h" +#endif + +#define ENABLE_DEBUG (0) +#include "debug.h" + +#ifdef MODULE_AT86RF2XX /* is mutual exclusive with above ifdef */ +#define OPENTHREAD_NETIF_NUMOF (sizeof(at86rf2xx_params) / sizeof(at86rf2xx_params[0])) +#endif + +#ifdef MODULE_AT86RF2XX +static at86rf2xx_t at86rf2xx_dev; +#endif + +#define OPENTHREAD_NETDEV_BUFLEN (ETHERNET_MAX_LEN) + +static uint8_t rx_buf[OPENTHREAD_NETDEV_BUFLEN]; +static uint8_t tx_buf[OPENTHREAD_NETDEV_BUFLEN]; +static char ot_thread_stack[2 * THREAD_STACKSIZE_MAIN]; + +/* init and run OpeanThread's UART simulation (stdio) */ +void openthread_uart_run(void) +{ + char buf[256]; + msg_t msg; + + msg.type = OPENTHREAD_SERIAL_MSG_TYPE_EVENT; + msg.content.ptr = buf; + + buf[1] = 0; + while (1) { + char c = getchar(); + buf[0] = c; + msg_send(&msg, openthread_get_pid()); + } +} + +void openthread_bootstrap(void) +{ + /* init random */ + ot_random_init(); + + /* setup netdev modules */ +#ifdef MODULE_AT86RF2XX + at86rf2xx_setup(&at86rf2xx_dev, &at86rf2xx_params[0]); + netdev_t *netdev = (netdev_t *) &at86rf2xx_dev; +#endif + + openthread_radio_init(netdev, tx_buf, rx_buf); + openthread_netdev_init(ot_thread_stack, sizeof(ot_thread_stack), THREAD_PRIORITY_MAIN - 5, "openthread", netdev); +} diff --git a/pkg/openthread/contrib/platform_alarm.c b/pkg/openthread/contrib/platform_alarm.c new file mode 100644 index 0000000000000000000000000000000000000000..6a6f3d6f32258355a586e795e7220034ed86485f --- /dev/null +++ b/pkg/openthread/contrib/platform_alarm.c @@ -0,0 +1,68 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread alarm platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <stdint.h> + +#include "msg.h" +#include "openthread/platform/alarm.h" +#include "ot.h" +#include "thread.h" +#include "xtimer.h" +#include "timex.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +static xtimer_t ot_timer; +static msg_t ot_alarm_msg; + +/** + * Set the alarm to fire at @p aDt milliseconds after @p aT0. + * + * @param[in] aInstance The OpenThread instance structure. + * @param[in] aT0 The reference time. + * @param[in] aDt The time delay in milliseconds from @p aT0. + */ +void otPlatAlarmStartAt(otInstance *aInstance, uint32_t aT0, uint32_t aDt) +{ + DEBUG("openthread: otPlatAlarmStartAt: aT0: %" PRIu32 ", aDT: %" PRIu32 "\n", aT0, aDt); + ot_alarm_msg.type = OPENTHREAD_XTIMER_MSG_TYPE_EVENT; + + if (aDt == 0) { + msg_send(&ot_alarm_msg, thread_getpid()); + } + else { + int dt = aDt * US_PER_MS; + xtimer_set_msg(&ot_timer, dt, &ot_alarm_msg, thread_getpid()); + } +} + +/* OpenThread will call this to stop alarms */ +void otPlatAlarmStop(otInstance *aInstance) +{ + DEBUG("openthread: otPlatAlarmStop\n"); + xtimer_remove(&ot_timer); +} + +/* OpenThread will call this for getting running time in millisecs */ +uint32_t otPlatAlarmGetNow(void) +{ + uint32_t now = xtimer_now_usec() / US_PER_MS; + DEBUG("openthread: otPlatAlarmGetNow: %" PRIu32 "\n", now); + return now; +} diff --git a/pkg/openthread/contrib/platform_diag.c b/pkg/openthread/contrib/platform_diag.c new file mode 100644 index 0000000000000000000000000000000000000000..63092fb0216490578a8ac6cb8b19ac5dac9a22a7 --- /dev/null +++ b/pkg/openthread/contrib/platform_diag.c @@ -0,0 +1,38 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread diagnostics platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <stdbool.h> +#include <stdio.h> + +static bool sDiagMode = false; + +void otPlatDiagProcess(int argc, char *argv[], char *aOutput, size_t aOutputMaxLen) +{ + /* add more plarform specific diagnostics features here */ + (void)argc; +} + +void otPlatDiagModeSet(bool aMode) +{ + sDiagMode = aMode; +} + +bool otPlatDiagModeGet(void) +{ + return sDiagMode; +} diff --git a/pkg/openthread/contrib/platform_functions_wrapper.c b/pkg/openthread/contrib/platform_functions_wrapper.c new file mode 100644 index 0000000000000000000000000000000000000000..e3eb45650bad18ac28088863660f5731beae9c8d --- /dev/null +++ b/pkg/openthread/contrib/platform_functions_wrapper.c @@ -0,0 +1,297 @@ +/* + * Copyright (C) + * + * 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 + * @file + * @brief Implementation of OpenThread functions wrapper. They are used to call OT functions from OT thread + * + * @author Jose Ignacio Alamos <jialamos@inria.cl> + * @author Baptiste CLENET <bapclenet@gmail.com> + * @} + */ + +#include <stdint.h> +#include <stdio.h> +#include "thread.h" +#include "openthread/ip6.h" +#include "openthread/thread.h" +#include "openthread/udp.h" +#include "ot.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +typedef uint8_t OT_COMMAND; + +OT_COMMAND ot_channel(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_eui64(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_extaddr(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_ipaddr(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_masterkey(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_networkname(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_mode(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_panid(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_parent(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_state(otInstance* ot_instance, void* arg, void* answer); +OT_COMMAND ot_thread(otInstance* ot_instance, void* arg, void* answer); + +/** + * @brief Struct containing an OpenThread job command + */ +typedef struct { + const char *name; /**< A pointer to the job name string. */ + OT_COMMAND (*function)(otInstance*, void*, void*); /**< function to be called */ +} ot_command_t; + +const ot_command_t otCommands[] = +{ + /* channel: arg NULL: get channel in answer | arg not NULL: set channel */ + { "channel", &ot_channel }, + /* eui64 : arg NULL: get eui64 in answer | arg not NULL: set eui64 */ + { "eui64", &ot_eui64 }, + /* extaddr: arg NULL: get extaddr in answer | arg not NULL: set extaddr */ + { "extaddr", &ot_extaddr }, + /* ipaddr: arg NULL: get nb ipaddr in answer | arg not NULL: get ipaddr[arg] */ + { "ipaddr", &ot_ipaddr }, + /* masterkey: arg NULL: get masterkey in answer | arg not NULL: set masterkey */ + { "masterkey", &ot_masterkey }, + /* mode: arg NULL: get mode in answer | arg not NULL: set mode */ + { "mode", ot_mode }, + /* networkname: arg NULL: get networkname in answer | arg not NULL: set networkname */ + { "networkname", &ot_networkname }, + /* panid: arg NULL: get panid in answer | arg not NULL: set panid */ + { "panid", &ot_panid }, + /* parent: arg NULL: get parent in answer */ + { "parent", &ot_parent }, + /* state: arg NULL: get state in answer */ + { "state", &ot_state }, + /* thread: arg "start"/"stop": start/stop thread operation */ + { "thread", &ot_thread }, +}; + +uint8_t ot_exec_command(otInstance *ot_instance, const char* command, void *arg, void* answer) { + uint8_t res = 0xFF; + /* Check running thread */ + if (openthread_get_pid() == thread_getpid()) { + for (uint8_t i = 0; i < sizeof(otCommands) / sizeof(otCommands[0]); i++) { + if (strcmp(command, otCommands[i].name) == 0) { + res = (*otCommands[i].function)(ot_instance, arg, answer); + break; + } + } + if (res == 0xFF) { + DEBUG("Wrong ot_COMMAND name\n"); + res = 1; + } + } else { + DEBUG("ERROR: ot_exec_job needs to run in OpenThread thread\n"); + } + return res; +} + +void output_bytes(const char* name, const uint8_t *aBytes, uint8_t aLength) +{ + DEBUG("%s: ", name); + for (int i = 0; i < aLength; i++) { + DEBUG("%02x", aBytes[i]); + } + DEBUG("\n"); +} + +OT_COMMAND ot_channel(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + *((uint8_t *) answer) = otLinkGetChannel(ot_instance); + DEBUG("Channel: %04x\n", *((uint8_t *) answer)); + } else if (arg != NULL) { + uint8_t channel = *((uint8_t *) arg); + otLinkSetChannel(ot_instance, channel); + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_eui64(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + otExtAddress address; + otLinkGetFactoryAssignedIeeeEui64(ot_instance, &address); + output_bytes("eui64", address.m8, OT_EXT_ADDRESS_SIZE); + *((otExtAddress *) answer) = address; + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + + +OT_COMMAND ot_extaddr(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + answer = (void*)otLinkGetExtendedAddress(ot_instance); + output_bytes("extaddr", (const uint8_t *)answer, OT_EXT_ADDRESS_SIZE); + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_ipaddr(otInstance* ot_instance, void* arg, void* answer) { + uint8_t cnt = 0; + for (const otNetifAddress *addr = otIp6GetUnicastAddresses(ot_instance); addr; addr = addr->mNext) { + if (arg != NULL && answer != NULL && cnt == *((uint8_t *) arg)) { + *((otNetifAddress *) answer) = *addr; + return 0; + } + cnt++; + } + if (answer != NULL) { + *((uint8_t *) answer) = cnt; + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_masterkey(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + const otMasterKey* masterkey = otThreadGetMasterKey(ot_instance); + *((otMasterKey *) answer) = *masterkey; + output_bytes("masterkey", (const uint8_t *)answer, OT_MASTER_KEY_SIZE); + } else if (arg != NULL) { + otThreadSetMasterKey(ot_instance, (otMasterKey*)arg); + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_mode(otInstance* ot_instance, void* arg, void* answer) { + if (arg != NULL) { + otLinkModeConfig link_mode; + memset(&link_mode, 0, sizeof(otLinkModeConfig)); + char mode[6]; + memcpy(mode, (char*)arg, 5); + mode[5] = '\0'; + for (char *arg = &mode[0]; *arg != '\0'; arg++) { + switch (*arg) { + case 'r': + link_mode.mRxOnWhenIdle = 1; + break; + case 's': + link_mode.mSecureDataRequests = 1; + break; + case 'd': + link_mode.mDeviceType = 1; + break; + case 'n': + link_mode.mNetworkData = 1; + break; + } + } + otThreadSetLinkMode(ot_instance, link_mode); + DEBUG("OT mode changed to %s\n", (char*)arg); + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_networkname(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + const char* networkName = otThreadGetNetworkName(ot_instance); + strcpy((char*) answer, networkName); + DEBUG("networkname: %.*s\n", OT_NETWORK_NAME_MAX_SIZE, networkName); + } else if (arg != NULL) { + otThreadSetNetworkName(ot_instance, (char*) arg); + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + + +OT_COMMAND ot_panid(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + *((uint16_t *) answer) = otLinkGetPanId(ot_instance); + DEBUG("PanID: %04x\n", *((uint16_t *) answer)); + } else if (arg != NULL) { + /* Thread operation needs to be stopped before setting panid */ + otThreadSetEnabled(ot_instance, false); + uint16_t panid = *((uint16_t *) arg); + otLinkSetPanId(ot_instance, panid); + otThreadSetEnabled(ot_instance, true); + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_parent(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + otRouterInfo parentInfo; + otThreadGetParentInfo(ot_instance, &parentInfo); + output_bytes("parent", (const uint8_t *)parentInfo.mExtAddress.m8, sizeof(parentInfo.mExtAddress)); + DEBUG("Rloc: %x\n", parentInfo.mRloc16); + *((otRouterInfo *) answer) = parentInfo; + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_state(otInstance* ot_instance, void* arg, void* answer) { + if (answer != NULL) { + uint8_t state = otThreadGetDeviceRole(ot_instance); + *((uint8_t *) answer) = state; + DEBUG("state: "); + switch (state) { + case kDeviceRoleOffline: + puts("offline"); + break; + case kDeviceRoleDisabled: + puts("disabled"); + break; + case kDeviceRoleDetached: + puts("detached"); + break; + case kDeviceRoleChild: + puts("child"); + break; + case kDeviceRoleRouter: + puts("router"); + break; + case kDeviceRoleLeader: + puts("leader"); + break; + default: + puts("invalid state"); + break; + } + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} + +OT_COMMAND ot_thread(otInstance* ot_instance, void* arg, void* answer) { + if (arg != NULL) { + if (strcmp((char*)arg, "start") == 0) { + otThreadSetEnabled(ot_instance, true); + DEBUG("Thread start\n"); + } else if (strcmp((char*)arg, "stop") == 0) { + otThreadSetEnabled(ot_instance, false); + DEBUG("Thread stop\n"); + } else { + DEBUG("ERROR: thread available args: start/stop\n"); + } + } else { + DEBUG("ERROR: wrong argument\n"); + } + return 0; +} diff --git a/pkg/openthread/contrib/platform_logging.c b/pkg/openthread/contrib/platform_logging.c new file mode 100644 index 0000000000000000000000000000000000000000..a3aed1db7d04d971376b5abed7dd43bda5c19abd --- /dev/null +++ b/pkg/openthread/contrib/platform_logging.c @@ -0,0 +1,97 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread logging platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <ctype.h> +#include <inttypes.h> +#include <stdarg.h> +#include <stdio.h> +#include <stdint.h> +#include <string.h> +#include <time.h> + +#include "openthread/platform/logging.h" + +/* adapted from OpenThread posix example: + * See: https://github.com/openthread/openthread/blob/master/examples/platforms/posix/logging.c */ +void otPlatLog(otLogLevel aLogLevel, otLogRegion aLogRegion, const char *aFormat, ...) +{ + va_list args; + + switch (aLogLevel) { + case kLogLevelNone: + fprintf(stderr, "NONE "); + break; + + case kLogLevelCrit: + fprintf(stderr, "CRIT "); + break; + + case kLogLevelWarn: + fprintf(stderr, "WARN "); + break; + + case kLogLevelInfo: + fprintf(stderr, "INFO "); + break; + + case kLogLevelDebg: + fprintf(stderr, "DEBG "); + break; + } + + switch (aLogRegion) { + case kLogRegionApi: + fprintf(stderr, "API "); + break; + + case kLogRegionMle: + fprintf(stderr, "MLE "); + break; + + case kLogRegionArp: + fprintf(stderr, "ARP "); + break; + + case kLogRegionNetData: + fprintf(stderr, "NETD "); + break; + + case kLogRegionIp6: + fprintf(stderr, "IPV6 "); + break; + + case kLogRegionIcmp: + fprintf(stderr, "ICMP "); + break; + + case kLogRegionMac: + fprintf(stderr, "MAC "); + break; + + case kLogRegionMem: + fprintf(stderr, "MEM "); + break; + default: + break; + } + + va_start(args, aFormat); + vfprintf(stderr, aFormat, args); + fprintf(stderr, "\r"); + va_end(args); +} diff --git a/pkg/openthread/contrib/platform_misc.c b/pkg/openthread/contrib/platform_misc.c new file mode 100644 index 0000000000000000000000000000000000000000..d8bc9af2b3c60c0fe2aa795ecc587f67cc90f544 --- /dev/null +++ b/pkg/openthread/contrib/platform_misc.c @@ -0,0 +1,36 @@ +/* + * Copyright (C) Baptiste Clenet + * + * 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 + * @file + * @brief Implementation of OpenThread misc platform abstraction + * + * @author Baptiste Clenet <bapclenet@gmail.com> + * @} + */ + + +#include "openthread/types.h" +#include "openthread/platform/misc.h" +#include "periph/pm.h" + +void otPlatReset(otInstance *aInstance) +{ + (void)aInstance; + printf("reboot...\n"); + pm_reboot(); +} + +otPlatResetReason otPlatGetResetReason(otInstance *aInstance) +{ + (void)aInstance; + /* TODO: Write me! */ + return kPlatResetReason_PowerOn; +} diff --git a/pkg/openthread/contrib/platform_radio.c b/pkg/openthread/contrib/platform_radio.c new file mode 100644 index 0000000000000000000000000000000000000000..6c4e8ba92319e61a3414002d120a748f029ac4af --- /dev/null +++ b/pkg/openthread/contrib/platform_radio.c @@ -0,0 +1,422 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread radio platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <assert.h> +#include <stdio.h> +#include <string.h> + +#include "byteorder.h" +#include "errno.h" +#include "net/ethernet/hdr.h" +#include "net/ethertype.h" +#include "net/ieee802154.h" +#include "net/netdev/ieee802154.h" +#include "openthread/platform/radio.h" +#include "ot.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +#define RADIO_IEEE802154_FCS_LEN (2U) + +static RadioPacket sTransmitFrame; +static RadioPacket sReceiveFrame; +static int8_t Rssi; + +static netdev_t *_dev; + +static bool sDisabled; + +/* set 15.4 channel */ +static int _set_channel(uint16_t channel) +{ + return _dev->driver->set(_dev, NETOPT_CHANNEL, &channel, sizeof(uint16_t)); +} + +/*get transmission power from driver */ +static int16_t _get_power(void) +{ + int16_t power; + + _dev->driver->get(_dev, NETOPT_TX_POWER, &power, sizeof(int16_t)); + return power; +} + +/* set transmission power */ +static int _set_power(int16_t power) +{ + return _dev->driver->set(_dev, NETOPT_TX_POWER, &power, sizeof(int16_t)); +} + +/* set IEEE802.15.4 PAN ID */ +static int _set_panid(uint16_t panid) +{ + return _dev->driver->set(_dev, NETOPT_NID, &panid, sizeof(uint16_t)); +} + +/* set extended HW address */ +static int _set_long_addr(uint8_t *ext_addr) +{ + return _dev->driver->set(_dev, NETOPT_ADDRESS_LONG, ext_addr, IEEE802154_LONG_ADDRESS_LEN); +} + +/* set short address */ +static int _set_addr(uint16_t addr) +{ + return _dev->driver->set(_dev, NETOPT_ADDRESS, &addr, sizeof(uint16_t)); +} + +/* check the state of promiscuous mode */ +static netopt_enable_t _is_promiscuous(void) +{ + netopt_enable_t en; + + _dev->driver->get(_dev, NETOPT_PROMISCUOUSMODE, &en, sizeof(en)); + return en == NETOPT_ENABLE ? true : false;; +} + +/* set the state of promiscuous mode */ +static int _set_promiscuous(netopt_enable_t enable) +{ + return _dev->driver->set(_dev, NETOPT_PROMISCUOUSMODE, &enable, sizeof(enable)); +} + +/* wrapper for setting device state */ +static void _set_state(netopt_state_t state) +{ + _dev->driver->set(_dev, NETOPT_STATE, &state, sizeof(netopt_state_t)); +} + +/* wrapper for getting device state */ +static netopt_state_t _get_state(void) +{ + netopt_state_t state; + _dev->driver->get(_dev, NETOPT_STATE, &state, sizeof(netopt_state_t)); + return state; +} + +/* sets device state to SLEEP */ +static void _set_sleep(void) +{ + _set_state(NETOPT_STATE_SLEEP); +} + +/* set device state to IDLE */ +static void _set_idle(void) +{ + _set_state(NETOPT_STATE_IDLE); +} + +/* init framebuffers and initial state */ +void openthread_radio_init(netdev_t *dev, uint8_t *tb, uint8_t *rb) +{ + sTransmitFrame.mPsdu = tb; + sTransmitFrame.mLength = 0; + sReceiveFrame.mPsdu = rb; + sReceiveFrame.mLength = 0; + _dev = dev; +} + +/* Called upon NETDEV_EVENT_RX_COMPLETE event */ +void recv_pkt(otInstance *aInstance, netdev_t *dev) +{ + DEBUG("Openthread: Received pkt\n"); + netdev_ieee802154_rx_info_t rx_info; + /* Read frame length from driver */ + int len = dev->driver->recv(dev, NULL, 0, &rx_info); + Rssi = rx_info.rssi; + + /* very unlikely */ + if ((len > (unsigned) UINT16_MAX)) { + DEBUG("Len too high: %d\n", len); + otPlatRadioReceiveDone(aInstance, NULL, kThreadError_Abort); + return; + } + + /* Fill OpenThread receive frame */ + /* Openthread needs a packet length with FCS included, + * OpenThread do not use the data so we don't need to calculate FCS */ + sReceiveFrame.mLength = len + RADIO_IEEE802154_FCS_LEN; + sReceiveFrame.mPower = _get_power(); + + /* Read received frame */ + int res = dev->driver->recv(dev, (char *) sReceiveFrame.mPsdu, len, NULL); + + DEBUG("Received message: len %d\n", (int) sReceiveFrame.mLength); + for (int i = 0; i < sReceiveFrame.mLength; ++i) { + DEBUG("%x ", sReceiveFrame.mPsdu[i]); + } + DEBUG("\n"); + + /* Tell OpenThread that receive has finished */ + otPlatRadioReceiveDone(aInstance, res > 0 ? &sReceiveFrame : NULL, res > 0 ? kThreadError_None : kThreadError_Abort); +} + +/* Called upon TX event */ +void send_pkt(otInstance *aInstance, netdev_t *dev, netdev_event_t event) +{ + /* Tell OpenThread transmission is done depending on the NETDEV event */ + switch (event) { + case NETDEV_EVENT_TX_COMPLETE: + DEBUG("openthread: NETDEV_EVENT_TX_COMPLETE\n"); + otPlatRadioTransmitDone(aInstance, &sTransmitFrame, false, kThreadError_None); + break; + case NETDEV_EVENT_TX_COMPLETE_DATA_PENDING: + DEBUG("openthread: NETDEV_EVENT_TX_COMPLETE_DATA_PENDING\n"); + otPlatRadioTransmitDone(aInstance, &sTransmitFrame, true, kThreadError_None); + break; + case NETDEV_EVENT_TX_NOACK: + DEBUG("openthread: NETDEV_EVENT_TX_NOACK\n"); + otPlatRadioTransmitDone(aInstance, &sTransmitFrame, false, kThreadError_NoAck); + break; + case NETDEV_EVENT_TX_MEDIUM_BUSY: + DEBUG("openthread: NETDEV_EVENT_TX_MEDIUM_BUSY\n"); + otPlatRadioTransmitDone(aInstance, &sTransmitFrame, false, kThreadError_ChannelAccessFailure); + break; + default: + break; + } +} + +/* OpenThread will call this for setting PAN ID */ +void otPlatRadioSetPanId(otInstance *aInstance, uint16_t panid) +{ + DEBUG("openthread: otPlatRadioSetPanId: setting PAN ID to %04x\n", panid); + _set_panid(panid); +} + +/* OpenThread will call this for setting extended address */ +void otPlatRadioSetExtendedAddress(otInstance *aInstance, uint8_t *aExtendedAddress) +{ + DEBUG("openthread: otPlatRadioSetExtendedAddress\n"); + uint8_t reversed_addr[IEEE802154_LONG_ADDRESS_LEN]; + for (int i = 0; i < IEEE802154_LONG_ADDRESS_LEN; i++) { + reversed_addr[i] = aExtendedAddress[IEEE802154_LONG_ADDRESS_LEN - 1 - i]; + } + _set_long_addr(reversed_addr); +} + +/* OpenThread will call this for setting short address */ +void otPlatRadioSetShortAddress(otInstance *aInstance, uint16_t aShortAddress) +{ + DEBUG("openthread: otPlatRadioSetShortAddress: setting address to %04x\n", aShortAddress); + _set_addr(((aShortAddress & 0xff) << 8) | ((aShortAddress >> 8) & 0xff)); +} + +/* OpenThread will call this for enabling the radio */ +ThreadError otPlatRadioEnable(otInstance *aInstance) +{ + DEBUG("openthread: otPlatRadioEnable\n"); + (void) aInstance; + + if (sDisabled) { + sDisabled = false; + _set_idle(); + } + + return kThreadError_None; +} + +/* OpenThread will call this for disabling the radio */ +ThreadError otPlatRadioDisable(otInstance *aInstance) +{ + DEBUG("openthread: otPlatRadioDisable\n"); + (void) aInstance; + + if (!sDisabled) { + sDisabled = true; + _set_sleep(); + } + + return kThreadError_None; +} + +bool otPlatRadioIsEnabled(otInstance *aInstance) +{ + DEBUG("otPlatRadioIsEnabled\n"); + (void) aInstance; + netopt_state_t state = _get_state(); + if (state == NETOPT_STATE_OFF || state == NETOPT_STATE_SLEEP) { + return false; + } else { + return true; + } +} + +/* OpenThread will call this for setting device state to SLEEP */ +ThreadError otPlatRadioSleep(otInstance *aInstance) +{ + DEBUG("otPlatRadioSleep\n"); + (void) aInstance; + + _set_sleep(); + return kThreadError_None; +} + +/*OpenThread will call this for waiting the reception of a packet */ +ThreadError otPlatRadioReceive(otInstance *aInstance, uint8_t aChannel) +{ + DEBUG("openthread: otPlatRadioReceive. Channel: %i\n", aChannel); + (void) aInstance; + + _set_idle(); + _set_channel(aChannel); + return kThreadError_None; +} + +/* OpenThread will call this function to get the transmit buffer */ +RadioPacket *otPlatRadioGetTransmitBuffer(otInstance *aInstance) +{ + DEBUG("openthread: otPlatRadioGetTransmitBuffer\n"); + return &sTransmitFrame; +} + +/* OpenThread will call this function to set the transmit power */ +void otPlatRadioSetDefaultTxPower(otInstance *aInstance, int8_t aPower) +{ + (void)aInstance; + + _set_power(aPower); +} + +/* OpenThread will call this for transmitting a packet*/ +ThreadError otPlatRadioTransmit(otInstance *aInstance, RadioPacket *aPacket) +{ + (void) aInstance; + struct iovec pkt; + + /* Populate iovec with transmit data + * Unlike RIOT, OpenThread includes two bytes FCS (0x00 0x00) so + * these bytes are removed + */ + pkt.iov_base = aPacket->mPsdu; + pkt.iov_len = aPacket->mLength - RADIO_IEEE802154_FCS_LEN; + + /*Set channel and power based on transmit frame */ + DEBUG("otPlatRadioTransmit->channel: %i, length %d\n", (int) aPacket->mChannel, (int)aPacket->mLength); + for (int i = 0; i < aPacket->mLength; ++i) { + DEBUG("%x ", aPacket->mPsdu[i]); + } + DEBUG("\n"); + _set_channel(aPacket->mChannel); + _set_power(aPacket->mPower); + + /* send packet though netdev */ + _dev->driver->send(_dev, &pkt, 1); + + return kThreadError_None; +} + +/* OpenThread will call this for getting the radio caps */ +otRadioCaps otPlatRadioGetCaps(otInstance *aInstance) +{ + DEBUG("openthread: otPlatRadioGetCaps\n"); + /* all drivers should handle ACK, including call of NETDEV_EVENT_TX_NOACK */ + return kRadioCapsNone; +} + +/* OpenThread will call this for getting the state of promiscuous mode */ +bool otPlatRadioGetPromiscuous(otInstance *aInstance) +{ + DEBUG("openthread: otPlatRadioGetPromiscuous\n"); + return _is_promiscuous(); +} + +/* OpenThread will call this for setting the state of promiscuous mode */ +void otPlatRadioSetPromiscuous(otInstance *aInstance, bool aEnable) +{ + DEBUG("openthread: otPlatRadioSetPromiscuous\n"); + _set_promiscuous((aEnable) ? NETOPT_ENABLE : NETOPT_DISABLE); +} + +int8_t otPlatRadioGetRssi(otInstance *aInstance) +{ + DEBUG("otPlatRadioGetRssi\n"); + (void) aInstance; + return Rssi; +} + +void otPlatRadioEnableSrcMatch(otInstance *aInstance, bool aEnable) +{ + DEBUG("otPlatRadioEnableSrcMatch\n"); + (void)aInstance; + (void)aEnable; +} + +ThreadError otPlatRadioAddSrcMatchShortEntry(otInstance *aInstance, const uint16_t aShortAddress) +{ + DEBUG("otPlatRadioAddSrcMatchShortEntry\n"); + (void)aInstance; + (void)aShortAddress; + return kThreadError_None; +} + +ThreadError otPlatRadioAddSrcMatchExtEntry(otInstance *aInstance, const uint8_t *aExtAddress) +{ + DEBUG("otPlatRadioAddSrcMatchExtEntry\n"); + (void)aInstance; + (void)aExtAddress; + return kThreadError_None; +} + +ThreadError otPlatRadioClearSrcMatchShortEntry(otInstance *aInstance, const uint16_t aShortAddress) +{ + DEBUG("otPlatRadioClearSrcMatchShortEntry\n"); + (void)aInstance; + (void)aShortAddress; + return kThreadError_None; +} + +ThreadError otPlatRadioClearSrcMatchExtEntry(otInstance *aInstance, const uint8_t *aExtAddress) +{ + DEBUG("otPlatRadioClearSrcMatchExtEntry\n"); + (void)aInstance; + (void)aExtAddress; + return kThreadError_None; +} + +void otPlatRadioClearSrcMatchShortEntries(otInstance *aInstance) +{ + DEBUG("otPlatRadioClearSrcMatchShortEntries\n"); + (void)aInstance; +} + +void otPlatRadioClearSrcMatchExtEntries(otInstance *aInstance) +{ + DEBUG("otPlatRadioClearSrcMatchExtEntries\n"); + (void)aInstance; +} + +ThreadError otPlatRadioEnergyScan(otInstance *aInstance, uint8_t aScanChannel, uint16_t aScanDuration) +{ + DEBUG("otPlatRadioEnergyScan\n"); + (void)aInstance; + (void)aScanChannel; + (void)aScanDuration; + return kThreadError_NotImplemented; +} + +void otPlatRadioGetIeeeEui64(otInstance *aInstance, uint8_t *aIeee64Eui64) +{ + _dev->driver->get(_dev, NETOPT_IPV6_IID, aIeee64Eui64, sizeof(eui64_t)); +} + +int8_t otPlatRadioGetReceiveSensitivity(otInstance *aInstance) +{ + return -100; +} diff --git a/pkg/openthread/contrib/platform_random.c b/pkg/openthread/contrib/platform_random.c new file mode 100644 index 0000000000000000000000000000000000000000..c82bb43ebc7451775caeb68d36ed5c6df7ad8fcc --- /dev/null +++ b/pkg/openthread/contrib/platform_random.c @@ -0,0 +1,51 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread random platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <stdint.h> + +#include "openthread/platform/random.h" +#include "periph/cpuid.h" +#include "random.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +/* init random */ +void ot_random_init(void) +{ +#ifdef CPUID_LEN + char cpu_id[CPUID_LEN]; + cpuid_get(cpu_id); + uint32_t seed = 0; + for (unsigned i = 0; i < CPUID_LEN; i++) { + seed += cpu_id[i]; + } + random_init(seed); +#else + #error "CPU not supported (current CPU doesn't provide CPUID, required for entropy)" +#endif +} + +/* OpenThread will call this to get a random number */ +uint32_t otPlatRandomGet(void) +{ + uint32_t rand_val = random_uint32(); + + DEBUG("otPlatRandomGet: %i\n", (int) rand_val); + return rand_val; +} diff --git a/pkg/openthread/contrib/platform_settings.c b/pkg/openthread/contrib/platform_settings.c new file mode 100644 index 0000000000000000000000000000000000000000..0e324c5406ecdddada1a07412fd29c0bf2f20318 --- /dev/null +++ b/pkg/openthread/contrib/platform_settings.c @@ -0,0 +1,72 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread settings platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include "assert.h" +#include "openthread/types.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +void otPlatSettingsInit(otInstance *aInstance) +{ +} + +ThreadError otPlatSettingsBeginChange(otInstance *aInstance) +{ + (void)aInstance; + return kThreadError_None; +} + +ThreadError otPlatSettingsCommitChange(otInstance *aInstance) +{ + DEBUG("openthread: otPlatSettingsCommitChange\n"); + (void)aInstance; + return kThreadError_None; +} + +ThreadError otPlatSettingsAbandonChange(otInstance *aInstance) +{ + (void)aInstance; + return kThreadError_None; +} + +ThreadError otPlatSettingsGet(otInstance *aInstance, uint16_t aKey, int aIndex, uint8_t *aValue, uint16_t *aValueLength) +{ + DEBUG("openthread: otPlatSettingsGet\n"); + *aValueLength = 0; + return kThreadError_NotImplemented; +} + +ThreadError otPlatSettingsSet(otInstance *aInstance, uint16_t aKey, const uint8_t *aValue, uint16_t aValueLength) +{ + return kThreadError_None; +} + +ThreadError otPlatSettingsAdd(otInstance *aInstance, uint16_t aKey, const uint8_t *aValue, uint16_t aValueLength) +{ + return kThreadError_None; +} + +ThreadError otPlatSettingsDelete(otInstance *aInstance, uint16_t aKey, int aIndex) +{ + return kThreadError_None; +} + +void otPlatSettingsWipe(otInstance *aInstance) +{ +} diff --git a/pkg/openthread/contrib/platform_uart.c b/pkg/openthread/contrib/platform_uart.c new file mode 100644 index 0000000000000000000000000000000000000000..d3aff67af3fc18216f255a0397bf004a3e230faf --- /dev/null +++ b/pkg/openthread/contrib/platform_uart.c @@ -0,0 +1,46 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 + * @file + * @brief Implementation of OpenThread UART platform abstraction + * + * @author Jose Ignacio Alamos <jialamos@uc.cl> + * @} + */ + +#include <stdint.h> +#include <stdio.h> + +#include "periph/uart.h" +#include "openthread/platform/uart.h" + +/* OpenThread will call this for enabling UART (required for OpenThread's CLI)*/ +ThreadError otPlatUartEnable(void) +{ + return kThreadError_None; +} + +/* OpenThread will call this for disabling UART */ +ThreadError otPlatUartDisable(void) +{ + return kThreadError_None; +} + +/* OpenThread will call this for sending data through UART */ +ThreadError otPlatUartSend(const uint8_t *aBuf, uint16_t aBufLength) +{ + uart_write(UART_DEV(0), aBuf, aBufLength); + + /* Tell OpenThread the sending of UART is done */ + otPlatUartSendDone(); + + return kThreadError_None; +} diff --git a/pkg/openthread/include/ot.h b/pkg/openthread/include/ot.h new file mode 100644 index 0000000000000000000000000000000000000000..e36649d732114934e416bbfac754faf1b77f8f28 --- /dev/null +++ b/pkg/openthread/include/ot.h @@ -0,0 +1,153 @@ +/* + * Copyright (C) 2017 Fundacion Inria Chile + * + * 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 pkg_openthread_cli OpenThread + * @ingroup pkg_openthread + * @brief An open source implementation of Thread stack + * @see https://github.com/openthread/openthread + * + * Thread if a mesh oriented network stack running for IEEE802.15.4 networks. + * @{ + * + * @file + * + * @author José Ignacio Alamos <jialamos@uc.cl> + */ + +#ifndef OT_H +#define OT_H + +#ifdef __cplusplus +extern "C" { +#endif + +#include "net/netopt.h" +#include "net/ieee802154.h" +#include "net/ethernet.h" +#include "net/gnrc/netdev.h" +#include "thread.h" +#include "openthread/types.h" + +#define OPENTHREAD_XTIMER_MSG_TYPE_EVENT (0x2235) /**< xtimer message receiver event*/ +#define OPENTHREAD_NETDEV_MSG_TYPE_EVENT (0x2236) /**< message received from driver */ +#define OPENTHREAD_SERIAL_MSG_TYPE_EVENT (0x2237) /**< event indicating a serial (UART) message was sent to OpenThread */ +#define OPENTHREAD_MSG_TYPE_RECV (0x2238) /**< event for frame reception */ +#define OPENTHREAD_JOB_MSG_TYPE_EVENT (0x2240) /**< event indicating an OT_JOB message */ + +/** + * @brief Struct containing a serial message + */ +typedef struct { + void *buf; /**< buffer containing the message */ + size_t len; /**< length of the message */ +} serial_msg_t; + +/** + * @brief Struct containing an OpenThread job + */ +typedef struct { + const char *command; /**< A pointer to the job name string. */ + void *arg; /**< arg for the job **/ + void *answer; /**< answer from the job **/ +} ot_job_t; + +/** + * @brief Gets packet from driver and tells OpenThread about the reception. + * + * @param[in] aInstance pointer to an OpenThread instance + */ +void recv_pkt(otInstance *aInstance, netdev_t *dev); + +/** + * @brief Inform OpenThread when tx is finished + * + * @param[in] aInstance pointer to an OpenThread instance + * @param[in] dev pointer to a netdev interface + * @param[in] event just occurred netdev event + */ +void send_pkt(otInstance *aInstance, netdev_t *dev, netdev_event_t event); + +/** + * @brief Bootstrap OpenThread + */ +void openthread_bootstrap(void); + +/** + * @brief Init OpenThread radio + * + * @param[in] dev pointer to a netdev interface + * @param[in] tb pointer to the TX buffer designed for OpenThread + * @param[in] event pointer to the RX buffer designed for Open_Thread + */ +void openthread_radio_init(netdev_t *dev, uint8_t *tb, uint8_t *rb); + + +/** + * @brief Starts OpenThread thread. + * + * @param[in] stack pointer to the stack designed for OpenThread + * @param[in] stacksize size of the stack + * @param[in] priority priority of the OpenThread stack + * @param[in] name name of the OpenThread stack + * @param[in] netdev pointer to the netdev interface + * + * @return PID of OpenThread thread + * @return -EINVAL if there was an error creating the thread + */ +int openthread_netdev_init(char *stack, int stacksize, char priority, const char *name, netdev_t *netdev); + +/** + * @brief get PID of OpenThread thread. + * + * @return PID of OpenThread thread + */ +kernel_pid_t openthread_get_pid(void); + +/** + * @brief Init OpenThread random + */ +void ot_random_init(void); + +/* + * @brief Run OpenThread UART simulator (stdio) + */ +void openthread_uart_run(void); + +/** + * @brief Execute OpenThread command. Call this function only in OpenThread thread + * + * @param[in] ot_instance OpenThread instance + * @param[in] command OpenThread command name + * @param[in] arg arg for the command + * @param[out] answer answer for the command + * + * @return 0 on success, 1 on error + */ +uint8_t ot_exec_command(otInstance *ot_instance, const char* command, void *arg, void* answer); + +/** + * @brief Call OpenThread command in same thread as OT core (due to concurrency). + * + * @note An OpenThread command allows direct calls to OpenThread API (otXXX functions) without worrying about concurrency + * issues. All API calls should be made in OT_JOB type functions. + * + * @param[in] command name of the command to call + * @param[in] arg arg for the command + * @param[out] answer answer for the command + * + * @return 0 on success, 1 on error + */ +uint8_t ot_call_command(char* command, void *arg, void* answer); + +#ifdef __cplusplus +} +#endif + +#endif /* OT_H */ +/** @} */ diff --git a/sys/auto_init/auto_init.c b/sys/auto_init/auto_init.c index 30fc3be9de854c03990cef6bbf22958b977cac19..0a91baa15789379fdca0e6c02a814882ab5efcd0 100644 --- a/sys/auto_init/auto_init.c +++ b/sys/auto_init/auto_init.c @@ -72,6 +72,10 @@ #include "lwip.h" #endif +#ifdef MODULE_OPENTHREAD +#include "ot.h" +#endif + #ifdef MODULE_FIB #include "net/fib.h" #endif @@ -140,6 +144,10 @@ void auto_init(void) DEBUG("Bootstraping lwIP.\n"); lwip_bootstrap(); #endif +#ifdef MODULE_OPENTHREAD + extern void openthread_bootstrap(void); + openthread_bootstrap(); +#endif #ifdef MODULE_GCOAP DEBUG("Auto init gcoap module.\n"); gcoap_init();