1081 lines
33 KiB
C
1081 lines
33 KiB
C
/*
|
|
* Copyright (c) 2014, NXP and SICS Swedish ICT.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the Institute nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* This file is part of the Contiki operating system.
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* \file
|
|
* Contiki driver for NXP JN516X using MMAC interface
|
|
* \authors
|
|
* Beshr Al Nahas <beshr@sics.se>
|
|
* Simon Duquennot <simonduq@sics.se>
|
|
* Atis Elsts <atis.elsts@sics.se>
|
|
*
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include "contiki.h"
|
|
#include "dev/leds.h"
|
|
#include "sys/rtimer.h"
|
|
#include "net/packetbuf.h"
|
|
#include "net/netstack.h"
|
|
#include "net/mac/framer/frame802154.h"
|
|
#include "lib/crc16.h"
|
|
#include "lib/ringbufindex.h"
|
|
|
|
#include "AppHardwareApi.h"
|
|
#include "MMAC.h"
|
|
#include "micromac-radio.h"
|
|
#include "JPT.h"
|
|
#include "PeripheralRegs.h"
|
|
|
|
/* This driver configures the radio in PHY mode and does address decoding
|
|
* and acknowledging in software. */
|
|
|
|
#define DEBUG DEBUG_NONE
|
|
#include "net/ipv6/uip-debug.h"
|
|
|
|
#ifdef MICROMAC_CONF_RADIO_MAC
|
|
#define MICROMAC_RADIO_MAC MICROMAC_CONF_RADIO_MAC
|
|
#else
|
|
#define MICROMAC_RADIO_MAC 0
|
|
#endif
|
|
|
|
#if MICROMAC_RADIO_MAC
|
|
#define MICROMAC_FRAME tsMacFrame
|
|
#else
|
|
#define MICROMAC_FRAME tsPhyFrame
|
|
#endif
|
|
|
|
/* Perform CRC check for received packets in SW,
|
|
* since we use PHY mode which does not calculate CRC in HW */
|
|
#define CRC_SW 1
|
|
|
|
#define CHECKSUM_LEN 2
|
|
|
|
/* Max packet duration: 5 + 127 + 2 bytes, 32us per byte */
|
|
#define MAX_PACKET_DURATION US_TO_RTIMERTICKS((127 + 2) * 32 + RADIO_DELAY_BEFORE_TX)
|
|
/* Max ACK duration: 5 + 3 + 2 bytes */
|
|
#define MAX_ACK_DURATION US_TO_RTIMERTICKS((3 + 2) * 32 + RADIO_DELAY_BEFORE_TX)
|
|
|
|
/* Test-mode pins output on dev-kit */
|
|
#define RADIO_TEST_MODE_HIGH_PWR 1
|
|
#define RADIO_TEST_MODE_ADVANCED 2
|
|
#define RADIO_TEST_MODE_DISABLED 0
|
|
|
|
#ifndef RADIO_TEST_MODE
|
|
#define RADIO_TEST_MODE RADIO_TEST_MODE_DISABLED
|
|
#endif /* RADIO_TEST_MODE */
|
|
|
|
/* The number of input buffers */
|
|
#ifndef MIRCOMAC_CONF_BUF_NUM
|
|
#define MIRCOMAC_CONF_BUF_NUM 2
|
|
#endif /* MIRCOMAC_CONF_BUF_NUM */
|
|
|
|
/* Init radio channel */
|
|
#ifndef MICROMAC_CONF_CHANNEL
|
|
#define MICROMAC_CONF_CHANNEL 26
|
|
#endif
|
|
|
|
/* Default energy level threshold for clear channel detection */
|
|
#ifndef MICROMAC_CONF_CCA_THR
|
|
#define MICROMAC_CONF_CCA_THR 39 /* approximately -85 dBm */
|
|
#endif /* MICROMAC_CONF_CCA_THR */
|
|
|
|
#if (JENNIC_CHIP == JN5169)
|
|
#define OUTPUT_POWER_MAX 10
|
|
#define OUTPUT_POWER_MIN (-32)
|
|
#define ABS_OUTPUT_POWER_MIN (32)
|
|
#else
|
|
#define OUTPUT_POWER_MAX 0
|
|
#define OUTPUT_POWER_MIN (-32)
|
|
#endif
|
|
|
|
/* Default Tx power [dBm] (between OUTPUT_POWER_MIN and OUTPUT_POWER_MAX) */
|
|
#ifndef MICROMAC_CONF_TX_POWER
|
|
#define MICROMAC_CONF_TX_POWER 0
|
|
#endif
|
|
|
|
/* Autoack */
|
|
#ifndef MICROMAC_CONF_AUTOACK
|
|
#define MICROMAC_CONF_AUTOACK 1
|
|
#endif /* MICROMAC_CONF_AUTOACK */
|
|
|
|
/* Set radio always on for now because this is what Contiki MAC layers
|
|
* expect. */
|
|
#ifndef MICROMAC_CONF_ALWAYS_ON
|
|
#define MICROMAC_CONF_ALWAYS_ON 1
|
|
#endif /* MICROMAC_CONF_ALWAYS_ON */
|
|
|
|
#define BUSYWAIT_UNTIL(cond, max_time) \
|
|
do { \
|
|
rtimer_clock_t t0; \
|
|
t0 = RTIMER_NOW(); \
|
|
while(!(cond) && RTIMER_CLOCK_LT(RTIMER_NOW(), t0 + (max_time))) ; \
|
|
} while(0)
|
|
|
|
/* Local variables */
|
|
static volatile signed char radio_last_rssi;
|
|
static volatile uint8_t radio_last_correlation; /* LQI */
|
|
|
|
/* Did we miss a request to turn the radio on due to overflow? */
|
|
static volatile uint8_t missed_radio_on_request = 0;
|
|
|
|
/* Poll mode disabled by default */
|
|
static uint8_t poll_mode = 0;
|
|
/* (Software) frame filtering enabled by default */
|
|
static uint8_t frame_filtering = 1;
|
|
/* (Software) autoack */
|
|
static uint8_t autoack_enabled = MICROMAC_CONF_AUTOACK;
|
|
/* CCA before sending? Disabled by default. */
|
|
static uint8_t send_on_cca = 0;
|
|
|
|
/* Current radio channel */
|
|
static int current_channel = MICROMAC_CONF_CHANNEL;
|
|
|
|
/* Current set point tx power
|
|
Actual tx power may be different. Use get_txpower() for actual power */
|
|
static int current_tx_power = MICROMAC_CONF_TX_POWER;
|
|
|
|
/* an integer between 0 and 255, used only with cca() */
|
|
static uint8_t cca_thershold = MICROMAC_CONF_CCA_THR;
|
|
|
|
/* Tx in progress? */
|
|
static volatile uint8_t tx_in_progress = 0;
|
|
/* Are we currently listening? */
|
|
static volatile uint8_t listen_on = 0;
|
|
|
|
/* Is the driver currently transmitting a software ACK? */
|
|
static uint8_t in_ack_transmission = 0;
|
|
|
|
/* TX frame buffer */
|
|
static MICROMAC_FRAME tx_frame_buffer;
|
|
|
|
/* RX frame buffer */
|
|
static MICROMAC_FRAME *rx_frame_buffer;
|
|
|
|
/* Frame buffer pointer to read from */
|
|
static MICROMAC_FRAME *input_frame_buffer = NULL;
|
|
|
|
/* Ringbuffer for received packets in interrupt enabled mode */
|
|
static struct ringbufindex input_ringbuf;
|
|
static MICROMAC_FRAME input_array[MIRCOMAC_CONF_BUF_NUM];
|
|
|
|
/* SFD timestamp in RTIMER ticks */
|
|
static volatile uint32_t last_packet_timestamp = 0;
|
|
|
|
/* Local functions prototypes */
|
|
static int on(void);
|
|
static int off(void);
|
|
#if !MICROMAC_RADIO_MAC
|
|
static int is_packet_for_us(uint8_t *buf, int len, int do_send_ack);
|
|
#endif
|
|
static void set_frame_filtering(uint8_t enable);
|
|
static rtimer_clock_t get_packet_timestamp(void);
|
|
static void set_txpower(int8_t power);
|
|
void set_channel(int c);
|
|
static void radio_interrupt_handler(uint32 mac_event);
|
|
static int get_detected_energy(void);
|
|
static int get_rssi(void);
|
|
static void read_last_rssi(void);
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
PROCESS(micromac_radio_process, "micromac_radio_driver");
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
/* Custom Radio parameters */
|
|
#ifndef RADIO_RX_MODE_POLL_MODE
|
|
#define RADIO_PARAM_LAST_RSSI 0x80
|
|
#define RADIO_PARAM_LAST_PACKET_TIMESTAMP 0x81
|
|
#define RADIO_RX_MODE_POLL_MODE (1 << 2)
|
|
#endif /* RADIO_RX_MODE_POLL_MODE */
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
static rtimer_clock_t
|
|
get_packet_timestamp(void)
|
|
{
|
|
/* Wait for an edge */
|
|
uint32_t t = u32MMAC_GetTime();
|
|
while(u32MMAC_GetTime() == t);
|
|
/* Save SFD timestamp, converted from radio timer to RTIMER */
|
|
last_packet_timestamp = RTIMER_NOW() -
|
|
RADIO_TO_RTIMER((uint32_t)(u32MMAC_GetTime() - (u32MMAC_GetRxTime() - 1)));
|
|
/* The remaining measured error is typically in range 0..16 usec.
|
|
* Center it around zero, in the -8..+8 usec range. */
|
|
last_packet_timestamp -= US_TO_RTIMERTICKS(8);
|
|
return last_packet_timestamp;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
init_software(void)
|
|
{
|
|
int put_index;
|
|
/* Initialize ring buffer and first input packet pointer */
|
|
ringbufindex_init(&input_ringbuf, MIRCOMAC_CONF_BUF_NUM);
|
|
/* get pointer to next input slot */
|
|
put_index = ringbufindex_peek_put(&input_ringbuf);
|
|
if(put_index == -1) {
|
|
rx_frame_buffer = NULL;
|
|
printf("micromac_radio init:! no buffer available. Abort init.\n");
|
|
off();
|
|
return 0;
|
|
} else {
|
|
rx_frame_buffer = &input_array[put_index];
|
|
}
|
|
input_frame_buffer = rx_frame_buffer;
|
|
|
|
process_start(µmac_radio_process, NULL);
|
|
|
|
return 1;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
init(void)
|
|
{
|
|
int ret = 1;
|
|
tsExtAddr node_long_address;
|
|
uint16_t node_short_address;
|
|
static uint8_t is_initialized;
|
|
|
|
tx_in_progress = 0;
|
|
|
|
u32JPT_Init();
|
|
vMMAC_Enable();
|
|
|
|
/* Enable/disable interrupts */
|
|
if(poll_mode) {
|
|
vMMAC_EnableInterrupts(NULL);
|
|
vMMAC_ConfigureInterruptSources(0);
|
|
} else {
|
|
vMMAC_EnableInterrupts(&radio_interrupt_handler);
|
|
}
|
|
vMMAC_ConfigureRadio();
|
|
set_channel(current_channel);
|
|
set_txpower(current_tx_power);
|
|
|
|
vMMAC_GetMacAddress(&node_long_address);
|
|
/* Short addresses are disabled by default */
|
|
node_short_address = (uint16_t)node_long_address.u32L;
|
|
vMMAC_SetRxAddress(frame802154_get_pan_id(), node_short_address, &node_long_address);
|
|
|
|
/* Disable hardware backoff */
|
|
vMMAC_SetTxParameters(1, 0, 0, 0);
|
|
vMMAC_SetCutOffTimer(0, FALSE);
|
|
|
|
#if RADIO_TEST_MODE == RADIO_TEST_MODE_HIGH_PWR
|
|
/* Enable high power mode.
|
|
* In this mode DIO2 goes high during RX
|
|
* and DIO3 goes high during TX
|
|
**/
|
|
vREG_SysWrite(REG_SYS_PWR_CTRL,
|
|
u32REG_SysRead(REG_SYS_PWR_CTRL)
|
|
| REG_SYSCTRL_PWRCTRL_RFRXEN_MASK
|
|
| REG_SYSCTRL_PWRCTRL_RFTXEN_MASK);
|
|
#elif RADIO_TEST_MODE == RADIO_TEST_MODE_ADVANCED
|
|
/* output internal radio status on IO pins.
|
|
* See Chris@NXP email */
|
|
vREG_SysWrite(REG_SYS_PWR_CTRL,
|
|
u32REG_SysRead(REG_SYS_PWR_CTRL) | (1UL << 26UL));
|
|
#endif /* TEST_MODE */
|
|
|
|
if(!is_initialized) {
|
|
is_initialized = 1;
|
|
ret = init_software();
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
on(void)
|
|
{
|
|
/* No address matching or frame decoding */
|
|
if(rx_frame_buffer != NULL) {
|
|
#if MICROMAC_RADIO_MAC
|
|
vMMAC_StartMacReceive(rx_frame_buffer,
|
|
(uint16_t)(E_MMAC_RX_START_NOW
|
|
| E_MMAC_RX_USE_AUTO_ACK
|
|
| E_MMAC_RX_NO_MALFORMED
|
|
| E_MMAC_RX_NO_FCS_ERROR
|
|
| E_MMAC_RX_ADDRESS_MATCH
|
|
| E_MMAC_RX_ALIGN_NORMAL)
|
|
);
|
|
#else
|
|
vMMAC_StartPhyReceive(rx_frame_buffer,
|
|
(uint16_t)(E_MMAC_RX_START_NOW
|
|
| E_MMAC_RX_NO_FCS_ERROR) /* means: reject FCS errors */
|
|
);
|
|
#endif
|
|
} else {
|
|
missed_radio_on_request = 1;
|
|
}
|
|
ENERGEST_ON(ENERGEST_TYPE_LISTEN);
|
|
listen_on = 1;
|
|
return 1;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
off(void)
|
|
{
|
|
listen_on = 0;
|
|
tx_in_progress = 0;
|
|
|
|
ENERGEST_OFF(ENERGEST_TYPE_LISTEN);
|
|
|
|
/* The following would be needed with delayed Tx/Rx functions
|
|
* vMMAC_SetCutOffTimer(0, FALSE);*/
|
|
vMMAC_RadioOff();
|
|
|
|
return 1;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
transmit(unsigned short payload_len)
|
|
{
|
|
if(tx_in_progress) {
|
|
return RADIO_TX_COLLISION;
|
|
}
|
|
tx_in_progress = 1;
|
|
|
|
/* Energest */
|
|
if(listen_on) {
|
|
ENERGEST_OFF(ENERGEST_TYPE_LISTEN);
|
|
}
|
|
ENERGEST_ON(ENERGEST_TYPE_TRANSMIT);
|
|
|
|
/* Transmit and wait */
|
|
#if MICROMAC_RADIO_MAC
|
|
vMMAC_StartMacTransmit(&tx_frame_buffer,
|
|
E_MMAC_TX_START_NOW |
|
|
E_MMAC_TX_USE_AUTO_ACK |
|
|
(send_on_cca ? E_MMAC_TX_USE_CCA : E_MMAC_TX_NO_CCA));
|
|
#else
|
|
vMMAC_StartPhyTransmit(&tx_frame_buffer,
|
|
E_MMAC_TX_START_NOW |
|
|
(send_on_cca ? E_MMAC_TX_USE_CCA : E_MMAC_TX_NO_CCA));
|
|
#endif
|
|
if(poll_mode) {
|
|
BUSYWAIT_UNTIL(u32MMAC_PollInterruptSource(E_MMAC_INT_TX_COMPLETE), MAX_PACKET_DURATION);
|
|
} else {
|
|
if(in_ack_transmission) {
|
|
/* as nested interupts are not possible, the tx flag will never be cleared */
|
|
BUSYWAIT_UNTIL(FALSE, MAX_ACK_DURATION);
|
|
} else {
|
|
/* wait until the tx flag is cleared */
|
|
BUSYWAIT_UNTIL(!tx_in_progress, MAX_PACKET_DURATION);
|
|
}
|
|
}
|
|
|
|
/* Energest */
|
|
ENERGEST_OFF(ENERGEST_TYPE_TRANSMIT);
|
|
if(listen_on) {
|
|
ENERGEST_ON(ENERGEST_TYPE_LISTEN);
|
|
}
|
|
tx_in_progress = 0;
|
|
|
|
/* Check error code */
|
|
int ret;
|
|
uint32_t tx_error = u32MMAC_GetTxErrors();
|
|
if(tx_error == 0) {
|
|
ret = RADIO_TX_OK;
|
|
} else if(tx_error & E_MMAC_TXSTAT_ABORTED) {
|
|
ret = RADIO_TX_ERR;
|
|
} else if(tx_error & E_MMAC_TXSTAT_CCA_BUSY) {
|
|
ret = RADIO_TX_COLLISION;
|
|
} else if(tx_error & E_MMAC_TXSTAT_NO_ACK) {
|
|
ret = RADIO_TX_NOACK;
|
|
} else {
|
|
ret = RADIO_TX_ERR;
|
|
}
|
|
return ret;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
prepare(const void *payload, unsigned short payload_len)
|
|
{
|
|
#if !MICROMAC_RADIO_MAC
|
|
uint8_t i;
|
|
uint16_t checksum;
|
|
#endif
|
|
|
|
if(tx_in_progress) {
|
|
return 1;
|
|
}
|
|
if(payload_len > 127 || payload == NULL) {
|
|
return 1;
|
|
}
|
|
#if MICROMAC_RADIO_MAC
|
|
frame802154_t info154;
|
|
int hdr_len = frame802154_parse((unsigned char *)payload, payload_len, &info154);
|
|
//TODO: hdr_len contains security header, which are not managed by micromac
|
|
tx_frame_buffer.u8PayloadLength = payload_len - hdr_len;
|
|
tx_frame_buffer.u8SequenceNum = info154.seq;
|
|
tx_frame_buffer.u16FCF = ((uint8_t*)payload)[0] | (((uint8_t*)payload)[1] << 8);
|
|
tx_frame_buffer.u16DestPAN = info154.dest_pid;
|
|
tx_frame_buffer.u16SrcPAN = info154.src_pid;
|
|
if(info154.fcf.dest_addr_mode == FRAME802154_SHORTADDRMODE) {
|
|
tx_frame_buffer.uDestAddr.u16Short = info154.dest_addr[0] | (info154.dest_addr[0] << 8);
|
|
} else if(info154.fcf.dest_addr_mode == FRAME802154_LONGADDRMODE) {
|
|
tx_frame_buffer.uDestAddr.sExt.u32L = *(uint32_t*)(&info154.dest_addr[4]);
|
|
tx_frame_buffer.uDestAddr.sExt.u32H = *(uint32_t*)(&info154.dest_addr[0]);
|
|
}
|
|
if(info154.fcf.src_addr_mode == FRAME802154_SHORTADDRMODE) {
|
|
tx_frame_buffer.uSrcAddr.u16Short = info154.src_addr[0] | (info154.src_addr[0] << 8);
|
|
} else if(info154.fcf.src_addr_mode == FRAME802154_LONGADDRMODE) {
|
|
tx_frame_buffer.uSrcAddr.sExt.u32L = *(uint32_t*)(&info154.src_addr[4]);
|
|
tx_frame_buffer.uSrcAddr.sExt.u32H = *(uint32_t*)(&info154.src_addr[0]);
|
|
}
|
|
tx_frame_buffer.u16FCS = crc16_data(payload, payload_len, 0);
|
|
memcpy(tx_frame_buffer.uPayload.au8Byte, info154.payload, info154.payload_len);
|
|
#else
|
|
/* Copy payload to (soft) Ttx buffer */
|
|
memcpy(tx_frame_buffer.uPayload.au8Byte, payload, payload_len);
|
|
i = payload_len;
|
|
#if CRC_SW
|
|
/* Compute CRC */
|
|
checksum = crc16_data(payload, payload_len, 0);
|
|
tx_frame_buffer.uPayload.au8Byte[i++] = checksum;
|
|
tx_frame_buffer.uPayload.au8Byte[i++] = (checksum >> 8) & 0xff;
|
|
tx_frame_buffer.u8PayloadLength = payload_len + CHECKSUM_LEN;
|
|
#else
|
|
tx_frame_buffer.u8PayloadLength = payload_len;
|
|
#endif
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
send(const void *payload, unsigned short payload_len)
|
|
{
|
|
if(prepare(payload, payload_len) == 0) {
|
|
return transmit(payload_len);
|
|
} else {
|
|
return RADIO_TX_ERR;
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
int
|
|
get_channel(void)
|
|
{
|
|
return current_channel;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
void
|
|
set_channel(int c)
|
|
{
|
|
current_channel = c;
|
|
/* will fine tune TX power as well */
|
|
vMMAC_SetChannel(current_channel);
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
#if !MICROMAC_RADIO_MAC
|
|
static int
|
|
is_broadcast_addr(uint8_t mode, uint8_t *addr)
|
|
{
|
|
int i = ((mode == FRAME802154_SHORTADDRMODE) ? 2 : 8);
|
|
while(i-- > 0) {
|
|
if(addr[i] != 0xff) {
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
/* Send an ACK */
|
|
static void
|
|
send_ack(const frame802154_t *frame)
|
|
{
|
|
uint8_t buffer[3];
|
|
/* FCF: 2 octets */
|
|
buffer[0] = FRAME802154_ACKFRAME;
|
|
buffer[1] = 0;
|
|
/* Seqnum: 1 octets */
|
|
buffer[2] = frame->seq;
|
|
in_ack_transmission = 1;
|
|
send(&buffer, sizeof(buffer));
|
|
in_ack_transmission = 0;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
/* Check if a packet is for us */
|
|
static int
|
|
is_packet_for_us(uint8_t *buf, int len, int do_send_ack)
|
|
{
|
|
frame802154_t frame;
|
|
int result;
|
|
uint8_t parsed = frame802154_parse(buf, len, &frame);
|
|
if(parsed) {
|
|
if(frame.fcf.dest_addr_mode) {
|
|
int has_dest_panid;
|
|
frame802154_has_panid(&frame.fcf, NULL, &has_dest_panid);
|
|
if(has_dest_panid
|
|
&& frame802154_get_pan_id() != FRAME802154_BROADCASTPANDID
|
|
&& frame.dest_pid != frame802154_get_pan_id()
|
|
&& frame.dest_pid != FRAME802154_BROADCASTPANDID) {
|
|
/* Packet to another PAN */
|
|
return 0;
|
|
}
|
|
if(!is_broadcast_addr(frame.fcf.dest_addr_mode, frame.dest_addr)) {
|
|
result = linkaddr_cmp((linkaddr_t *)frame.dest_addr, &linkaddr_node_addr);
|
|
if(autoack_enabled && result && do_send_ack) {
|
|
/* this is a unicast frame and sending ACKs is enabled */
|
|
send_ack(&frame);
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
#endif
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
read(void *buf, unsigned short bufsize)
|
|
{
|
|
int len = 0;
|
|
#if MICROMAC_RADIO_MAC
|
|
frame802154_fcf_t fcf;
|
|
uint8_t *p = (uint8_t*)buf;
|
|
int has_src_panid;
|
|
int has_dest_panid;
|
|
int c;
|
|
|
|
p[len++] = input_frame_buffer->u16FCF & 0xff;
|
|
p[len++] = (input_frame_buffer->u16FCF >> 8) & 0xff;
|
|
frame802154_parse_fcf(p, &fcf);
|
|
p[len++] = input_frame_buffer->u8SequenceNum;
|
|
frame802154_has_panid(&fcf, &has_src_panid, &has_dest_panid);
|
|
if(has_dest_panid) {
|
|
p[len++] = input_frame_buffer->u16DestPAN & 0xff;
|
|
p[len++] = (input_frame_buffer->u16DestPAN >> 8) & 0xff;
|
|
}
|
|
if(fcf.dest_addr_mode == FRAME802154_SHORTADDRMODE) {
|
|
p[len++] = input_frame_buffer->uDestAddr.u16Short & 0xff;
|
|
p[len++] = (input_frame_buffer->uDestAddr.u16Short >> 8) & 0xff;
|
|
} else if(fcf.dest_addr_mode == FRAME802154_LONGADDRMODE) {
|
|
for(c = 0; c < 4; c++) {
|
|
p[len + c] = ((uint8_t*)(&input_frame_buffer->uDestAddr.sExt.u32L))[3 - c];
|
|
}
|
|
for(c = 0; c < 4; c++) {
|
|
p[len + c + 4] = ((uint8_t*)(&input_frame_buffer->uDestAddr.sExt.u32H))[3 - c];
|
|
}
|
|
len += 8;
|
|
}
|
|
if(has_src_panid) {
|
|
p[len++] = input_frame_buffer->u16SrcPAN & 0xff;
|
|
p[len++] = (input_frame_buffer->u16SrcPAN >> 8) & 0xff;
|
|
}
|
|
if(fcf.src_addr_mode == FRAME802154_SHORTADDRMODE) {
|
|
p[len++] = input_frame_buffer->uSrcAddr.u16Short & 0xff;
|
|
p[len++] = (input_frame_buffer->uSrcAddr.u16Short >> 8) & 0xff;
|
|
} else if(fcf.src_addr_mode == FRAME802154_LONGADDRMODE) {
|
|
for(c = 0; c < 4; c++) {
|
|
p[len + c] = ((uint8_t*)(&input_frame_buffer->uSrcAddr.sExt.u32L))[3 - c];
|
|
}
|
|
for(c = 0; c < 4; c++) {
|
|
p[len + c + 4] = ((uint8_t*)(&input_frame_buffer->uSrcAddr.sExt.u32H))[3 - c];
|
|
}
|
|
len += 8;
|
|
}
|
|
memcpy(&p[len], input_frame_buffer->uPayload.au8Byte, input_frame_buffer->u8PayloadLength);
|
|
len += input_frame_buffer->u8PayloadLength;
|
|
#else
|
|
uint16_t radio_last_rx_crc;
|
|
uint8_t radio_last_rx_crc_ok = 1;
|
|
|
|
len = input_frame_buffer->u8PayloadLength;
|
|
|
|
if(len <= CHECKSUM_LEN) {
|
|
input_frame_buffer->u8PayloadLength = 0;
|
|
return 0;
|
|
} else {
|
|
len -= CHECKSUM_LEN;
|
|
/* Check CRC */
|
|
#if CRC_SW
|
|
uint16_t checksum = crc16_data(input_frame_buffer->uPayload.au8Byte, len, 0);
|
|
radio_last_rx_crc =
|
|
(uint16_t)(input_frame_buffer->uPayload.au8Byte[len + 1] << (uint16_t)8)
|
|
| input_frame_buffer->uPayload.au8Byte[len];
|
|
radio_last_rx_crc_ok = (checksum == radio_last_rx_crc);
|
|
if(!radio_last_rx_crc_ok) {
|
|
}
|
|
#endif /* CRC_SW */
|
|
if(radio_last_rx_crc_ok) {
|
|
/* If we are in poll mode we need to check the frame here */
|
|
if(poll_mode) {
|
|
if(frame_filtering &&
|
|
!is_packet_for_us(input_frame_buffer->uPayload.au8Byte, len, 0)) {
|
|
len = 0;
|
|
} else {
|
|
read_last_rssi();
|
|
}
|
|
}
|
|
if(len != 0) {
|
|
bufsize = MIN(len, bufsize);
|
|
memcpy(buf, input_frame_buffer->uPayload.au8Byte, bufsize);
|
|
if(!poll_mode) {
|
|
/* Not in poll mode: packetbuf should not be accessed in interrupt context */
|
|
packetbuf_set_attr(PACKETBUF_ATTR_RSSI, radio_last_rssi);
|
|
packetbuf_set_attr(PACKETBUF_ATTR_LINK_QUALITY, radio_last_correlation);
|
|
}
|
|
}
|
|
} else {
|
|
len = 0;
|
|
}
|
|
/* Disable further read attempts */
|
|
input_frame_buffer->u8PayloadLength = 0;
|
|
}
|
|
#endif
|
|
return len;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static void
|
|
set_txpower(int8_t power)
|
|
{
|
|
if(power > OUTPUT_POWER_MAX) {
|
|
current_tx_power = OUTPUT_POWER_MAX;
|
|
} else {
|
|
if(power < OUTPUT_POWER_MIN) {
|
|
current_tx_power = OUTPUT_POWER_MIN;
|
|
} else {
|
|
current_tx_power = power;
|
|
}
|
|
}
|
|
vMMAC_SetChannelAndPower(current_channel, current_tx_power);
|
|
}
|
|
/*--------------------------------------------------------------------------*/
|
|
static int
|
|
get_txpower(void)
|
|
{
|
|
int actual_tx_power;
|
|
#if (JENNIC_CHIP == JN5169)
|
|
/* Actual tx power value rounded to nearest integer number */
|
|
const static int8 power_table [] = {
|
|
-32, -30, -29, -29, /* -32 .. -29 */
|
|
-28, -28, -28, -28, /* -28 .. -25 */
|
|
-21, -21, -21, -2, /* -24 .. -21 */
|
|
-20, -19, -18, -17, /* -20 .. -17 */
|
|
-17, -17, -17, -10, /* -16 .. -13 */
|
|
-10, -10, -10, -9, /* -12 .. -09 */
|
|
-8, -7, -6, -6, /* -08 .. -05 */
|
|
-6, -6, 1, 1, /* -04 .. -01 */
|
|
1, 1, 2, 3, /* 00 .. 03 */
|
|
4, 5, 6, 7, /* 04 .. 07 */
|
|
9, 9, 10 }; /* 08 .. 10 */
|
|
if(current_tx_power > OUTPUT_POWER_MAX) {
|
|
actual_tx_power = OUTPUT_POWER_MAX;
|
|
} else if(current_tx_power < OUTPUT_POWER_MIN) {
|
|
actual_tx_power = OUTPUT_POWER_MIN;
|
|
} else {
|
|
actual_tx_power = power_table[current_tx_power + ABS_OUTPUT_POWER_MIN];
|
|
}
|
|
#else
|
|
/* Other JN516x chips */
|
|
if(current_tx_power < (-24)) {
|
|
actual_tx_power = OUTPUT_POWER_MIN;
|
|
} else if(current_tx_power < (-12)) {
|
|
actual_tx_power = (-20);
|
|
} else if(current_tx_power < 0) {
|
|
actual_tx_power = (-9);
|
|
} else {
|
|
actual_tx_power = OUTPUT_POWER_MAX;
|
|
}
|
|
#endif
|
|
return (int)actual_tx_power;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
get_detected_energy(void)
|
|
{
|
|
const uint32 u32Samples = 8;
|
|
return u8JPT_EnergyDetect(current_channel, u32Samples);
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
get_rssi(void)
|
|
{
|
|
/* this approximate formula for RSSI is taken from NXP internal docs */
|
|
return (7 * get_detected_energy() - 1970) / 20;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static void
|
|
read_last_rssi(void)
|
|
{
|
|
uint8_t radio_last_rx_energy;
|
|
radio_last_rx_energy = u8MMAC_GetRxLqi((uint8_t *)&radio_last_correlation);
|
|
radio_last_rssi = i16JPT_ConvertEnergyTodBm(radio_last_rx_energy);
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
int
|
|
receiving_packet(void)
|
|
{
|
|
return bMMAC_RxDetected();
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
pending_packet(void)
|
|
{
|
|
if(!poll_mode) {
|
|
return ringbufindex_peek_get(&input_ringbuf) != -1;
|
|
} else {
|
|
return u32MMAC_PollInterruptSource(
|
|
E_MMAC_INT_RX_COMPLETE | E_MMAC_INT_RX_HEADER);
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static int
|
|
cca(void)
|
|
{
|
|
bool_t is_channel_busy = bJPT_CCA(current_channel,
|
|
E_JPT_CCA_MODE_CARRIER_OR_ENERGY,
|
|
cca_thershold);
|
|
return is_channel_busy == FALSE;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static void
|
|
radio_interrupt_handler(uint32 mac_event)
|
|
{
|
|
uint32_t rx_status;
|
|
uint8_t overflow = 0;
|
|
int get_index;
|
|
int put_index;
|
|
#if !MICROMAC_RADIO_MAC
|
|
int packet_for_me = 0;
|
|
#endif
|
|
|
|
if(mac_event & E_MMAC_INT_TX_COMPLETE) {
|
|
/* Transmission attempt has finished */
|
|
tx_in_progress = 0;
|
|
} else if(mac_event & E_MMAC_INT_RX_COMPLETE) {
|
|
rx_status = u32MMAC_GetRxErrors();
|
|
/* If rx is successful */
|
|
if(rx_status == 0) {
|
|
/* Save SFD timestamp */
|
|
last_packet_timestamp = get_packet_timestamp();
|
|
|
|
if(!poll_mode && (mac_event & E_MMAC_INT_RX_COMPLETE)) {
|
|
#if MICROMAC_RADIO_MAC
|
|
/* read and cache RSSI and LQI values */
|
|
read_last_rssi();
|
|
/* Put received frame in queue */
|
|
ringbufindex_put(&input_ringbuf);
|
|
|
|
if((get_index = ringbufindex_peek_get(&input_ringbuf)) != -1) {
|
|
input_frame_buffer = &input_array[get_index];
|
|
}
|
|
process_poll(µmac_radio_process);
|
|
|
|
/* get pointer to next input slot */
|
|
put_index = ringbufindex_peek_put(&input_ringbuf);
|
|
/* is there space? */
|
|
if(put_index != -1) {
|
|
/* move rx_frame_buffer to next empty slot */
|
|
rx_frame_buffer = &input_array[put_index];
|
|
} else {
|
|
overflow = 1;
|
|
rx_frame_buffer = NULL;
|
|
}
|
|
#else
|
|
if(rx_frame_buffer->u8PayloadLength > CHECKSUM_LEN) {
|
|
if(frame_filtering) {
|
|
/* Check RX address */
|
|
packet_for_me = is_packet_for_us(rx_frame_buffer->uPayload.au8Byte, rx_frame_buffer->u8PayloadLength - CHECKSUM_LEN, 1);
|
|
} else if(!frame_filtering) {
|
|
packet_for_me = 1;
|
|
}
|
|
}
|
|
if(!packet_for_me) {
|
|
/* Prevent reading */
|
|
rx_frame_buffer->u8PayloadLength = 0;
|
|
} else {
|
|
/* read and cache RSSI and LQI values */
|
|
read_last_rssi();
|
|
/* Put received frame in queue */
|
|
ringbufindex_put(&input_ringbuf);
|
|
|
|
if((get_index = ringbufindex_peek_get(&input_ringbuf)) != -1) {
|
|
input_frame_buffer = &input_array[get_index];
|
|
}
|
|
process_poll(µmac_radio_process);
|
|
|
|
/* get pointer to next input slot */
|
|
put_index = ringbufindex_peek_put(&input_ringbuf);
|
|
/* is there space? */
|
|
if(put_index != -1) {
|
|
/* move rx_frame_buffer to next empty slot */
|
|
rx_frame_buffer = &input_array[put_index];
|
|
} else {
|
|
overflow = 1;
|
|
rx_frame_buffer = NULL;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
if(overflow) {
|
|
off();
|
|
} else if(MICROMAC_CONF_ALWAYS_ON
|
|
&& (mac_event & (E_MMAC_INT_TX_COMPLETE | E_MMAC_INT_RX_COMPLETE))) {
|
|
on();
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
PROCESS_THREAD(micromac_radio_process, ev, data)
|
|
{
|
|
PROCESS_BEGIN();
|
|
|
|
while(1) {
|
|
PROCESS_YIELD_UNTIL(ev == PROCESS_EVENT_POLL);
|
|
|
|
/* Pass received packets to upper layer */
|
|
int16_t read_index;
|
|
/* Loop on accessing (without removing) a pending input packet */
|
|
while((read_index = ringbufindex_peek_get(&input_ringbuf)) != -1) {
|
|
input_frame_buffer = &input_array[read_index];
|
|
/* Put packet into packetbuf for input callback */
|
|
packetbuf_clear();
|
|
int len = read(packetbuf_dataptr(), PACKETBUF_SIZE);
|
|
/* is packet valid? */
|
|
if(len > 0) {
|
|
packetbuf_set_datalen(len);
|
|
NETSTACK_MAC.input();
|
|
}
|
|
/* Remove packet from ringbuf */
|
|
ringbufindex_get(&input_ringbuf);
|
|
/* Disable further read attempts */
|
|
input_frame_buffer->u8PayloadLength = 0;
|
|
}
|
|
|
|
/* Are we recovering from overflow? */
|
|
if(rx_frame_buffer == NULL) {
|
|
/* get pointer to next input slot */
|
|
int put_index = ringbufindex_peek_put(&input_ringbuf);
|
|
/* is there space? */
|
|
if(put_index != -1) {
|
|
/* move rx_frame_buffer to next empty slot */
|
|
rx_frame_buffer = &input_array[put_index];
|
|
/* do we need to turn radio on? */
|
|
if(MICROMAC_CONF_ALWAYS_ON || missed_radio_on_request) {
|
|
missed_radio_on_request = 0;
|
|
on();
|
|
}
|
|
} else {
|
|
rx_frame_buffer = NULL;
|
|
}
|
|
}
|
|
}
|
|
PROCESS_END();
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static void
|
|
set_frame_filtering(uint8_t enable)
|
|
{
|
|
frame_filtering = enable;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static void
|
|
set_autoack(uint8_t enable)
|
|
{
|
|
autoack_enabled = enable;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static void
|
|
set_poll_mode(uint8_t enable)
|
|
{
|
|
poll_mode = enable;
|
|
if(poll_mode) {
|
|
/* Disable interrupts */
|
|
vMMAC_EnableInterrupts(NULL);
|
|
vMMAC_ConfigureInterruptSources(0);
|
|
} else {
|
|
/* Initialize and enable interrupts */
|
|
/* TODO: enable E_MMAC_INT_RX_HEADER & filter out frames after header rx */
|
|
vMMAC_ConfigureInterruptSources(
|
|
E_MMAC_INT_RX_COMPLETE | E_MMAC_INT_TX_COMPLETE);
|
|
vMMAC_EnableInterrupts(&radio_interrupt_handler);
|
|
}
|
|
}
|
|
/* Enable or disable CCA before sending */
|
|
static void
|
|
set_send_on_cca(uint8_t enable)
|
|
{
|
|
send_on_cca = enable;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static radio_result_t
|
|
get_value(radio_param_t param, radio_value_t *value)
|
|
{
|
|
if(!value) {
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
}
|
|
switch(param) {
|
|
case RADIO_PARAM_POWER_MODE:
|
|
*value = listen_on || tx_in_progress ? RADIO_POWER_MODE_ON : RADIO_POWER_MODE_OFF;
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_CHANNEL:
|
|
*value = get_channel();
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_RX_MODE:
|
|
*value = 0;
|
|
if(frame_filtering) {
|
|
*value |= RADIO_RX_MODE_ADDRESS_FILTER;
|
|
}
|
|
if(autoack_enabled) {
|
|
*value |= RADIO_RX_MODE_AUTOACK;
|
|
}
|
|
if(poll_mode) {
|
|
*value |= RADIO_RX_MODE_POLL_MODE;
|
|
}
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_TX_MODE:
|
|
*value = 0;
|
|
if(send_on_cca) {
|
|
*value |= RADIO_TX_MODE_SEND_ON_CCA;
|
|
}
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_TXPOWER:
|
|
*value = get_txpower();
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_RSSI:
|
|
*value = get_rssi();
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_LAST_RSSI:
|
|
*value = radio_last_rssi;
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_CCA_THRESHOLD:
|
|
*value = cca_thershold;
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_CONST_CHANNEL_MIN:
|
|
*value = 11;
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_CONST_CHANNEL_MAX:
|
|
*value = 26;
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_CONST_TXPOWER_MIN:
|
|
*value = OUTPUT_POWER_MIN;
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_CONST_TXPOWER_MAX:
|
|
*value = OUTPUT_POWER_MAX;
|
|
return RADIO_RESULT_OK;
|
|
default:
|
|
return RADIO_RESULT_NOT_SUPPORTED;
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static radio_result_t
|
|
set_value(radio_param_t param, radio_value_t value)
|
|
{
|
|
switch(param) {
|
|
case RADIO_PARAM_POWER_MODE:
|
|
if(value == RADIO_POWER_MODE_ON) {
|
|
on();
|
|
return RADIO_RESULT_OK;
|
|
}
|
|
if(value == RADIO_POWER_MODE_OFF) {
|
|
off();
|
|
return RADIO_RESULT_OK;
|
|
}
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
case RADIO_PARAM_CHANNEL:
|
|
if(value < 11 || value > 26) {
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
}
|
|
set_channel(value);
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_RX_MODE:
|
|
if(value & ~(RADIO_RX_MODE_ADDRESS_FILTER |
|
|
RADIO_RX_MODE_AUTOACK | RADIO_RX_MODE_POLL_MODE)) {
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
}
|
|
set_frame_filtering((value & RADIO_RX_MODE_ADDRESS_FILTER) != 0);
|
|
set_autoack((value & RADIO_RX_MODE_AUTOACK) != 0);
|
|
set_poll_mode((value & RADIO_RX_MODE_POLL_MODE) != 0);
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_TX_MODE:
|
|
if(value & ~(RADIO_TX_MODE_SEND_ON_CCA)) {
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
}
|
|
set_send_on_cca((value & RADIO_TX_MODE_SEND_ON_CCA) != 0);
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_TXPOWER:
|
|
if(value < OUTPUT_POWER_MIN || value > OUTPUT_POWER_MAX) {
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
/* Find the closest higher PA_LEVEL for the desired output power */
|
|
}
|
|
set_txpower(value);
|
|
return RADIO_RESULT_OK;
|
|
case RADIO_PARAM_CCA_THRESHOLD:
|
|
cca_thershold = value;
|
|
return RADIO_RESULT_OK;
|
|
default:
|
|
return RADIO_RESULT_NOT_SUPPORTED;
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static radio_result_t
|
|
get_object(radio_param_t param, void *dest, size_t size)
|
|
{
|
|
if(param == RADIO_PARAM_LAST_PACKET_TIMESTAMP) {
|
|
if(size != sizeof(rtimer_clock_t) || !dest) {
|
|
return RADIO_RESULT_INVALID_VALUE;
|
|
}
|
|
*(rtimer_clock_t *)dest = get_packet_timestamp();
|
|
|
|
return RADIO_RESULT_OK;
|
|
}
|
|
return RADIO_RESULT_NOT_SUPPORTED;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
static radio_result_t
|
|
set_object(radio_param_t param, const void *src, size_t size)
|
|
{
|
|
return RADIO_RESULT_NOT_SUPPORTED;
|
|
}
|
|
/*---------------------------------------------------------------------------*/
|
|
const struct radio_driver micromac_radio_driver = {
|
|
init,
|
|
prepare,
|
|
transmit,
|
|
send,
|
|
read,
|
|
cca,
|
|
receiving_packet,
|
|
pending_packet,
|
|
on,
|
|
off,
|
|
get_value,
|
|
set_value,
|
|
get_object,
|
|
set_object
|
|
};
|