Porting IEEE 802.15.4 MAC driver
Nanostack has a lower-level API for the IEEE 802.15.4-2006 MAC standard. This enables developers to support different MACs for software- and hardware-based solutions. Nanostack offers SW MAC, which you can use when your board does not have 15.4 MAC available. SW MAC also supports a subset of IEEE 802.15.4-2015 MAC standard. Supported features are:
- IEEE 802.15.4-2015 enhanced ack.
- Header and payload information elements set by MAC user.
SW MAC
Nanostack includes an IEEE 802.15.4-based SW MAC class. You can use SW MAC when your board does not support MAC. To use the SW MAC service, you must have a working RF driver registered to Nanostack. To create SW MAC, call the following function:
ns_sw_mac_create()
This creates an SW MAC class and sets a callback function to be used by Nanostack.
Note: You must not call ns_sw_mac_create() more than once.
SW MAC IEEE 802.15.4-2015 subset extension
SW MAC supports a subset of the IEEE 802.15.4-2015 standard. The subset includes the following supported features:
- Data frame send with payload and header information elements defined by MAC user.
- Enhanced ACK generation and send by MAC.
- Enhanced ACK payload and Information elements write for MAC user.
The subset features need a proper supported RF driver with extended driver API commands.
Initializing SW MAC
Deploy SW MAC:
- 
Call arm_net_phy_register()to register the configured RF driver class to Nanostack.
- 
Call ns_sw_mac_create()to create SW MAC with needed list sizes:- A sleepy device needs only 1-4 as the size of the device_decription_table_size.
- The recommended value for key_lookup_sizeis 1 and forkey_usage_size3.
 
