sdrangel/plugins/channeltx/mod802.15.4/ieee_802_15_4_macframe.h

119 wiersze
5.6 KiB
C

///////////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2020 Jon Beniston, M7RCE //
// //
// This program is free software; you can redistribute it and/or modify //
// it under the terms of the GNU General Public License as published by //
// the Free Software Foundation as version 3 of the License, or //
// (at your option) any later version. //
// //
// This program is distributed in the hope that it will be useful, //
// but WITHOUT ANY WARRANTY; without even the implied warranty of //
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
// GNU General Public License V3 for more details. //
// //
// You should have received a copy of the GNU General Public License //
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
///////////////////////////////////////////////////////////////////////////////////
#ifndef INCLUDE_IEEE_802_15_4_MACFRAME_H
#define INCLUDE_IEEE_802_15_4_MACFRAME_H
#include <stdint.h>
#include <stdio.h>
// Frame control field values
#define IEEE_802_15_4_MAC_FRAME_TYPE_MASK 0x0003
#define IEEE_802_15_4_MAC_BEACON 0x0000
#define IEEE_802_15_4_MAC_DATA 0x0001
#define IEEE_802_15_4_MAC_ACK 0x0002
#define IEEE_802_15_4_MAC_COMMAND 0x0003
#define IEEE_802_15_4_MAC_SECURITY_ENABLED 0x0008
#define IEEE_802_15_4_MAC_FRAME_PENDING 0x0010
#define IEEE_802_15_4_MAC_ACK_REQUIRED 0x0020
#define IEEE_802_15_4_MAC_PAN_COMPRESSION 0x0040
#define IEEE_802_15_4_MAC_NO_SEQ_NUMBER 0x0100
#define IEEE_802_15_4_MAC_DEST_ADDRESS_MASK 0x0c00
#define IEEE_802_15_4_MAC_DEST_ADDRESS_NONE 0x0000
#define IEEE_802_15_4_MAC_DEST_ADDRESS_SHORT 0x0400
#define IEEE_802_15_4_MAC_DEST_ADDRESS_EXT 0x0c00
#define IEEE_802_15_4_MAC_SOURCE_ADDRESS_MASK 0xc000
#define IEEE_802_15_4_MAC_SOURCE_ADDRESS_NONE 0x0000
#define IEEE_802_15_4_MAC_SOURCE_ADDRESS_SHORT 0x4000
#define IEEE_802_15_4_MAC_SOURCE_ADDRESS_EXT 0xc000
#define IEEE_802_15_4_MAC_PAYLOAD_MAX_LENGTH 124
#define IEEE_802_15_5_MAX_EXT_ADDRESS_LENGTH 8
typedef uint8_t ieee_802_15_4_address[IEEE_802_15_5_MAX_EXT_ADDRESS_LENGTH];
struct IEEE_802_15_4_MacFrame
{
uint16_t m_frameControl;
uint8_t m_sequenceNumber;
uint16_t m_destPANID;
uint16_t m_destShortAddress;
ieee_802_15_4_address m_destAddress;
uint16_t m_sourcePANID;
uint16_t m_sourceShortAddress;
ieee_802_15_4_address m_sourceAddress;
uint8_t m_payload[IEEE_802_15_4_MAC_PAYLOAD_MAX_LENGTH];
uint8_t m_payloadLength;
IEEE_802_15_4_MacFrame()
{
if (false)
{
// Example ACK frame
m_frameControl = IEEE_802_15_4_MAC_ACK;
m_sequenceNumber = 0;
m_payloadLength = 0;
}
else
{
ieee_802_15_4_address dst = {0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77};
ieee_802_15_4_address src = {0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff};
// Example data frame
m_frameControl = IEEE_802_15_4_MAC_DATA | IEEE_802_15_4_MAC_DEST_ADDRESS_EXT | IEEE_802_15_4_MAC_SOURCE_ADDRESS_EXT;
m_sequenceNumber = 0;
m_destPANID = 0xbabe;
memcpy(m_destAddress, dst, sizeof(m_destAddress));
m_sourcePANID = 0xbabe;
memcpy(m_sourceAddress, src, sizeof(m_sourceAddress));
strcpy((char *)m_payload, "SDR Angel does 15.4");
m_payloadLength = strlen((char *)m_payload);
}
}
char *bytesToHex(char *buf, uint8_t *data, int len)
{
for (int i = 0; i < len; i++)
buf += sprintf(buf, "%02x ", data[i]);
return buf;
}
void toHexCharArray(char *buf)
{
buf += sprintf(buf, "%02x %02x %02x ", m_frameControl & 0xff, (m_frameControl >> 8) & 0xff, m_sequenceNumber);
if ((m_frameControl & IEEE_802_15_4_MAC_FRAME_TYPE_MASK) != IEEE_802_15_4_MAC_ACK)
buf += sprintf(buf, "%02x %02x ", m_destPANID & 0xff, (m_destPANID >> 8) & 0xff);
if ((m_frameControl & IEEE_802_15_4_MAC_DEST_ADDRESS_MASK) == IEEE_802_15_4_MAC_DEST_ADDRESS_EXT)
buf = bytesToHex(buf, m_destAddress, sizeof(m_destAddress));
else if ((m_frameControl & IEEE_802_15_4_MAC_DEST_ADDRESS_MASK) == IEEE_802_15_4_MAC_DEST_ADDRESS_SHORT)
buf += sprintf(buf, "%02x %02x ", m_destShortAddress & 0xff, (m_destShortAddress >> 8) & 0xff);
if (((m_frameControl & IEEE_802_15_4_MAC_FRAME_TYPE_MASK) != IEEE_802_15_4_MAC_ACK)
&& (!(m_frameControl & IEEE_802_15_4_MAC_PAN_COMPRESSION) || (m_destPANID != m_sourcePANID)))
buf += sprintf(buf, "%02x %02x ", m_sourcePANID & 0xff, (m_sourcePANID >> 8) & 0xff);
if ((m_frameControl & IEEE_802_15_4_MAC_SOURCE_ADDRESS_MASK) == IEEE_802_15_4_MAC_SOURCE_ADDRESS_EXT)
buf = bytesToHex(buf, m_sourceAddress, sizeof(m_sourceAddress));
else if ((m_frameControl & IEEE_802_15_4_MAC_SOURCE_ADDRESS_MASK) == IEEE_802_15_4_MAC_SOURCE_ADDRESS_SHORT)
buf += sprintf(buf, "%02x %02x ", m_sourceShortAddress & 0xff, (m_sourceShortAddress >> 8) & 0xff);
buf = bytesToHex(buf, m_payload, m_payloadLength);
}
};
#endif /* INCLUDE_IEEE_802_15_4_MACFRAME_H */