From 66dd84eff2847657d8009f4277817bd3c3bed21a Mon Sep 17 00:00:00 2001 From: Yonezawa-T2 Date: Thu, 26 Nov 2015 16:27:28 +0900 Subject: [PATCH] xbee: disable short address when the address length is set to 8 XBee sends short address even for `API_ID_TX_LONG_ADDR` if short address is enabled. This results in check sum error of ICMPv6 since the IP address is computed based on long address on the sender side while it is computed based on short address on the receiver side. --- drivers/xbee/xbee.c | 40 ++++++++++++++++++++++++++++++---------- 1 file changed, 30 insertions(+), 10 deletions(-) diff --git a/drivers/xbee/xbee.c b/drivers/xbee/xbee.c index ff9934f9d9..4d4741d712 100644 --- a/drivers/xbee/xbee.c +++ b/drivers/xbee/xbee.c @@ -252,31 +252,41 @@ static int _get_addr_long(xbee_t *dev, uint8_t *val, size_t len) return -ECANCELED; } -static int _set_addr(xbee_t *dev, uint8_t *val, size_t len) +static int _set_short_addr(xbee_t *dev, uint8_t *address) { uint8_t cmd[4]; resp_t resp; + cmd[0] = 'M'; + cmd[1] = 'Y'; + cmd[2] = address[0]; + cmd[3] = address[1]; + _api_at_cmd(dev, cmd, 4, &resp); + + return resp.status; +} + +static int _set_addr(xbee_t *dev, uint8_t *val, size_t len) +{ /* device only supports setting the short address */ if (len != 2) { return -ENOTSUP; } - cmd[0] = 'M'; - cmd[1] = 'Y'; - cmd[2] = val[0]; - cmd[3] = val[1]; + + if (dev->addr_flags & XBEE_ADDR_FLAGS_LONG || + _set_short_addr(dev, val) == 0) { #ifdef MODULE_SIXLOWPAN - /* https://tools.ietf.org/html/rfc4944#section-12 requires the first bit to - * 0 for unicast addresses */ - val[1] &= 0x7F; + /* https://tools.ietf.org/html/rfc4944#section-12 requires the first bit + * to 0 for unicast addresses */ + val[1] &= 0x7F; #endif - _api_at_cmd(dev, cmd, 4, &resp); - if (resp.status == 0) { memcpy(dev->addr_short, val, 2); + return 2; } + return -ECANCELED; } @@ -289,9 +299,19 @@ static int _set_addr_len(xbee_t *dev, uint16_t *val, size_t len) switch (*val) { case IEEE802154_LONG_ADDRESS_LEN: dev->addr_flags |= XBEE_ADDR_FLAGS_LONG; + + /* disable short address */ + uint8_t disabled_addr[] = { 0xFF, 0xFF }; + + _set_short_addr(dev, disabled_addr); + break; case IEEE802154_SHORT_ADDRESS_LEN: dev->addr_flags &= ~XBEE_ADDR_FLAGS_LONG; + + /* restore short address */ + _set_short_addr(dev, dev->addr_short); + break; default: return -EINVAL;