mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 20:32:43 +01:00
623 lines
16 KiB
C
623 lines
16 KiB
C
/*
|
|
* 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 <m.lenders@fu-berlin.de>
|
|
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <err.h>
|
|
#include <errno.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/time.h>
|
|
|
|
#include "async_read.h"
|
|
#include "byteorder.h"
|
|
#include "checksum/ucrc16.h"
|
|
#include "native_internal.h"
|
|
|
|
#include "net/ieee802154/radio.h"
|
|
#include "socket_zep.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
#define _UNIX_NTP_ERA_OFFSET (2208988800U)
|
|
/* can't use timex.h's US_PER_SEC as timeval's tv_usec is signed long
|
|
* (https://pubs.opengroup.org/onlinepubs/9699919799.2016edition/basedefs/time.h.html) */
|
|
#define TV_USEC_PER_SEC (1000000L)
|
|
|
|
/* dummy packet to register with ZEP dispatcher */
|
|
#define SOCKET_ZEP_V2_TYPE_HELLO (255)
|
|
|
|
static size_t _zep_hdr_fill_v2_data(socket_zep_t *dev, zep_v2_data_hdr_t *hdr,
|
|
size_t payload_len)
|
|
{
|
|
struct timeval tv;
|
|
|
|
real_gettimeofday(&tv, NULL);
|
|
hdr->hdr.version = 2;
|
|
hdr->type = ZEP_V2_TYPE_DATA;
|
|
hdr->chan = dev->chan;
|
|
hdr->dev = byteorder_htons((uint16_t)((((intptr_t)dev)) & 0xffff));
|
|
hdr->lqi_mode = 1;
|
|
hdr->lqi_val = 0xff; /* set by ZEP dispatcher */
|
|
hdr->time.seconds = byteorder_htonl(tv.tv_sec + _UNIX_NTP_ERA_OFFSET);
|
|
assert(tv.tv_usec < TV_USEC_PER_SEC);
|
|
hdr->time.fraction = byteorder_htonl(
|
|
(uint32_t)((uint64_t)tv.tv_usec * TV_USEC_PER_SEC) / 232U
|
|
);
|
|
hdr->seq = byteorder_htonl(dev->seq);
|
|
memset(hdr->resv, 0, sizeof(hdr->resv));
|
|
hdr->length = payload_len;
|
|
|
|
return sizeof(zep_v2_data_hdr_t);
|
|
}
|
|
|
|
static inline size_t _zep_hdr_fill(socket_zep_t *dev, zep_hdr_t *hdr,
|
|
size_t payload_len)
|
|
{
|
|
hdr->preamble[0] = 'E';
|
|
hdr->preamble[1] = 'X';
|
|
|
|
/* keep possibility for ZEPv1 open */
|
|
return _zep_hdr_fill_v2_data(dev, (zep_v2_data_hdr_t *)hdr,
|
|
payload_len);
|
|
}
|
|
|
|
static void _continue_reading(socket_zep_t *dev)
|
|
{
|
|
/* work around lost signals */
|
|
fd_set rfds;
|
|
struct timeval t;
|
|
memset(&t, 0, sizeof(t));
|
|
FD_ZERO(&rfds);
|
|
FD_SET(dev->sock_fd, &rfds);
|
|
|
|
_native_in_syscall++; /* no switching here */
|
|
|
|
if (real_select(dev->sock_fd + 1, &rfds, NULL, NULL, &t) == 1) {
|
|
int sig = SIGIO;
|
|
extern int _sig_pipefd[2];
|
|
real_write(_sig_pipefd[1], &sig, sizeof(sig));
|
|
_native_sigpend++;
|
|
}
|
|
else {
|
|
native_async_read_continue(dev->sock_fd);
|
|
}
|
|
|
|
_native_in_syscall--;
|
|
}
|
|
|
|
static inline bool _dst_not_me(socket_zep_t *dev, const void *buf)
|
|
{
|
|
uint8_t dst_addr[IEEE802154_LONG_ADDRESS_LEN] = { 0 };
|
|
int dst_len;
|
|
le_uint16_t dst_pan = { .u16 = 0 };
|
|
bool is_ack = *(uint8_t *)buf & IEEE802154_FCF_TYPE_ACK;
|
|
|
|
/* no need to check address if we are in promiscuous mode */
|
|
if (dev->filter_mode == IEEE802154_FILTER_PROMISC ||
|
|
dev->filter_mode == IEEE802154_FILTER_SNIFFER) {
|
|
return false;
|
|
}
|
|
|
|
/* ignore everything but ACK frames */
|
|
if ((dev->filter_mode == IEEE802154_FILTER_ACK_ONLY) && !is_ack) {
|
|
DEBUG("socket_zep::dst_not_me: ignoring non-ACK frame\n");
|
|
return true;
|
|
}
|
|
|
|
/* ACKs carry no address */
|
|
if (is_ack) {
|
|
DEBUG("socket_zep::dst_not_me: got ACK\n");
|
|
return false;
|
|
}
|
|
|
|
dst_len = ieee802154_get_dst(buf, dst_addr, &dst_pan);
|
|
|
|
if (dst_pan.u16 != dev->pan_id) {
|
|
DEBUG("socket_zep::dst_not_me: PAN ID %x != %x\n", dst_pan.u16, dev->pan_id);
|
|
return true;
|
|
}
|
|
|
|
switch (dst_len) {
|
|
case IEEE802154_LONG_ADDRESS_LEN:
|
|
return memcmp(dst_addr, dev->addr_long, dst_len);
|
|
case IEEE802154_SHORT_ADDRESS_LEN:
|
|
return memcmp(dst_addr, ieee802154_addr_bcast, dst_len) &&
|
|
memcmp(dst_addr, dev->addr_short, dst_len);
|
|
default:
|
|
return false; /* better safe than sorry ;-) */
|
|
}
|
|
}
|
|
|
|
static int _bind_local(const socket_zep_params_t *params)
|
|
{
|
|
int res;
|
|
static const struct addrinfo hints = { .ai_family = AF_UNSPEC,
|
|
.ai_socktype = SOCK_DGRAM };
|
|
struct addrinfo *ai = NULL;
|
|
|
|
if (params->local_addr == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
/* bind and connect socket */
|
|
if ((res = real_getaddrinfo(params->local_addr, params->local_port, &hints,
|
|
&ai)) < 0) {
|
|
errx(EXIT_FAILURE, "ZEP: unable to get local address: %s\n",
|
|
gai_strerror(res));
|
|
}
|
|
|
|
for (struct addrinfo *local = ai; local != NULL; local = local->ai_next) {
|
|
if ((res = real_socket(local->ai_family, local->ai_socktype,
|
|
local->ai_protocol)) < 0) {
|
|
continue;
|
|
}
|
|
if (real_bind(res, local->ai_addr, local->ai_addrlen) == 0) {
|
|
break; /* successfully bound */
|
|
}
|
|
}
|
|
real_freeaddrinfo(ai);
|
|
|
|
if (res < 0) {
|
|
err(EXIT_FAILURE, "ZEP: Unable to bind socket");
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static int _connect_remote(socket_zep_t *dev, const socket_zep_params_t *params)
|
|
{
|
|
int res;
|
|
static const struct addrinfo hints = { .ai_family = AF_UNSPEC,
|
|
.ai_socktype = SOCK_DGRAM };
|
|
struct addrinfo *ai = NULL, *remote;
|
|
|
|
if (params->remote_addr == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
if ((res = real_getaddrinfo(params->remote_addr, params->remote_port, &hints,
|
|
&ai)) < 0) {
|
|
errx(EXIT_FAILURE, "ZEP: unable to get remote address: %s\n",
|
|
gai_strerror(res));
|
|
}
|
|
|
|
for (remote = ai; remote != NULL; remote = remote->ai_next) {
|
|
if (real_connect(dev->sock_fd, remote->ai_addr, remote->ai_addrlen) == 0) {
|
|
break; /* successfully connected */
|
|
}
|
|
}
|
|
|
|
if (remote == NULL) {
|
|
err(EXIT_FAILURE, "ZEP: Unable to connect socket");
|
|
}
|
|
|
|
real_freeaddrinfo(ai);
|
|
|
|
return res;
|
|
}
|
|
|
|
static void _send_zep_hello(socket_zep_t *dev)
|
|
{
|
|
if (IS_USED(MODULE_SOCKET_ZEP_HELLO) && dev->send_hello) {
|
|
/* dummy packet */
|
|
zep_v2_data_hdr_t hdr = {
|
|
.hdr.preamble = "EX",
|
|
.hdr.version = 2,
|
|
.type = SOCKET_ZEP_V2_TYPE_HELLO,
|
|
.resv = "HELLO",
|
|
.length = sizeof(dev->addr_long),
|
|
};
|
|
|
|
/* append HW addr */
|
|
real_send(dev->sock_fd, &hdr, sizeof(hdr), MSG_MORE);
|
|
real_send(dev->sock_fd, dev->addr_long, sizeof(dev->addr_long), 0);
|
|
}
|
|
}
|
|
|
|
static void _socket_isr(int fd, void *arg)
|
|
{
|
|
ieee802154_dev_t *dev = arg;
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
DEBUG("socket_zep::_socket_isr: bytes on %d\n", fd);
|
|
|
|
if (zepdev->state == IEEE802154_TRX_STATE_RX_ON) {
|
|
dev->cb(dev, IEEE802154_RADIO_INDICATION_RX_DONE);
|
|
} else {
|
|
/* discard frame */
|
|
uint8_t tmp;
|
|
real_read(fd, &tmp, sizeof(tmp));
|
|
_continue_reading(zepdev);
|
|
}
|
|
}
|
|
|
|
void socket_zep_setup(socket_zep_t *dev, const socket_zep_params_t *params)
|
|
{
|
|
DEBUG("socket_zep_setup(%p, %p)\n", (void *)dev, (void *)params);
|
|
assert((params->remote_addr != NULL) && (params->remote_port != NULL));
|
|
|
|
dev->params = params;
|
|
native_async_read_setup();
|
|
}
|
|
|
|
void socket_zep_cleanup(socket_zep_t *dev)
|
|
{
|
|
assert(dev != NULL);
|
|
/* cleanup signal handling */
|
|
native_async_read_cleanup();
|
|
/* close the socket */
|
|
close(dev->sock_fd);
|
|
dev->sock_fd = 0;
|
|
}
|
|
|
|
static int _request_on(ieee802154_dev_t *dev)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
DEBUG("socket_zep::request_on()\n");
|
|
|
|
int res = _bind_local(zepdev->params);
|
|
|
|
if (res < 0) {
|
|
zepdev->sock_fd = socket(AF_INET6, SOCK_DGRAM, 0);
|
|
res = zepdev->sock_fd;
|
|
} else {
|
|
zepdev->sock_fd = res;
|
|
}
|
|
|
|
if (res < 0) {
|
|
return res;
|
|
}
|
|
|
|
native_async_read_add_handler(zepdev->sock_fd, dev, _socket_isr);
|
|
|
|
/* only send hello if we are connected to a remote */
|
|
zepdev->send_hello = !_connect_remote(zepdev, zepdev->params);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _confirm_on(ieee802154_dev_t *dev)
|
|
{
|
|
(void) dev;
|
|
return 0;
|
|
}
|
|
|
|
static int _off(ieee802154_dev_t *dev)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
DEBUG("socket_zep::off()\n");
|
|
|
|
close(zepdev->sock_fd);
|
|
zepdev->sock_fd = -1;
|
|
return 0;
|
|
}
|
|
|
|
static int _set_cca_mode(ieee802154_dev_t *dev, ieee802154_cca_mode_t mode)
|
|
{
|
|
(void) dev;
|
|
(void) mode;
|
|
return 0;
|
|
}
|
|
|
|
static int _request_cca(ieee802154_dev_t *dev)
|
|
{
|
|
(void) dev;
|
|
return 0;
|
|
}
|
|
|
|
static int _confirm_cca(ieee802154_dev_t *dev)
|
|
{
|
|
(void) dev;
|
|
assert(false);
|
|
return 0;
|
|
}
|
|
|
|
static int _set_cca_threshold(ieee802154_dev_t *dev, int8_t threshold)
|
|
{
|
|
(void) dev;
|
|
(void) threshold;
|
|
return 0;
|
|
}
|
|
|
|
static int _set_csma_params(ieee802154_dev_t *dev, const ieee802154_csma_be_t *bd,
|
|
int8_t retries)
|
|
{
|
|
(void) dev;
|
|
(void) bd;
|
|
(void) retries;
|
|
return 0;
|
|
}
|
|
|
|
static int _config_phy(ieee802154_dev_t *dev, const ieee802154_phy_conf_t *conf)
|
|
{
|
|
(void) dev;
|
|
(void) conf;
|
|
return 0;
|
|
}
|
|
|
|
static int _config_addr_filter(ieee802154_dev_t *dev, ieee802154_af_cmd_t cmd, const void *value)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
switch (cmd) {
|
|
case IEEE802154_AF_SHORT_ADDR:
|
|
memcpy(zepdev->addr_short, value, IEEE802154_SHORT_ADDRESS_LEN);
|
|
break;
|
|
case IEEE802154_AF_EXT_ADDR:
|
|
memcpy(zepdev->addr_long, value, IEEE802154_LONG_ADDRESS_LEN);
|
|
_send_zep_hello(zepdev);
|
|
break;
|
|
case IEEE802154_AF_PANID:
|
|
memcpy(&zepdev->pan_id, value, sizeof(zepdev->pan_id));
|
|
break;
|
|
case IEEE802154_AF_PAN_COORD:
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _config_src_addr_match(ieee802154_dev_t *dev, ieee802154_src_match_t cmd,
|
|
const void *value)
|
|
{
|
|
(void) dev;
|
|
(void) cmd;
|
|
(void) value;
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
static int _set_frame_filter_mode(ieee802154_dev_t *dev, ieee802154_filter_mode_t mode)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
zepdev->filter_mode = mode;
|
|
return 0;
|
|
}
|
|
|
|
static int _request_set_trx_state(ieee802154_dev_t *dev, ieee802154_trx_state_t state)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
zepdev->state = state;
|
|
return 0;
|
|
}
|
|
|
|
static int _confirm_set_trx_state(ieee802154_dev_t *dev)
|
|
{
|
|
(void) dev;
|
|
return 0;
|
|
}
|
|
|
|
static int _write(ieee802154_dev_t *dev, const iolist_t *iolist)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
unsigned n = iolist_count(iolist);
|
|
size_t bytes = iolist_size(iolist) + sizeof(uint16_t); /* FCS field */
|
|
uint8_t *out = zepdev->snd_buf;
|
|
uint16_t chksum = 0;
|
|
|
|
DEBUG("socket_zep::write(%zu bytes)\n", bytes);
|
|
|
|
out += _zep_hdr_fill(zepdev, (void *)out, bytes);
|
|
|
|
/* make sure we are not overflowing the TX buffer */
|
|
if (out + bytes > zepdev->snd_buf + sizeof(zepdev->snd_buf)) {
|
|
return -ENOBUFS;
|
|
}
|
|
|
|
for (unsigned i = 0; i < n; i++) {
|
|
memcpy(out, iolist->iol_base, iolist->iol_len);
|
|
chksum = ucrc16_calc_le(iolist->iol_base, iolist->iol_len,
|
|
UCRC16_CCITT_POLY_LE, chksum);
|
|
out += iolist->iol_len;
|
|
iolist = iolist->iol_next;
|
|
}
|
|
chksum = byteorder_htols(chksum).u16;
|
|
memcpy(out, &chksum, sizeof(chksum));
|
|
out += sizeof(chksum);
|
|
|
|
zepdev->snd_len = out - zepdev->snd_buf;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _request_transmit(ieee802154_dev_t *dev)
|
|
{
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
DEBUG("socket_zep::request_transmit(%zu bytes)\n", zepdev->snd_len);
|
|
|
|
dev->cb(dev, IEEE802154_RADIO_INDICATION_TX_START);
|
|
|
|
int res = real_write(zepdev->sock_fd, zepdev->snd_buf, zepdev->snd_len);
|
|
|
|
dev->cb(dev, IEEE802154_RADIO_CONFIRM_TX_DONE);
|
|
|
|
if (res < 0) {
|
|
return res;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int _confirm_transmit(ieee802154_dev_t *dev, ieee802154_tx_info_t *info)
|
|
{
|
|
(void) dev;
|
|
|
|
if (info) {
|
|
info->status = TX_STATUS_SUCCESS;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int _len(ieee802154_dev_t *dev)
|
|
{
|
|
size_t size;
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
int res = real_ioctl(zepdev->sock_fd, FIONREAD, &size);
|
|
if (res < 0) {
|
|
DEBUG("socket_zep::len: error reading FIONREAD: %s", strerror(errno));
|
|
return 0;
|
|
}
|
|
|
|
DEBUG("socket_zep::len %zu bytes on %d\n", size, zepdev->sock_fd);
|
|
|
|
if (size < sizeof(zep_v2_data_hdr_t)) {
|
|
return 0;
|
|
}
|
|
|
|
/* report size without ZEP header and checksum */
|
|
return size - (sizeof(zep_v2_data_hdr_t) + 2);
|
|
}
|
|
|
|
static void _send_ack(socket_zep_t *zepdev, const void *frame)
|
|
{
|
|
const uint8_t *rxbuf = frame;
|
|
uint8_t ack[3];
|
|
zep_v2_data_hdr_t hdr;
|
|
|
|
if ((rxbuf[0] & IEEE802154_FCF_ACK_REQ) == 0) {
|
|
return;
|
|
}
|
|
|
|
DEBUG("socket_zep::send_ack: seq_no: %u\n", rxbuf[2]);
|
|
|
|
_zep_hdr_fill(zepdev, &hdr.hdr, sizeof(ack) + 2);
|
|
|
|
ack[0] = IEEE802154_FCF_TYPE_ACK; /* FCF */
|
|
ack[1] = 0; /* FCF */
|
|
ack[2] = rxbuf[2]; /* SeqNum */
|
|
|
|
/* calculate checksum */
|
|
uint16_t chksum = ucrc16_calc_le(ack, 3, UCRC16_CCITT_POLY_LE, 0);
|
|
|
|
real_send(zepdev->sock_fd, &hdr, sizeof(hdr), MSG_MORE);
|
|
real_send(zepdev->sock_fd, ack, sizeof(ack), MSG_MORE);
|
|
real_send(zepdev->sock_fd, &chksum, sizeof(chksum), 0);
|
|
}
|
|
|
|
static int _read(ieee802154_dev_t *dev, void *buf, size_t max_size,
|
|
ieee802154_rx_info_t *info)
|
|
{
|
|
int res;
|
|
socket_zep_t *zepdev = dev->priv;
|
|
|
|
DEBUG("socket_zep::read: reading up to %u bytes into %p\n", max_size, buf);
|
|
|
|
if (max_size + sizeof(zep_v2_data_hdr_t) > sizeof(zepdev->rcv_buf)) {
|
|
return 0;
|
|
}
|
|
|
|
res = real_read(zepdev->sock_fd, zepdev->rcv_buf, max_size + sizeof(zep_v2_data_hdr_t));
|
|
|
|
DEBUG("socket_zep::read: got %d bytes\n", res);
|
|
|
|
if (res <= (int)sizeof(zep_v2_data_hdr_t)) {
|
|
res = 0;
|
|
goto out;
|
|
}
|
|
|
|
zep_hdr_t *tmp = (zep_hdr_t *)zepdev->rcv_buf;
|
|
|
|
if ((tmp->preamble[0] != 'E') || (tmp->preamble[1] != 'X')) {
|
|
DEBUG("socket_zep::read: invalid ZEP header\n");
|
|
res = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
if (tmp->version != 2) {
|
|
DEBUG("socket_zep::read: unsupported ZEP version %u\n", tmp->version);
|
|
res = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
switch (((zep_v2_ack_hdr_t *)tmp)->type) {
|
|
case ZEP_V2_TYPE_DATA: {
|
|
zep_v2_data_hdr_t *zep = (zep_v2_data_hdr_t *)tmp;
|
|
|
|
if (info) {
|
|
info->lqi = zep->lqi_val;
|
|
info->rssi = -IEEE802154_RADIO_RSSI_OFFSET;
|
|
}
|
|
|
|
if (_dst_not_me(zepdev, zep + 1)) {
|
|
DEBUG("socket_zep::read: dst not me\n");
|
|
break;
|
|
}
|
|
|
|
_send_ack(zepdev, zep + 1);
|
|
|
|
memcpy(buf, zep + 1, max_size);
|
|
res = max_size;
|
|
|
|
break;
|
|
}
|
|
default:
|
|
DEBUG("socket_zep::read: unknown type %u\n", ((zep_v2_ack_hdr_t *)tmp)->type);
|
|
res = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
out:
|
|
_continue_reading(zepdev);
|
|
|
|
return res;
|
|
}
|
|
|
|
static const ieee802154_radio_ops_t socket_zep_rf_ops = {
|
|
.caps = IEEE802154_CAP_24_GHZ
|
|
| IEEE802154_CAP_AUTO_CSMA
|
|
| IEEE802154_CAP_IRQ_TX_DONE
|
|
| IEEE802154_CAP_IRQ_TX_START
|
|
| IEEE802154_CAP_PHY_OQPSK
|
|
| IEEE802154_CAP_RX_CONTINUOUS,
|
|
|
|
.write = _write,
|
|
.read = _read,
|
|
.request_transmit = _request_transmit,
|
|
.confirm_transmit = _confirm_transmit,
|
|
.len = _len,
|
|
.off = _off,
|
|
.request_on = _request_on,
|
|
.confirm_on = _confirm_on,
|
|
.request_set_trx_state = _request_set_trx_state,
|
|
.confirm_set_trx_state = _confirm_set_trx_state,
|
|
.request_cca = _request_cca,
|
|
.confirm_cca = _confirm_cca,
|
|
.set_cca_threshold = _set_cca_threshold,
|
|
.set_cca_mode = _set_cca_mode,
|
|
.config_phy = _config_phy,
|
|
.config_addr_filter = _config_addr_filter,
|
|
.config_src_addr_match = _config_src_addr_match,
|
|
.set_csma_params = _set_csma_params,
|
|
.set_frame_filter_mode = _set_frame_filter_mode,
|
|
};
|
|
|
|
void socket_zep_hal_setup(socket_zep_t *dev, ieee802154_dev_t *hal)
|
|
{
|
|
hal->driver = &socket_zep_rf_ops;
|
|
hal->priv = dev;
|
|
}
|
|
|
|
/** @} */
|