2016-01-14 09:19:56 +01:00
|
|
|
/*
|
|
|
|
* 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 <errno.h>
|
2016-09-30 13:06:20 +02:00
|
|
|
#include <string.h>
|
2016-01-14 09:19:56 +01:00
|
|
|
|
|
|
|
#include "net/ieee802154.h"
|
|
|
|
|
2016-09-30 13:06:20 +02:00
|
|
|
const uint8_t ieee802154_addr_bcast[IEEE802154_ADDR_BCAST_LEN] = IEEE802154_ADDR_BCAST;
|
|
|
|
|
2016-01-14 09:19:56 +01:00
|
|
|
size_t ieee802154_set_frame_hdr(uint8_t *buf, const uint8_t *src, size_t src_len,
|
|
|
|
const uint8_t *dst, size_t dst_len,
|
|
|
|
le_uint16_t src_pan, le_uint16_t dst_pan,
|
|
|
|
uint8_t flags, uint8_t seq)
|
|
|
|
{
|
|
|
|
int pos = 3; /* 0-1: FCS, 2: seq */
|
|
|
|
uint8_t type = (flags & IEEE802154_FCF_TYPE_MASK);
|
|
|
|
|
2016-09-30 13:06:20 +02:00
|
|
|
buf[0] = flags;
|
2016-01-14 09:19:56 +01:00
|
|
|
buf[1] = IEEE802154_FCF_VERS_V1;
|
|
|
|
|
|
|
|
if (((src_len != 0) && (src == NULL)) ||
|
2016-09-30 13:06:20 +02:00
|
|
|
((dst_len != 0) && (dst == NULL))) {
|
2016-01-14 09:19:56 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Frame type is not beacon or ACK, but both address modes are zero */
|
|
|
|
if ((type != IEEE802154_FCF_TYPE_BEACON) && (type != IEEE802154_FCF_TYPE_ACK) &&
|
|
|
|
(src_len == 0) && (dst_len == 0)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set sequence number */
|
|
|
|
buf[2] = seq;
|
|
|
|
|
2016-09-30 13:06:20 +02:00
|
|
|
if (dst_len != 0) {
|
2016-01-14 09:19:56 +01:00
|
|
|
buf[pos++] = dst_pan.u8[0];
|
|
|
|
buf[pos++] = dst_pan.u8[1];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in destination address */
|
2016-09-30 13:06:20 +02:00
|
|
|
switch (dst_len) {
|
|
|
|
case 0:
|
|
|
|
buf[1] |= IEEE802154_FCF_DST_ADDR_VOID;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
if (memcmp(dst, ieee802154_addr_bcast,
|
|
|
|
sizeof(ieee802154_addr_bcast)) == 0) {
|
|
|
|
/* do not request ACKs for broadcast address */
|
|
|
|
buf[0] &= ~IEEE802154_FCF_ACK_REQ;
|
|
|
|
}
|
|
|
|
buf[1] |= IEEE802154_FCF_DST_ADDR_SHORT;
|
|
|
|
buf[pos++] = dst[1];
|
|
|
|
buf[pos++] = dst[0];
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
buf[1] |= IEEE802154_FCF_DST_ADDR_LONG;
|
|
|
|
for (int i = 7; i >= 0; i--) {
|
|
|
|
buf[pos++] = dst[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
2016-01-14 09:19:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in source PAN ID (if applicable) */
|
2016-09-30 13:04:14 +02:00
|
|
|
if (src_len != 0) {
|
|
|
|
if ((dst_len != 0) && (src_pan.u16 == dst_pan.u16)) {
|
|
|
|
buf[0] |= IEEE802154_FCF_PAN_COMP;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* (little endian) */
|
|
|
|
buf[pos++] = src_pan.u8[0];
|
|
|
|
buf[pos++] = src_pan.u8[1];
|
|
|
|
}
|
2016-01-14 09:19:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* fill in source address */
|
|
|
|
switch (src_len) {
|
|
|
|
case 0:
|
|
|
|
buf[1] |= IEEE802154_FCF_SRC_ADDR_VOID;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
buf[1] |= IEEE802154_FCF_SRC_ADDR_SHORT;
|
|
|
|
buf[pos++] = src[1];
|
|
|
|
buf[pos++] = src[0];
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
buf[1] |= IEEE802154_FCF_SRC_ADDR_LONG;
|
|
|
|
for (int i = 7; i >= 0; i--) {
|
|
|
|
buf[pos++] = src[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* return actual header length */
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t ieee802154_get_frame_hdr_len(const uint8_t *mhr)
|
|
|
|
{
|
|
|
|
/* TODO: include security header implications */
|
2022-10-07 11:38:06 +02:00
|
|
|
uint8_t tmp, has_dst = 0;
|
2016-01-14 09:19:56 +01:00
|
|
|
size_t len = 3; /* 2 byte FCF, 1 byte sequence number */
|
|
|
|
|
2022-10-07 11:38:06 +02:00
|
|
|
tmp = (mhr[0] & IEEE802154_FCF_TYPE_MASK);
|
|
|
|
if (tmp == IEEE802154_FCF_TYPE_ACK) {
|
|
|
|
/* ACK contains no other fields */
|
|
|
|
return len;
|
|
|
|
} else if (tmp != IEEE802154_FCF_TYPE_BEACON) {
|
|
|
|
/* Beacon contains no dst address */
|
|
|
|
tmp = (mhr[1] & IEEE802154_FCF_DST_ADDR_MASK);
|
|
|
|
if (tmp == IEEE802154_FCF_DST_ADDR_SHORT) {
|
|
|
|
len += 4; /* 2 byte dst PAN + 2 byte dst short address */
|
|
|
|
has_dst = 1;
|
|
|
|
}
|
|
|
|
else if (tmp == IEEE802154_FCF_DST_ADDR_LONG) {
|
|
|
|
len += 10; /* 2 byte dst PAN + 8 byte dst long address */
|
|
|
|
has_dst = 1;
|
|
|
|
}
|
|
|
|
else if (tmp != IEEE802154_FCF_DST_ADDR_VOID) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else if (mhr[0] & IEEE802154_FCF_PAN_COMP) {
|
|
|
|
/* PAN compression, but no destination address => illegal state */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else if (mhr[0] & IEEE802154_FCF_PAN_COMP) {
|
|
|
|
/* Beacon can't use PAN compression */
|
2016-01-14 09:19:56 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
tmp = (mhr[1] & IEEE802154_FCF_SRC_ADDR_MASK);
|
|
|
|
if (tmp == IEEE802154_FCF_SRC_ADDR_VOID) {
|
2022-10-07 11:38:06 +02:00
|
|
|
/* One of dst or src address must be present */
|
|
|
|
return has_dst ? len : 0;
|
2016-01-14 09:19:56 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (!(mhr[0] & IEEE802154_FCF_PAN_COMP)) {
|
|
|
|
len += 2;
|
|
|
|
}
|
|
|
|
if (tmp == IEEE802154_FCF_SRC_ADDR_SHORT) {
|
|
|
|
return len + 2;
|
|
|
|
}
|
|
|
|
else if (tmp == IEEE802154_FCF_SRC_ADDR_LONG) {
|
|
|
|
return len + 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ieee802154_get_src(const uint8_t *mhr, uint8_t *src, le_uint16_t *src_pan)
|
|
|
|
{
|
|
|
|
int offset = 3; /* FCF: 0-1, Seq: 2 */
|
|
|
|
uint8_t tmp;
|
|
|
|
|
|
|
|
assert((src != NULL) && (src_pan != NULL));
|
|
|
|
tmp = mhr[1] & IEEE802154_FCF_DST_ADDR_MASK;
|
|
|
|
if (tmp == IEEE802154_FCF_DST_ADDR_SHORT) {
|
|
|
|
if (mhr[0] & IEEE802154_FCF_PAN_COMP) {
|
|
|
|
src_pan->u8[0] = mhr[offset];
|
|
|
|
src_pan->u8[1] = mhr[offset + 1];
|
|
|
|
}
|
|
|
|
offset += 4;
|
|
|
|
}
|
|
|
|
else if (tmp == IEEE802154_FCF_DST_ADDR_LONG) {
|
|
|
|
if (mhr[0] & IEEE802154_FCF_PAN_COMP) {
|
|
|
|
src_pan->u8[0] = mhr[offset];
|
|
|
|
src_pan->u8[1] = mhr[offset + 1];
|
|
|
|
}
|
|
|
|
offset += 10;
|
|
|
|
}
|
|
|
|
else if (tmp != IEEE802154_FCF_DST_ADDR_VOID) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
else if (mhr[0] & IEEE802154_FCF_PAN_COMP) {
|
|
|
|
/* PAN compression, but no destination address => illegal state */
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = mhr[1] & IEEE802154_FCF_SRC_ADDR_MASK;
|
|
|
|
if (tmp != IEEE802154_FCF_SRC_ADDR_VOID) {
|
2021-12-03 11:25:48 +01:00
|
|
|
if (!(mhr[0] & IEEE802154_FCF_PAN_COMP) &&
|
|
|
|
(tmp != IEEE802154_FCF_SRC_ADDR_RESV)) {
|
2016-01-14 09:19:56 +01:00
|
|
|
src_pan->u8[0] = mhr[offset++];
|
|
|
|
src_pan->u8[1] = mhr[offset++];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (tmp == IEEE802154_FCF_SRC_ADDR_SHORT) {
|
|
|
|
/* read src PAN and address in little endian */
|
|
|
|
src[1] = mhr[offset++];
|
|
|
|
src[0] = mhr[offset++];
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
else if (tmp == IEEE802154_FCF_SRC_ADDR_LONG) {
|
|
|
|
/* read src PAN and address in little endian */
|
|
|
|
for (int i = 7; i >= 0; i--) {
|
|
|
|
src[i] = mhr[offset++];
|
|
|
|
}
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
else if (tmp != IEEE802154_FCF_SRC_ADDR_VOID) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ieee802154_get_dst(const uint8_t *mhr, uint8_t *dst, le_uint16_t *dst_pan)
|
|
|
|
{
|
|
|
|
int offset = 3; /* FCF: 0-1, Seq: 2 */
|
|
|
|
uint8_t tmp;
|
|
|
|
|
|
|
|
assert((dst != NULL) && (dst_pan != NULL));
|
|
|
|
tmp = mhr[1] & IEEE802154_FCF_DST_ADDR_MASK;
|
|
|
|
if (tmp == IEEE802154_FCF_DST_ADDR_SHORT) {
|
|
|
|
/* read dst PAN and address in little endian */
|
|
|
|
dst_pan->u8[0] = mhr[offset++];
|
|
|
|
dst_pan->u8[1] = mhr[offset++];
|
|
|
|
dst[1] = mhr[offset++];
|
|
|
|
dst[0] = mhr[offset++];
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
else if (tmp == IEEE802154_FCF_DST_ADDR_LONG) {
|
|
|
|
dst_pan->u8[0] = mhr[offset++];
|
|
|
|
dst_pan->u8[1] = mhr[offset++];
|
|
|
|
for (int i = 7; i >= 0; i--) {
|
|
|
|
dst[i] = mhr[offset++];
|
|
|
|
}
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
else if (tmp != IEEE802154_FCF_DST_ADDR_VOID) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
else if (mhr[0] & IEEE802154_FCF_PAN_COMP) {
|
|
|
|
/* PAN compression, but no destination address => illegal state */
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-06-21 13:47:22 +02:00
|
|
|
int ieee802154_dst_filter(const uint8_t *mhr, uint16_t pan,
|
|
|
|
network_uint16_t short_addr, const eui64_t *ext_addr)
|
|
|
|
{
|
|
|
|
uint8_t dst_addr[IEEE802154_LONG_ADDRESS_LEN];
|
|
|
|
le_uint16_t dst_pan;
|
|
|
|
uint8_t pan_bcast[] = IEEE802154_PANID_BCAST;
|
|
|
|
|
|
|
|
int addr_len = ieee802154_get_dst(mhr, dst_addr, &dst_pan);
|
|
|
|
|
|
|
|
/* filter PAN ID */
|
|
|
|
if ((memcmp(pan_bcast, dst_pan.u8, 2) != 0) &&
|
|
|
|
(memcmp(&pan, dst_pan.u8, 2) != 0)) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check destination address */
|
|
|
|
if (((addr_len == IEEE802154_SHORT_ADDRESS_LEN) &&
|
|
|
|
(memcmp(&short_addr.u8, dst_addr, addr_len) == 0 ||
|
|
|
|
memcmp(ieee802154_addr_bcast, dst_addr, addr_len) == 0)) ||
|
|
|
|
((addr_len == IEEE802154_LONG_ADDRESS_LEN) &&
|
|
|
|
(memcmp(ext_addr->uint8, dst_addr, addr_len) == 0))) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-01-14 09:19:56 +01:00
|
|
|
/** @} */
|