Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Fork of XBeeLib by
XBeeZB/XBeeZB.cpp
- Committer:
- kkalsi
- Date:
- 2017-12-02
- Revision:
- 10:ee65d28bf721
- Parent:
- 9:780db84ce891
File content as of revision 10:ee65d28bf721:
/** * Copyright (c) 2015 Digi International Inc., * All rights not expressly granted are reserved. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. * * Digi International Inc. 11001 Bren Road East, Minnetonka, MN 55343 * ======================================================================= */ #include "XBeeZB.h" #include "IO/IOSampleZB.h" #include "Frames/ZigbeeFrames.h" using namespace XBeeLib; #define BROADCAST_RADIUS_USE_NH 0x00 /* Class constructor */ XBeeZB::XBeeZB(PinName tx, PinName rx, PinName reset, PinName rts, PinName cts, int baud) : XBee(tx, rx, reset, rts, cts, baud), _nd_handler(NULL), _rx_pkt_handler(NULL), _io_data_handler(NULL) { } RadioStatus XBeeZB::init() { RadioStatus retval = XBee::init(); uint16_t addr16; RadioStatus error = get_network_address(&addr16); if (error == Success) { digi_log(LogLevelInfo, "ADDR16: %04x\r\n", addr16); } else { digi_log(LogLevelInfo, "ADDR16: UNKNOWN\r\n"); } const RadioProtocol radioProtocol = get_radio_protocol(); if (radioProtocol != ZigBee) { digi_log(LogLevelError, "Radio protocol does not match, needed a %d got a %d\r\n", ZigBee, radioProtocol); retval = Failure; } assert(radioProtocol == ZigBee); return retval; } /* Class destructor */ XBeeZB::~XBeeZB() { unregister_node_discovery_cb(); unregister_receive_cb(); unregister_io_sample_cb(); } RadioStatus XBeeZB::set_channel_mask(uint16_t chmask) { AtCmdFrame::AtCmdResp cmdresp; cmdresp = set_param("SC", chmask); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } return Success; } RadioStatus XBeeZB::get_channel_mask(uint16_t * const chmask) { if (chmask == NULL) { return Failure; } AtCmdFrame::AtCmdResp cmdresp; uint32_t var32; cmdresp = get_param("SC", &var32); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } *chmask = var32; return Success; } RadioStatus XBeeZB::set_panid(uint64_t panid) { uint8_t panid_u8[8]; AtCmdFrame::AtCmdResp cmdresp; rmemcpy(panid_u8, (const uint8_t *) &panid, sizeof panid_u8); cmdresp = set_param("ID", panid_u8, sizeof panid_u8); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } return Success; } RadioStatus XBeeZB::get_operating_panid(uint64_t * const opanid) { if (opanid == NULL) { return Failure; } uint8_t opanid_u8[8]; uint16_t len = sizeof opanid_u8; AtCmdFrame::AtCmdResp cmdresp; cmdresp = get_param("OP", opanid_u8, &len); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } if (len != sizeof opanid_u8) { digi_log(LogLevelError, "XBeeZB::get_operating_panid: Read %d bytes instead of %d for OP", len, sizeof opanid_u8); return Failure; } rmemcpy((uint8_t *)opanid, opanid_u8, len); return Success; } RadioStatus XBeeZB::get_configured_panid(uint64_t * const panid) { if (panid == NULL) { return Failure; } uint8_t panid_u8[8]; uint16_t len = sizeof panid_u8; AtCmdFrame::AtCmdResp cmdresp; cmdresp = get_param("ID", panid_u8, &len); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } if (len != sizeof panid_u8) { digi_log(LogLevelError, "XBeeZB::get_configured_panid: Read %d bytes instead of %d for ID", len, sizeof panid_u8); return Failure; } rmemcpy((uint8_t *)panid, panid_u8, len); return Success; } RadioStatus XBeeZB::set_panid(const RemoteXBee& remote, uint64_t panid) { uint8_t panid_u8[8]; AtCmdFrame::AtCmdResp cmdresp; rmemcpy(panid_u8, (const uint8_t *) &panid, sizeof panid_u8); cmdresp = set_param(remote, "ID", panid_u8, sizeof panid_u8); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } return Success; } RadioStatus XBeeZB::get_operating_panid(const RemoteXBee& remote, uint64_t * const opanid) { if (opanid == NULL) { return Failure; } uint8_t opanid_u8[8]; uint16_t len = sizeof opanid_u8; AtCmdFrame::AtCmdResp cmdresp; cmdresp = get_param(remote, "OP", opanid_u8, &len); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } if (len != sizeof opanid_u8) { digi_log(LogLevelError, "XBeeZB::get_operating_panid: Read %d bytes instead of %d for OP", len, sizeof opanid_u8); return Failure; } rmemcpy((uint8_t *)opanid, opanid_u8, len); return Success; } RadioStatus XBeeZB::get_configured_panid(const RemoteXBee& remote, uint64_t * const panid) { if (panid == NULL) { return Failure; } uint8_t panid_u8[8]; uint16_t len = sizeof panid_u8; AtCmdFrame::AtCmdResp cmdresp; cmdresp = get_param(remote, "ID", panid_u8, &len); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } if (len != sizeof panid_u8) { digi_log(LogLevelError, "XBeeZB::get_configured_panid: Read %d bytes instead of %d for ID", len, sizeof panid_u8); return Failure; } rmemcpy((uint8_t *)panid, panid_u8, len); return Success; } RadioStatus XBeeZB::get_network_address(uint16_t * const addr16) { if (addr16 == NULL) { return Failure; } AtCmdFrame::AtCmdResp cmdresp; uint32_t var32; cmdresp = get_param("MY", &var32); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } *addr16 = var32; return Success; } RadioStatus XBeeZB::get_node_discovery_timeout(uint16_t * const timeout_ms) { AtCmdFrame::AtCmdResp cmdresp; uint32_t var32; cmdresp = get_param("NT", &var32); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } *timeout_ms = (uint16_t)var32; /* No N? command available for this protocol. Add a fix 1s guard time */ *timeout_ms += 1000; return Success; } RadioStatus XBeeZB::get_node_discovery_timeout(uint16_t * const timeout_ms, bool * const wait_for_complete_timeout) { const RadioStatus status = get_node_discovery_timeout(timeout_ms); *wait_for_complete_timeout = false; return status; } RadioStatus XBeeZB::check_for_coordinator_at_start(bool enable) { AtCmdFrame::AtCmdResp cmdresp; cmdresp = set_param("JV", (uint8_t)enable); return cmdresp == AtCmdFrame::AtCmdRespOk ? Success : Failure; } RadioStatus XBeeZB::set_network_security_key(const uint8_t * const key, const uint16_t length) { if (key == NULL || length == 0 || length > 16) { return Failure; } AtCmdFrame::AtCmdResp cmdresp; cmdresp = set_param("NK", key, length); return cmdresp == AtCmdFrame::AtCmdRespOk ? Success : Failure; } RadioStatus XBeeZB::set_encryption_options(const uint8_t options) { AtCmdFrame::AtCmdResp cmdresp; cmdresp = set_param("EO", options); return cmdresp == AtCmdFrame::AtCmdRespOk ? Success : Failure; } void XBeeZB::radio_status_update(AtCmdFrame::ModemStatus modem_status) { /* Update the radio status variables */ if (modem_status == AtCmdFrame::HwReset) { _hw_reset_cnt++; } else if (modem_status == AtCmdFrame::WdReset) { _wd_reset_cnt++; } _modem_status = modem_status; digi_log(LogLevelDebug, "\r\nUpdating radio status: %02x\r\n", modem_status); } TxStatus XBeeZB::send_data(const RemoteXBee& remote, const uint8_t *const data, uint16_t len, bool syncr) { if (!remote.is_valid_addr64b()) { return TxStatusInvalidAddr; } const uint64_t remote64 = remote.get_addr64(); const uint16_t remote16 = remote.get_addr16(); TxFrameZB frame = TxFrameZB(remote64, remote16, BROADCAST_RADIUS_USE_NH, _tx_options, data, len); if (syncr) { return send_data(&frame); } else { frame.set_data(0, 0); /* Set frame id to 0 so there is no answer */ send_api_frame(&frame); return TxStatusSuccess; } } TxStatus XBeeZB::send_data(const RemoteXBee& remote, uint8_t source_ep, uint8_t dest_ep, uint16_t cluster_id, uint16_t profile_id, const uint8_t *const data, uint16_t len, bool syncr) { if (!remote.is_valid_addr64b()) { return TxStatusInvalidAddr; } const uint64_t remote64 = remote.get_addr64(); const uint16_t remote16 = remote.get_addr16(); TxFrameZB frame = TxFrameZB(remote64, remote16, source_ep, dest_ep, cluster_id, profile_id, BROADCAST_RADIUS_USE_NH, _tx_options, data, len); if (syncr) { return send_data(&frame); } else { frame.set_data(0, 0); /* Set frame id to 0 so there is no answer */ send_api_frame(&frame); return TxStatusSuccess; } } TxStatus XBeeZB::send_data_to_coordinator(const uint8_t *const data, uint16_t len, bool syncr) { const uint64_t remaddr = ADDR64_COORDINATOR; TxFrameZB frame = TxFrameZB(remaddr, ADDR16_UNKNOWN, BROADCAST_RADIUS_USE_NH, _tx_options, data, len); if (syncr) { return send_data(&frame); } else { frame.set_data(0, 0); /* Set frame id to 0 so there is no answer */ send_api_frame(&frame); return TxStatusSuccess; } } RemoteXBeeZB XBeeZB::get_remote_node_by_id(const char * const node_id) { uint64_t addr64; uint16_t addr16; _get_remote_node_by_id(node_id, &addr64, &addr16); return RemoteXBeeZB(addr64, addr16); } XBeeZB::AssocStatus XBeeZB::get_assoc_status(void) { return (AssocStatus)get_AI(); } bool XBeeZB::is_joined() { return get_assoc_status() == Joined ? true : false; } void XBeeZB::register_node_discovery_cb(node_discovery_zb_cb_t function) { if (_nd_handler == NULL) { _nd_handler = new FH_NodeDiscoveryZB(); register_frame_handler(_nd_handler); } _nd_handler->register_node_discovery_cb(function); } void XBeeZB::unregister_node_discovery_cb() { if (_nd_handler != NULL) { _nd_handler->unregister_node_discovery_cb(); unregister_frame_handler(_nd_handler); delete _nd_handler; _nd_handler = NULL; /* as delete does not set to NULL */ } } void XBeeZB::register_receive_cb(receive_zb_cb_t function) { if (_rx_pkt_handler == NULL) { _rx_pkt_handler = new FH_RxPacketZB(); register_frame_handler(_rx_pkt_handler); } _rx_pkt_handler->register_receive_cb(function); } void XBeeZB::unregister_receive_cb() { if (_rx_pkt_handler != NULL) { _rx_pkt_handler->unregister_receive_cb(); unregister_frame_handler(_rx_pkt_handler); delete _rx_pkt_handler; _rx_pkt_handler = NULL; /* as delete does not set to NULL */ } } void XBeeZB::register_io_sample_cb(io_data_cb_zb_t function) { if (_io_data_handler == NULL) { _io_data_handler = new FH_IoDataSampeZB(); register_frame_handler(_io_data_handler); } _io_data_handler->register_io_data_cb(function); } void XBeeZB::unregister_io_sample_cb() { if (_io_data_handler != NULL) { _io_data_handler->unregister_io_data_cb(); unregister_frame_handler(_io_data_handler); delete _io_data_handler; _io_data_handler = NULL; /* as delete does not set to NULL */ } } AtCmdFrame::AtCmdResp XBeeZB::get_param(const RemoteXBee& remote, const char * const param, uint32_t * const data) { if (!remote.is_valid_addr64b()) { return AtCmdFrame::AtCmdRespInvalidAddr; } const uint64_t remote64 = remote.get_addr64(); const uint16_t remote16 = remote.get_addr16(); uint16_t len = sizeof *data; AtCmdFrame::AtCmdResp atCmdResponse; AtCmdFrame cmd_frame = AtCmdFrame(remote64, remote16, param); atCmdResponse = send_at_cmd(&cmd_frame, (uint8_t *)data, &len, RadioRemote); if (atCmdResponse == AtCmdFrame::AtCmdRespOk && len > sizeof *data) { atCmdResponse = AtCmdFrame::AtCmdRespLenMismatch; } return atCmdResponse; } AtCmdFrame::AtCmdResp XBeeZB::set_param(const RemoteXBee& remote, const char * const param, uint32_t data) { if (!remote.is_valid_addr64b()) { return AtCmdFrame::AtCmdRespInvalidAddr; } const uint64_t remote64 = remote.get_addr64(); const uint16_t remote16 = remote.get_addr16(); AtCmdFrame cmd_frame = AtCmdFrame(remote64, remote16, param, data); return send_at_cmd(&cmd_frame, NULL, NULL, RadioRemote); } AtCmdFrame::AtCmdResp XBeeZB::set_param(const RemoteXBee& remote, const char * const param, const uint8_t * data, uint16_t len) { if (!remote.is_valid_addr64b()) { return AtCmdFrame::AtCmdRespInvalidAddr; } const uint64_t remote64 = remote.get_addr64(); const uint16_t remote16 = remote.get_addr16(); AtCmdFrame cmd_frame = AtCmdFrame(remote64, remote16, param, data, len); return send_at_cmd(&cmd_frame, NULL, NULL, RadioRemote); } AtCmdFrame::AtCmdResp XBeeZB::get_param(const RemoteXBee& remote, const char * const param, uint8_t * const data, uint16_t * const len) { if (!remote.is_valid_addr64b()) { return AtCmdFrame::AtCmdRespInvalidAddr; } const uint64_t remote64 = remote.get_addr64(); const uint16_t remote16 = remote.get_addr16(); AtCmdFrame cmd_frame = AtCmdFrame(remote64, remote16, param); return send_at_cmd(&cmd_frame, data, len, RadioRemote, false); } static void get_dio_cmd(XBeeZB::IoLine line, char * const iocmd) { if (line >= XBeeZB::DIO10) { iocmd[0] = 'P'; iocmd[1] = '0' + line - XBeeZB::DIO10; } else { iocmd[0] = 'D'; iocmd[1] = '0' + line; } iocmd[2] = '\0'; } RadioStatus XBeeZB::set_pin_config(const RemoteXBee& remote, IoLine line, IoMode mode) { AtCmdFrame::AtCmdResp cmdresp; char iocmd[3]; get_dio_cmd(line, iocmd); cmdresp = set_param(remote, iocmd, (uint8_t)mode); if (cmdresp != AtCmdFrame::AtCmdRespOk) { digi_log(LogLevelError, "set_pin_config: set_param returned %d\r\n", cmdresp); return Failure; } return Success; } RadioStatus XBeeZB::get_pin_config(const RemoteXBee& remote, IoLine line, IoMode * const mode) { AtCmdFrame::AtCmdResp cmdresp; char iocmd[3]; get_dio_cmd(line, iocmd); uint32_t var32; cmdresp = get_param(remote, iocmd, &var32); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } *mode = (IoMode)var32; return Success; } RadioStatus XBeeZB::set_dio(const RemoteXBee& remote, IoLine line, DioVal val) { return set_pin_config(remote, line, val == Low ? DigitalOutLow : DigitalOutHigh); } RadioStatus XBeeZB::get_dio(const RemoteXBee& remote, IoLine line, DioVal * const val) { return get_iosample(remote).get_dio(line, val); } RadioStatus XBeeZB::get_adc(const RemoteXBee& remote, IoLine line, uint16_t * const val) { return get_iosample(remote).get_adc(line, val); } IOSampleZB XBeeZB::get_iosample(const RemoteXBee& remote) { uint8_t io_sample[MAX_IO_SAMPLE_ZB_LEN]; uint16_t len = sizeof io_sample; RadioStatus resp = _get_iosample(remote, io_sample, &len); if (resp != Success) { digi_log(LogLevelError, "XBeeZB::get_iosample failed to get an IOSample\r\n"); len = 0; } return IOSampleZB(io_sample, len); } static uint16_t get_dio_pr_mask(XBeeZB::IoLine line) { switch (line) { case XBeeZB::DIO4: return (1 << 0); case XBeeZB::DIO3_AD3: return (1 << 1); case XBeeZB::DIO2_AD2: return (1 << 2); case XBeeZB::DIO1_AD1: return (1 << 3); case XBeeZB::DIO0_AD0: return (1 << 4); case XBeeZB::DIO6: return (1 << 5); case XBeeZB::DIO5: return (1 << 8); case XBeeZB::DIO12: return (1 << 10); case XBeeZB::DIO10: return (1 << 11); case XBeeZB::DIO11: return (1 << 12); case XBeeZB::DIO7: return (1 << 13); default: return 0; } } RadioStatus XBeeZB::set_pin_pull_up(const RemoteXBee& remote, IoLine line, bool enable) { AtCmdFrame::AtCmdResp cmdresp; uint32_t var32; uint16_t pr; cmdresp = get_param(remote, "PR", &var32); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } pr = var32; const uint16_t dio_mask = get_dio_pr_mask(line); if (dio_mask == 0) { digi_log(LogLevelError, "XBeeZB::set_pin_pull_up: invalid pin %d\r\n", line); return Failure; } if (enable) { pr |= dio_mask; } else { pr &= ~dio_mask; } cmdresp = set_param(remote, "PR", pr); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } return Success; } static uint16_t get_dio_ic_mask(XBeeZB::IoLine line) { if (line < XBeeZB::DIO12) { return (1 << line); } return 0; } RadioStatus XBeeZB::enable_dio_change_detection(const RemoteXBee& remote, IoLine line, bool enable) { if (line > DIO11) { digi_log(LogLevelError, "XBeeZB::enable_dio_change_detection: pin not supported (%d)\r\n", line); return Failure; } AtCmdFrame::AtCmdResp cmdresp; uint32_t var32; uint16_t ic; cmdresp = get_param(remote, "IC", &var32); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } ic = var32; const uint16_t dio_mask = get_dio_ic_mask(line); if (dio_mask == 0) { digi_log(LogLevelError, "XBeeZB::enable_dio_change_detection: invalid pin %d\r\n", line); return Failure; } if (enable) { ic |= dio_mask; } else { ic &= ~dio_mask; } cmdresp = set_param(remote, "IC", ic); if (cmdresp != AtCmdFrame::AtCmdRespOk) { return Failure; } return Success; }