- A sleepy device needs only 1-4 as the size of the 
- 
Call arm_nwk_interface_lowpan_init()to create Nanostack with the created SW MAC class. Nanostack initializes SW MAC before using it.
Example
This code snippet creates SW MAC with a neighbor table of size 16 and three key descriptions:
int8_t generate_6lowpan_interface(int8_t rf_phy_device_register_id)
{
    mac_description_storage_size_t storage_sizes;
    storage_sizes.device_decription_table_size = 16;
    storage_sizes.key_description_table_size = 3;
    storage_sizes.key_lookup_size = 1;
    storage_sizes.key_usage_size = 3;
    mac_api_t *mac_api = ns_sw_mac_create(rf_phy_device_register_id, &storage_sizes);
    if (!mac_api) {
        tr_error("Mac create fail!");
        return -1;
    }
    return arm_nwk_interface_lowpan_init(mac_api, "6LoWPAN_ROUTER");
}
Enabling FHSS
SW MAC supports FHSS. To enable it, you need to do the following:
- Call arm_net_phy_register()to register the configured RF driver class to Nanostack.
- Call ns_sw_mac_create()to create SW MAC with needed list sizes.
- Call ns_fhss_create()to configure and define the FHSS class.
- Call ns_sw_mac_fhss_register()to register FHSS to SW MAC.
- Call arm_nwk_interface_lowpan_init()to create Nanostack with the created SW MAC class.
IEEE 802.15.4 MAC sublayer APIs
The stack uses the IEEE 802.15.4 defined MAC management service entity (MLME-SAP) and MAC data service (MCPS-SAP) interfaces. MAC API follows MCPS and MLME primitives defined by the IEEE 802.15.4-2006 standard.
The following primitives are used in MAC layer:
| Primitive | Description | 
|---|---|
| Request | Request made by service user. | 
| Confirm | MAC layer response to earlier request. | 
| Indication | Indication event from MAC to service user. | 
| Response | Service user's response to received indication. | 
The following header files define the MAC API:
- mac_api.h- Main header, which defines a transparent MAC API for Nanostack to use.
- mlme.h- Definitions for MLME-SAP primitives.
- mac_mcps.h- Definitions for MCPS-SAP primitives.
- mac_common_defines.h- Definitions for common MAC constants.
MCPS-SAP interface
MCPS-SAP defines 802.15.4 data flow API with the following primitives:
| Primitive | Description | 
|---|---|
| MCPS-DATA-REQ | Data request primitive to MAC. | 
| MCPS-DATA-CONF | MAC generated confirmation for ongoing MCPS-DATA-REQ. | 
| MCPS-DATA-IND | MAC generated data indication event. | 
| MCPS-PURGE-REQ | Cancel ongoing MCPS-DATA-REQfrom MAC. | 
| MCPS-PURGE-CONF | Confirmation from MAC to MCPS-PURGE-REQoperation. | 
MLME-SAP interface
MLME-SAP defines a set of different management primitives, and this document introduces both supported and unsupported primitives in Nanostack.
Supported MLME APIs
MLME-SAP primitives used by Nanostack:
| Primitive | Description | 
|---|---|
| MLME-BEACON-NOTIFY | MAC generated event for received beacons. | 
| MLME-GET-REQ | Request information about a specified PAN Information Base (PIB) attribute. | 
| MLME-GET-CONF | MAC generated response to MLME-GET-REQ. | 
| MLME-RESET-REQ | Request to reset MAC to idle state and clean data queues. | 
| MLME-SCAN-REQ | Start MAC scan process. Orphan scan is not supported. | 
| MLME-SCAN-CONF | Result of the scan made by MLME-SCAN-REQ. | 
| MLME-COMM-STATUS-IND | MAC generated indication about the communications status. | 
| MLME-SET-REQ | Request to write data into a specified PIB attribute. | 
| MLME-SET-CONF | MAC generated response to MLME-SET-REQ. | 
| MLME-START-REQ | Starts or enables MAC with specified options. Nanostack uses this also for RFD devices. | 
| MLME-SYNCH-LOSS-IND | Indicates synchronization loss from wireless PAN. Only used by SW MAC when FHSS is in use! | 
| MLME-POLL-REQ | Request MAC to do data poll to parent. | 
Unsupported MLME APIs
Unsupported MLME-SAP primitives:
| Primitive | Support planned | Description | 
|---|---|---|
| MLME-ASSOCIATE-REQ | Not yet | Start MAC association process. | 
| MLME-ASSOCIATE-CONF | Not yet | MAC association process confirmation status. | 
| MLME-ASSOCIATE-IND | Not yet | MAC association indication to indicate the reception of assocation request. | 
| MLME-ASSOCIATE-RES | Not yet | MAC association response for indication. | 
| MLME-DISASSOCIATE-REQ | Not yet | MAC disassociation request from service user. | 
| MLME-DISASSOCIATE-IND | Not yet | MAC disassociation indication event to service user. | 
| MLME-DISASSOCIATE-CONF | Not yet | MAC disassociation confirmation when the disassociation request is handled. | 
| MLME-GTS-REQ | Not yet | MAC Guaranteed Time Slot (GTS) request. | 
| MLME-GTS-IND | Not yet | MAC GTS allocate event indication. | 
| MLME-GTS-CONF | Not yet | MAC GTS request confirmation. | 
| MLME-ORPHAN-IND | Not yet | Service user indicated by orphaned device. | 
| MLME-ORPHAN-RES | Not yet | Service user response to orphan indication event. | 
| MLME-RESET-CONF | Yes | MAC reset confirmation. | 
| MLME-RX-ENABLE-REQ | Yes | Enable (or disable) RX receiver for a specified time. | 
| MLME-RX-ENABLE-CONF | Yes | Confirmation for MLME-RX-ENABLE-REQ. | 
| MLME-START-CONF | Yes | Confirmation for MLME start request. | 
| MLME-SYNCH-REQ | Not yet | Request MAC to synchronize with coordinator. | 
MAC API class introduction
This chapter introduces MAC mesh interface mac_api_s. It is a structure that defines the function callbacks needed by a service user.
The base class defines the functions for two-way communications between an external MAC and service user. The application creates a mac_api_s object by calling the MAC adapter's create function. The newly created object is then passed to Nanostack, which initializes its own callback functions by calling the mac_api_initialize() function. A service user operates MAC by calling MLME or MCPS primitive functions.
The MAC API class structure mac_api_t is defined as:
typedef struct mac_api_s {
    //Service user defined initialization function which is called when Nanostack takes MAC into use
    mac_api_initialize              *mac_initialize;
    mac_api_enable_mcps_ext     *mac_mcps_extension_enable;
    //MAC adapter function callbacks for MLME & MCPS SAP
    mlme_request                    *mlme_req;
    mcps_data_request               *mcps_data_req;
    mcps_data_request_ext       *mcps_data_req_ext;
    mcps_purge_request                  *mcps_purge_req;
    //Service user defined function callbacks
    mcps_data_confirm               *data_conf_cb;
    mcps_data_confirm_ext       *data_conf_ext_cb;
    mcps_data_indication                *data_ind_cb;
    mcps_data_indication_ext    *data_ind_ext_cb;
    mcps_ack_data_req_ext       *enhanced_ack_data_req_cb;
    mcps_purge_confirm                  *purge_conf_cb;
    mlme_confirm                        *mlme_conf_cb;
    mlme_indication                     *mlme_ind_cb;
    //MAC extension API for service user
    mac_ext_mac64_address_set           *mac64_set;
    mac_ext_mac64_address_get           *mac64_get;
    mac_storage_decription_sizes_get *mac_storage_sizes_get;
    int8_t                              parent_id;
    uint16_t                            phyMTU;
};
| Member | Description | 
|---|---|
| mac_initialize | MAC initialize function called by Nanostack. | 
| mac_mcps_extension_enable | MAC MCPS IE extension enable function, optional feature. Enable is only possible when driver extension is supported. | 
| mlme_req | MLME request function to use MLME-SAP commands, MAC defines. | 
| mcps_data_req | MCPS data request function to use, MAC defines. | 
| mcps_data_req_ext | MAC MCPS data request with Information element extension function to use. | 
| mcps_purge_req | MCPS purge request function to use, MAC defines. | 
| mcps_data_confirm | MCPS data confirm callback function, service user defines. | 
| mcps_data_confirm_ext | MAC MCPS data confirm with payload callback function. | 
| data_ind_cb | MCPS data indication callback function, service user defines. | 
| data_ind_ext_cb | MAC MCPS data indication with IE extension's callback function. | 
| enhanced_ack_data_req_cb | Enhanced ACK IE element and payload request from MAC user. | 
| purge_conf_cb | MCPS purge confirm callback function, service user defines. | 
| mlme_conf_cb | MLME confirm callback function, service user defines. | 
| mlme_ind_cb | MLME indication callback function, service user defines. | 
| mac64_set | MAC extension function to set mac64 address. | 
| mac64_get | MAC extension function to get mac64 address. | 
| mac_storage_sizes_get | Getter function to query data storage sizes from MAC. | 
| parent_id | Service user ID used to indentify the MAC service user. Optional. | 
| phyMTU | Maximum Transmission Unit (MTU) used by MAC. Standard 802.15.4 MAC must set 127. | 
IEEE 802.15.4 MAC sublayer extended APIs
This document introduces how to use extented features.
Initialize API
typedef int8_t mac_api_enable_mcps_ext(mac_api_t *api, mcps_data_indication_ext *data_ind_cb, mcps_data_confirm_ext *data_cnf_cb, mcps_ack_data_req_ext *ack_data_req_cb);
| Parameter | Description | 
|---|---|
| api | Pointer, which the application creates. | 
| data_ind_cb | Upper layer function to handle MCPS indications with information elements. | 
| data_cnf_cb | Upper layer function to handle MCPS confirmation with information elements. | 
| ack_data_req_cb | Upper layer function for requesting enhanced ACK payload. | 
Function can be called when SW MAC is created and initialization procedure is done. Enable could fail if delivered driver does not support required extensions.
Data Indication API
typedef void mcps_data_indication_ext(const mac_api_t* api, const mcps_data_ind_t *data, const mcps_data_ie_list_t *ie_ext);
| Parameter | Description | 
|---|---|
| api | Pointer, which the application creates. | 
| data | Data MCPS-DATA.indication specific values. | 
| ie_ext | Information element list. | 
Extended data indication handler includes an information element list, which is defined as:
typedef struct mcps_data_ie_list {
    uint8_t *headerIeList;
    uint8_t *payloadIeList;
    uint16_t headerIeListLength;
    uint16_t payloadIeListLength;
} mcps_data_ie_list_t;
| Member | Description | 
|---|---|
| headerIeList | Header information IE's list without terminator. | 
| payloadIeList | Payload information IE's list without terminator. | 
| headerIeListLength | Header information IE's list length in bytes. | 
| headerIeListLength | Payload information IE's list length in bytes. | 
Data request API
typedef void mcps_data_request_ext(const mac_api_t* api, const mcps_data_req_t *data, const mcps_data_req_ie_list_t *ie_ext, const struct channel_list_s *asynch_channel_list);
| Parameter | Description | 
|---|---|
| api | pointer, which is created by application | 
| data | MCPS-DATA.request specific values. | 
| ie_ext | Information element list to MCPS-DATA.request. | 
| asynch_channel_list | Optional channel list to asynch data request. Give NULL when normal data request. | 
Asynch data request is MAC standard extension. asynch_channel_list includes a channel mask, which the channel message is requested to send.
Structure for IEEE 802.15.4-2015 MCPS data extension to Request.
typedef struct mcps_data_req_ie_list {
    ns_ie_iovec_t *headerIeVectorList;
    ns_ie_iovec_t *payloadIeVectorList;
    uint16_t headerIovLength;
    uint16_t payloadIovLength;
} mcps_data_req_ie_list_t;
| Member | Description | 
|---|---|
| headerIeVectorList | Header IE element list. NULLif no elements. | 
| payloadIeVectorList | Payload IE element list. NULLif no elements. | 
| headerIovLength | Header IE element list size, set to 0 if no elements. | 
| payloadIovLength | Payload IE element list size, set to 0 if no elements. | 
IE element could be divided to multiple vector, which MAC just write to message direct. One vector could also include multiple information elements or just header part. That's enable flexible to way to generate list and should enable memory friendly way to share information elements.
Set headerIovLength or payloadIovLength to 0 if not send any element.
Scatter-gather descriptor for MCPS request IE element list:
typedef struct ns_ie_iovec {
    void *ieBase;
    uint_fast16_t iovLen;
} ns_ie_iovec_t;
| Member | Description | 
|---|---|
| ieBase | IE element pointer. | 
| iovLen | IE element length. | 
Data confirmation API
typedef void mcps_data_confirm_ext(const mac_api_t* api, const mcps_data_conf_t *data, const mcps_data_conf_payload_t *conf_data);
| Parameter | Description | 
|---|---|
| api | Pointer, which the application creates. | 
| data | MCPS-DATA.confirmspecific values. | 
| conf_data | Possible confirmation data. | 
Confirmation data is packet to next structure:
typedef struct mcps_data_conf_payload_s {
    uint8_t *headerIeList;
    uint8_t *payloadIeList;
    uint8_t *payloadPtr;
    uint16_t headerIeListLength;
    uint16_t payloadIeListLength;
    uint16_t payloadLength;
} mcps_data_conf_payload_t;
| Member | Description | 
|---|---|
| headerIeList | Header information IE's list without terminator | 
| payloadIeList | Payload information IE's list without terminator. | 
| payloadPtr | Ack payload pointer. | 
| headerIeListLength | Header information IE's list length in bytes. | 
| payloadIeListLength | Payload information IE's list length in bytes. | 
| payloadLength | Payload length in bytes. | 
Enhanced ACK payload and IE write API
typedef void mcps_ack_data_req_ext(const mac_api_t* api, mcps_ack_data_payload_t *data, int8_t rssi, uint8_t lqi);
| Parameter | Description | 
|---|---|
| api | Pointer, which the application creates. | 
| data | Pointer where MAC user sets payload, IE pointers and length. | 
| rssi | Signal strength for received packet. | 
| lqi | Link quality to neighbor. | 
Signal strength and link quality are additional information that devices can share in an Ack message.
Structure for give Ack payload:
typedef struct mcps_ack_data_payload {
    struct mcps_data_req_ie_list ie_elements;
    uint8_t *payloadPtr;
    uint16_t payloadLength;
} mcps_ack_data_payload_t;
| Member | Description | 
|---|---|
| ie_elements | Information element header and payload elements. | 
| payloadPtr | Ack payload pointer. | 
| payloadLength | Payload length in bytes. | 
If payload length is zero, payloadPtr is NULL. Information element headers may also contain empty vectors.
MAC API standard extensions
This section introduces MAC API standard extensions.
MAC 64-bit address set and get
Nanostack uses 64-bit address set and get. There are two 64-bit addresses available:
- NVM EUI64.
- Dynamic 64-bit address used at the link layer.
Address set and get support two different 64-bit addresses:
| Address enumeration type | Description | 
|---|---|
| MAC_EXTENDED_READ_ONLY | A unique EUI64. | 
| MAC_EXTENDED_DYNAMIC | Dynamic 64-bit address. Same as EUI64 after boot. | 
MAC maximum storage information get
Usually, HW MAC and SW MAC have static keys and neighbor list sizes. Nanostack always asks the maximum size to limit its neighbor table size. The service user must define the mac_storage_sizes_get() function returning the following values:
- MAC device description list size (must be > 1).
- MAC key description list size (must be > 1).
MLME attribute extension
Nanostack uses MLME attribute extensions, which you must port to the HW MAC adapter. To configure the extensions, use the MLME-SET-REQ command.
| Enumeration type | Value | Description | 
|---|---|---|
| macAcceptByPassUnknowDevice | 0xfc | Accept data through MAC if the group key and MIC can authenticate the packet data. Security enforcement point must carefully handle these packets. | 
| macLoadBalancingBeaconTx | 0xfd | Trigger to MAC layer to send a beacon. Called by the load balancer module periodically. | 
| macLoadBalancingAcceptAnyBeacon | 0xfe | Configure MAC layer to accept beacons from other networks. Enabled by load balancer, default value is False. Value size boolean,true=enable,false=disable. | 
HW MAC
To use HW MAC, you need to create an adapter class that links function calls between Nanostack and HW MAC. To create the adapter class, implement the functions defined in the mac_api_s structure. When HW MAC generates an event, the adapter must handle it and do a parameter adaptation before calling the correct function from the mac_api_s structure. You may need the same parameter adaptation for requests from Nanostack to HW MAC.
Note: Function calls from Nanostack to HW MAC must be nonblocking in the adapter layer.