incubator-nuttx/drivers/serial/uart_bth5.c

1281 lines
28 KiB
C

/****************************************************************************
* drivers/serial/uart_bth5.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/fs/fs.h>
#include <nuttx/kmalloc.h>
#include <nuttx/circbuf.h>
#include <nuttx/mutex.h>
#include <nuttx/semaphore.h>
#include <debug.h>
#include <fcntl.h>
#include <poll.h>
#include <string.h>
#include <nuttx/wireless/bluetooth/bt_driver.h>
#include <nuttx/wireless/bluetooth/bt_hci.h>
#include <nuttx/wireless/bluetooth/bt_uart.h>
/****************************************************************************
* Private Types
****************************************************************************/
#define MAX_OPENCNT (255) /* Limit of uint8_t */
#define HCI_3WIRE_ACK_PKT 0x00
#define HCI_COMMAND_PKT 0x01
#define HCI_ACLDATA_PKT 0x02
#define HCI_SCODATA_PKT 0x03
#define HCI_EVENT_PKT 0x04
#define HCI_ISODATA_PKT 0x05
#define HCI_3WIRE_LINK_PKT 0x0f
#define HCI_VENDOR_PKT 0xff
#define SLIP_DELIMITER 0xc0
#define SLIP_ESC 0xdb
#define SLIP_ESC_DELIM 0xdc
#define SLIP_ESC_ESC 0xdd
#define H5_BIT_RX_ESC (0x00000001 << 0)
#define H5_BIT_RX_CRC (0x00000001 << 1)
#define H5_HDR_SEQ(hdr) ((hdr)[0] & 0x07)
#define H5_HDR_ACK(hdr) (((hdr)[0] >> 3) & 0x07)
#define H5_HDR_CRC(hdr) (((hdr)[0] >> 6) & 0x01)
#define H5_HDR_RELIABLE(hdr) (((hdr)[0] >> 7) & 0x01)
#define H5_HDR_PKT_TYPE(hdr) ((hdr)[1] & 0x0f)
#define H5_HDR_LEN(hdr) ((((hdr)[1] >> 4) & 0x0f) + ((hdr)[2] << 4))
#define H5_SET_SEQ(hdr, seq) ((hdr)[0] |= (seq))
#define H5_SET_ACK(hdr, ack) ((hdr)[0] |= (ack) << 3)
#define H5_SET_RELIABLE(hdr) ((hdr)[0] |= 1 << 7)
#define H5_SET_TYPE(hdr, type) ((hdr)[1] |= (type))
#define H5_SET_LEN(hdr, len) (((hdr)[1] |= ((len)&0x0f) << 4), ((hdr)[2] |= (len) >> 4))
#define H5_ACK_TIMEOUT MSEC2TICK(250) /* 250ms */
#define H5_RTX_TIMEOUT MSEC2TICK(150) /* 150ms */
union bt_hdr_u
{
struct bt_hci_cmd_hdr_s cmd;
struct bt_hci_acl_hdr_s acl;
struct bt_hci_evt_hdr_s evt;
struct bt_hci_iso_hdr_s iso;
};
enum
{
H5_MSG_INVALID,
H5_MSG_SYNC_REQ,
H5_MSG_SYNC_RSP,
H5_MSG_CONF_REQ,
H5_MSG_CONF_RSP,
};
struct unack_pool_s
{
size_t start;
size_t end;
size_t size;
uint8_t buf[CONFIG_UART_BTH5_TXWIN][CONFIG_UART_BTH5_TXBUFSIZE];
};
struct uart_bth5_s
{
FAR struct bt_driver_s *drv;
struct circbuf_s circbuf;
uint8_t sendbuf[CONFIG_UART_BTH5_TXBUFSIZE];
uint8_t recvbuf[CONFIG_UART_BTH5_TXBUFSIZE * 2];
bool crcvalid;
uint8_t openrefs;
uint16_t crcvalue;
unsigned long flags;
size_t sendlen; /* sendbuffer hci data len */
size_t recvlen;
size_t rxpending; /* Expecting more bytes */
uint8_t rxack; /* Last ack number received */
uint8_t txseq; /* Next seq number to send */
uint8_t txack; /* Next ack number to send */
uint8_t txwin; /* Sliding window size */
mutex_t openlock;
mutex_t sendlock;
mutex_t recvlock;
sem_t opensem;
sem_t recvsem;
sem_t acksem;
struct work_s retxworker;
struct work_s ackworker;
struct unack_pool_s unackpool;
CODE int (*rxfunc)(FAR struct uart_bth5_s *dev, uint8_t c);
enum
{
H5_UNINITIALIZED,
H5_INITIALIZED,
H5_ACTIVE,
} state;
FAR struct pollfd *fds[CONFIG_UART_BTH5_NPOLLWAITERS];
};
struct unack_frame_s
{
enum bt_buf_type_e type;
size_t pktlen;
uint8_t data[1];
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static int uart_bth5_open(FAR struct file *filep);
static int uart_bth5_close(FAR struct file *filep);
static ssize_t uart_bth5_read(FAR struct file *filep, FAR char *buffer,
size_t buflen);
static ssize_t uart_bth5_write(FAR struct file *filep,
FAR const char *buffer, size_t buflen);
static int uart_bth5_ioctl(FAR struct file *filep, int cmd,
unsigned long arg);
static void uart_bth5_post(FAR sem_t *sem);
static int uart_bth5_poll(FAR struct file *filep, FAR struct pollfd *fds,
bool setup);
static void uart_bth5_pollnotify(FAR struct uart_bth5_s *dev,
pollevent_t eventset);
static void h5_rx_reset(FAR struct uart_bth5_s *dev);
static int uart_h5_send(FAR struct uart_bth5_s *dev, uint8_t type,
FAR const uint8_t *payload, size_t len);
static void h5_ack_work(FAR void *arg);
static void h5_retx_work(FAR void *arg);
/****************************************************************************
* Private Data
****************************************************************************/
static const struct file_operations g_uart_bth5_ops =
{
uart_bth5_open, /* open */
uart_bth5_close, /* close */
uart_bth5_read, /* read */
uart_bth5_write, /* write */
NULL, /* seek */
uart_bth5_ioctl, /* ioctl */
NULL, /* mmap */
NULL, /* truncate */
uart_bth5_poll /* poll */
};
/****************************************************************************
* Private Functions
****************************************************************************/
static void
h5_peer_reset(FAR struct uart_bth5_s *dev)
{
dev->state = H5_UNINITIALIZED;
dev->txseq = 0;
dev->txack = 0;
work_cancel(HPWORK, &dev->retxworker);
work_cancel(HPWORK, &dev->ackworker);
}
static uint8_t
h5_crc_rev8(uint8_t byte)
{
static uint8_t rev8table[256] =
{
0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0, 0x10, 0x90, 0x50, 0xd0,
0x30, 0xb0, 0x70, 0xf0, 0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,
0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8, 0x04, 0x84, 0x44, 0xc4,
0x24, 0xa4, 0x64, 0xe4, 0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,
0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec, 0x1c, 0x9c, 0x5c, 0xdc,
0x3c, 0xbc, 0x7c, 0xfc, 0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,
0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2, 0x0a, 0x8a, 0x4a, 0xca,
0x2a, 0xaa, 0x6a, 0xea, 0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,
0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6, 0x16, 0x96, 0x56, 0xd6,
0x36, 0xb6, 0x76, 0xf6, 0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,
0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe, 0x01, 0x81, 0x41, 0xc1,
0x21, 0xa1, 0x61, 0xe1, 0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,
0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9, 0x19, 0x99, 0x59, 0xd9,
0x39, 0xb9, 0x79, 0xf9, 0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,
0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5, 0x0d, 0x8d, 0x4d, 0xcd,
0x2d, 0xad, 0x6d, 0xed, 0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,
0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3, 0x13, 0x93, 0x53, 0xd3,
0x33, 0xb3, 0x73, 0xf3, 0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,
0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb, 0x07, 0x87, 0x47, 0xc7,
0x27, 0xa7, 0x67, 0xe7, 0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,
0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef, 0x1f, 0x9f, 0x5f, 0xdf,
0x3f, 0xbf, 0x7f, 0xff
};
return rev8table[byte];
}
static uint16_t
h5_crc_rev16(uint16_t x)
{
return (h5_crc_rev8(x & 0xff) << 8) | h5_crc_rev8(x >> 8);
}
static void
h5_crc_update(FAR uint16_t *crc, uint8_t d)
{
uint16_t reg;
static const uint16_t crctable[16] =
{
0x0000, 0x1081, 0x2102, 0x3183, 0x4204, 0x5285, 0x6306, 0x7387,
0x8408, 0x9489, 0xa50a, 0xb58b, 0xc60c, 0xd68d, 0xe70e, 0xf78f
};
reg = *crc;
reg = (reg >> 4) ^ crctable[(reg ^ d) & 0x000f];
reg = (reg >> 4) ^ crctable[(reg ^ (d >> 4)) & 0x000f];
*crc = reg;
}
static uint16_t
h5_get_crc(FAR struct uart_bth5_s *dev)
{
uint8_t *hdr = dev->recvbuf;
uint8_t *data = hdr + 4 + H5_HDR_LEN(hdr);
return data[1] + (data[0] << 8);
}
static void
h5_unack_init(FAR struct unack_pool_s *pool)
{
pool->start = 0;
pool->end = 0;
pool->size = 0;
}
static FAR void *
h5_unack_ctor(FAR struct unack_pool_s *pool)
{
FAR void *p;
p = pool->buf[pool->start++];
pool->start %= CONFIG_UART_BTH5_TXWIN;
pool->size++;
return p;
}
static FAR void *
h5_unack_dtor(FAR struct unack_pool_s *pool)
{
FAR void *p;
p = pool->buf[pool->end++];
pool->end %= CONFIG_UART_BTH5_TXWIN;
pool->size--;
return p;
}
static size_t
h5_unack_size(FAR struct unack_pool_s *pool)
{
return pool->size;
}
static void
h5_unack_cleanup(FAR struct unack_pool_s *pool)
{
h5_unack_init(pool);
}
static void
h5_link_control(FAR struct uart_bth5_s *dev, FAR const void *data,
size_t len)
{
uart_h5_send(dev, HCI_3WIRE_LINK_PKT, data, len);
}
static void
h5_message_handle(FAR struct uart_bth5_s *dev)
{
FAR const uint8_t *hdr = dev->recvbuf;
FAR const uint8_t *data = &dev->recvbuf[4];
const uint8_t sync_req[] =
{
0x01, 0x7e
};
const uint8_t sync_rsp[] =
{
0x02, 0x7d
};
uint8_t conf_req[] =
{
0x03, 0xfc, 0x10 | CONFIG_UART_BTH5_TXWIN
};
const uint8_t conf_rsp[] =
{
0x04, 0x7b
};
if (H5_HDR_LEN(hdr) < 2 || (H5_HDR_PKT_TYPE(hdr) != HCI_3WIRE_LINK_PKT))
{
return;
}
if (memcmp(data, sync_req, 2) == 0)
{
h5_link_control(dev, sync_rsp, 2);
}
else if (memcmp(data, sync_rsp, 2) == 0)
{
wlinfo("h5: initialized");
dev->state = H5_INITIALIZED;
h5_link_control(dev, conf_req, 3);
}
else if (memcmp(data, conf_req, 2) == 0)
{
h5_link_control(dev, conf_rsp, 2);
h5_link_control(dev, conf_req, 3);
}
else if (memcmp(data, conf_rsp, 2) == 0)
{
if (dev->state == H5_ACTIVE)
{
return;
}
if (H5_HDR_LEN(hdr) > 2)
{
dev->txwin = (data[2] & 0x07);
dev->crcvalid = ((data[2] & 0x10) == 0x10);
wlinfo("h5 txwin:%d, crcvalid:%d", dev->txwin, dev->crcvalid);
}
if (dev->txwin < CONFIG_UART_BTH5_TXWIN)
{
wlerr("h5, txwin(%d) overflow(%d)", dev->txwin,
CONFIG_UART_BTH5_TXWIN);
return;
}
wlinfo("h5: active");
dev->state = H5_ACTIVE;
nxsem_post(&dev->opensem);
}
else
{
wlerr("ERROR Link Control: 0x%02hhx 0x%02hhx", data[0], data[1]);
}
}
static void
h5_unack_handle(FAR struct uart_bth5_s *dev)
{
size_t to_remove;
uint8_t seq;
nxmutex_lock(&dev->sendlock);
to_remove = h5_unack_size(&dev->unackpool);
if (to_remove == 0)
{
goto end;
}
seq = dev->txseq;
do
{
if (dev->rxack == seq)
{
break;
}
seq = (seq - 1) & 0x07;
}
while (--to_remove > 0);
if (seq != dev->rxack)
{
wlerr("error, %s seq:%d != rxack:%d", __func__, seq, dev->rxack);
goto end;
}
while (to_remove > 0)
{
h5_unack_dtor(&dev->unackpool);
if (to_remove == dev->txwin)
{
uart_bth5_post(&dev->acksem);
}
to_remove--;
}
if (!h5_unack_size(&dev->unackpool))
{
work_cancel(HPWORK, &dev->retxworker);
}
end:
nxmutex_unlock(&dev->sendlock);
}
static void
h5_recv_handle(FAR struct uart_bth5_s *dev)
{
int reserve = dev->drv->head_reserve;
FAR uint8_t *hdr = dev->recvbuf;
uint8_t type;
if (H5_HDR_RELIABLE(hdr))
{
dev->txack = (dev->txack + 1) & 0x07;
if (work_available(&dev->ackworker))
{
work_queue(HPWORK, &dev->ackworker, h5_ack_work, dev,
H5_ACK_TIMEOUT);
}
}
dev->rxack = H5_HDR_ACK(hdr);
type = H5_HDR_PKT_TYPE(hdr);
h5_unack_handle(dev);
switch (H5_HDR_PKT_TYPE(hdr))
{
case HCI_EVENT_PKT:
case HCI_ACLDATA_PKT:
case HCI_SCODATA_PKT:
case HCI_ISODATA_PKT:
{
nxmutex_lock(&dev->recvlock);
if (circbuf_space(&dev->circbuf) >= dev->recvlen + reserve)
{
circbuf_write(&dev->circbuf, &type, reserve);
circbuf_write(&dev->circbuf, dev->recvbuf + 4,
dev->recvlen - 4);
}
uart_bth5_pollnotify(dev, POLLIN);
nxmutex_unlock(&dev->recvlock);
break;
}
case HCI_3WIRE_LINK_PKT:
{
h5_message_handle(dev);
break;
}
default:
break;
}
h5_rx_reset(dev);
}
static int
h5_rx_crc(FAR struct uart_bth5_s *dev, uint8_t c)
{
if (h5_crc_rev16(dev->crcvalue) != h5_get_crc(dev))
{
wlerr("error, crcvalue(%04x) recv(%04x)", h5_crc_rev16(dev->crcvalue),
h5_get_crc(dev));
h5_rx_reset(dev);
return -EINVAL;
}
dev->recvlen -= 2;
h5_recv_handle(dev);
return 0;
}
static int
h5_rx_payload(FAR struct uart_bth5_s *dev, uint8_t c)
{
FAR uint8_t *hdr = dev->recvbuf;
if (H5_HDR_CRC(hdr))
{
dev->rxfunc = h5_rx_crc;
dev->rxpending = 2;
dev->flags |= H5_BIT_RX_CRC;
}
else
{
h5_recv_handle(dev);
}
return 0;
}
static int
h5_rx_header(FAR struct uart_bth5_s *dev, uint8_t c)
{
FAR uint8_t *hdr = dev->recvbuf;
wlinfo("rx t:%d l:%d s:%d a:%d", H5_HDR_PKT_TYPE(hdr), H5_HDR_LEN(hdr),
H5_HDR_SEQ(hdr), H5_HDR_ACK(hdr));
if (((hdr[0] + hdr[1] + hdr[2] + hdr[3]) & 0xff) != 0xff)
{
wlerr("error: invalid header checksum");
h5_rx_reset(dev);
return -EINVAL;
}
if (H5_HDR_RELIABLE(hdr) && H5_HDR_SEQ(hdr) != dev->txack)
{
work_queue(HPWORK, &dev->ackworker, h5_ack_work, dev, 0);
h5_rx_reset(dev);
return -EINVAL;
}
if (dev->state != H5_ACTIVE && H5_HDR_PKT_TYPE(hdr) != HCI_3WIRE_LINK_PKT)
{
wlerr("error: non-link packet received in non-active state");
h5_rx_reset(dev);
return -EINVAL;
}
dev->rxfunc = h5_rx_payload;
dev->rxpending = H5_HDR_LEN(hdr);
return 0;
}
static int
h5_rx_start(FAR struct uart_bth5_s *dev, uint8_t c)
{
if (c == SLIP_DELIMITER)
{
return 1;
}
dev->rxfunc = h5_rx_header;
dev->rxpending = 4;
dev->crcvalue = 0xffff;
return 0;
}
static int
h5_rx_delimiter(FAR struct uart_bth5_s *dev, uint8_t c)
{
if (c == SLIP_DELIMITER)
{
dev->rxfunc = h5_rx_start;
}
return 1;
}
static void
h5_rx_reset(FAR struct uart_bth5_s *dev)
{
dev->recvlen = 0;
dev->rxfunc = h5_rx_delimiter;
dev->rxpending = 0;
dev->flags = 0;
}
static int
h5_unslip_one_byte(FAR struct uart_bth5_s *dev, uint8_t c)
{
uint8_t byte = c;
if (!(dev->flags & H5_BIT_RX_ESC) && c == SLIP_ESC)
{
dev->flags |= H5_BIT_RX_ESC;
return 0;
}
if (dev->flags & H5_BIT_RX_ESC)
{
dev->flags &= ~H5_BIT_RX_ESC;
switch (c)
{
case SLIP_ESC_DELIM:
byte = SLIP_DELIMITER;
break;
case SLIP_ESC_ESC:
byte = SLIP_ESC;
break;
default:
wlerr("error: invalid esc byte 0x%02hhx", c);
h5_rx_reset(dev);
return -EINVAL;
}
}
dev->recvbuf[dev->recvlen++] = byte;
dev->rxpending--;
if (H5_HDR_CRC(dev->recvbuf) && !(dev->flags & H5_BIT_RX_CRC))
{
h5_crc_update(&dev->crcvalue, byte);
}
return 0;
}
static bool
h5_reliable_packet(uint8_t type)
{
switch (type)
{
case HCI_COMMAND_PKT:
case HCI_ACLDATA_PKT:
case HCI_EVENT_PKT:
case HCI_ISODATA_PKT:
return true;
default:
return false;
}
}
static int
h5_slip_delim(FAR uint8_t *frame, int index)
{
frame[index] = SLIP_DELIMITER;
return 1;
}
static int
h5_slip_one_byte(FAR uint8_t *frame, int index, uint8_t c)
{
int ret;
switch (c)
{
case SLIP_DELIMITER:
{
frame[index++] = SLIP_ESC;
frame[index] = SLIP_ESC_DELIM;
ret = 2;
break;
}
case SLIP_ESC:
{
frame[index++] = SLIP_ESC;
frame[index] = SLIP_ESC_ESC;
ret = 2;
break;
}
default:
frame[index] = c;
ret = 1;
break;
}
return ret;
}
static int
h5_uart_header(FAR uint8_t *data, enum bt_buf_type_e *type,
size_t *pktlen, size_t *hdrlen, size_t reserved)
{
int ret = OK;
FAR union bt_hdr_u *hdr = (FAR union bt_hdr_u *)data;
switch (*(data - reserved))
{
case H4_CMD:
{
*hdrlen = sizeof(struct bt_hci_cmd_hdr_s);
*pktlen = hdr->cmd.param_len;
*type = HCI_COMMAND_PKT;
break;
}
case H4_ACL:
{
*hdrlen = sizeof(struct bt_hci_acl_hdr_s);
*pktlen = hdr->acl.len;
*type = HCI_ACLDATA_PKT;
break;
}
case H4_ISO:
{
*hdrlen = sizeof(struct bt_hci_iso_hdr_s);
*pktlen = hdr->iso.len;
*type = HCI_ISODATA_PKT;
break;
}
default:
{
ret = -EINVAL;
break;
}
}
return ret;
}
static void
h5_ack_work(FAR void *arg)
{
FAR struct uart_bth5_s *dev = arg;
if (dev->state != H5_ACTIVE)
{
wlerr("%s state:%d not active", __func__, dev->state);
return;
}
uart_h5_send(dev, HCI_3WIRE_ACK_PKT, NULL, 0);
}
static void
h5_retx_work(FAR void *arg)
{
FAR struct uart_bth5_s *dev = arg;
FAR struct unack_frame_s *frame;
size_t size;
if (dev->state != H5_ACTIVE)
{
wlerr("%s state:%d not active", __func__, dev->state);
return;
}
nxmutex_lock(&dev->sendlock);
size = h5_unack_size(&dev->unackpool);
while (size > 0)
{
frame = (FAR struct unack_frame_s *)h5_unack_dtor(&dev->unackpool);
uart_h5_send(dev, frame->type, frame->data, frame->pktlen);
size--;
}
nxmutex_unlock(&dev->sendlock);
}
static void
uart_bth5_post(FAR sem_t *sem)
{
int semcount;
nxsem_get_value(sem, &semcount);
if (semcount < 1)
{
nxsem_post(sem);
}
}
static void
uart_bth5_pollnotify(FAR struct uart_bth5_s *dev, pollevent_t eventset)
{
poll_notify(dev->fds, CONFIG_UART_BTH5_NPOLLWAITERS, eventset);
if ((eventset & POLLIN) != 0)
{
uart_bth5_post(&dev->recvsem);
}
}
static int
uart_bth5_receive(FAR struct bt_driver_s *drv, enum bt_buf_type_e type,
FAR void *buffer, size_t buflen)
{
FAR struct uart_bth5_s *dev = drv->priv;
FAR const uint8_t *ptr = buffer;
int processed;
while (buflen > 0)
{
if (dev->rxpending > 0)
{
if (*ptr == SLIP_DELIMITER)
{
wlerr("error, too short H5 packet");
h5_rx_reset(dev);
continue;
}
h5_unslip_one_byte(dev, *ptr);
ptr++;
buflen--;
continue;
}
processed = dev->rxfunc(dev, *ptr);
if (processed < 0)
{
return processed;
}
ptr += processed;
buflen -= processed;
}
return 0;
}
static int
uart_bth5_open(FAR struct file *filep)
{
FAR struct inode *inode = filep->f_inode;
FAR struct uart_bth5_s *dev = inode->i_private;
int ret;
const uint8_t sync_req[] =
{
0x01, 0x7e
};
ret = nxmutex_lock(&dev->openlock);
if (ret < 0)
{
return ret;
}
if (dev->openrefs == MAX_OPENCNT)
{
ret = -EMFILE;
goto end;
}
else
{
dev->openrefs++;
}
if (dev->openrefs > 1)
{
goto end;
}
ret = dev->drv->open(dev->drv);
if (ret < 0)
{
goto end;
}
dev->sendlen = 0;
dev->state = H5_UNINITIALIZED;
h5_link_control(dev, sync_req, sizeof(sync_req));
ret = nxsem_tickwait_uninterruptible(&dev->opensem, SEC2TICK(3));
if (ret == -ETIMEDOUT)
{
wlerr("error, bluetooth driver open timeout");
nxmutex_unlock(&dev->openlock);
return ret;
}
h5_unack_init(&dev->unackpool);
end:
nxmutex_unlock(&dev->openlock);
return OK;
}
static int
uart_bth5_close(FAR struct file *filep)
{
FAR struct inode *inode = filep->f_inode;
FAR struct uart_bth5_s *dev = inode->i_private;
int ret = OK;
ret = nxmutex_lock(&dev->openlock);
if (ret < 0)
{
goto end;
}
if (dev->openrefs == 0)
{
ret = -EIO;
goto end;
}
else
{
dev->openrefs--;
}
if (dev->openrefs > 0)
{
goto end;
}
dev->drv->close(dev->drv);
dev->state = H5_UNINITIALIZED;
h5_peer_reset(dev);
h5_rx_reset(dev);
uart_bth5_pollnotify(dev, POLLIN | POLLOUT);
nxsem_post(&dev->opensem);
h5_unack_cleanup(&dev->unackpool);
end:
nxmutex_unlock(&dev->openlock);
return ret;
}
static ssize_t
uart_bth5_read(FAR struct file *filep, FAR char *buffer, size_t buflen)
{
FAR struct inode *inode = filep->f_inode;
FAR struct uart_bth5_s *dev = inode->i_private;
ssize_t nread;
nxmutex_lock(&dev->recvlock);
while (1)
{
nread = circbuf_read(&dev->circbuf, buffer, buflen);
if (nread != 0 || (filep->f_oflags & O_NONBLOCK))
{
break;
}
while (circbuf_is_empty(&dev->circbuf))
{
nxmutex_unlock(&dev->recvlock);
nxsem_wait_uninterruptible(&dev->recvsem);
nxmutex_lock(&dev->recvlock);
}
}
nxmutex_unlock(&dev->recvlock);
return nread;
}
static int
uart_h5_send(FAR struct uart_bth5_s *dev, uint8_t type,
FAR const uint8_t *payload, size_t len)
{
uint8_t frame[CONFIG_UART_BTH5_TXBUFSIZE];
uint8_t hdr[4];
int idx;
int length = 0;
uint16_t h5_txmsg_crc = 0xffff;
memset(hdr, 0, sizeof(hdr));
if (h5_reliable_packet(type))
{
H5_SET_RELIABLE(hdr);
H5_SET_SEQ(hdr, dev->txseq);
dev->txseq = (dev->txseq + 1) & 0x07;
work_queue(HPWORK, &dev->retxworker, h5_retx_work, dev,
H5_RTX_TIMEOUT);
}
H5_SET_ACK(hdr, dev->txack);
H5_SET_TYPE(hdr, type);
H5_SET_LEN(hdr, len);
if (dev->crcvalid)
{
hdr[0] |= 0x40;
}
/* Set head checksum */
hdr[3] = ~((hdr[0] + hdr[1] + hdr[2]) & 0xff);
length += h5_slip_delim(frame, length);
/* Put h5 header */
for (idx = 0; idx < 4; idx++)
{
length += h5_slip_one_byte(frame, length, hdr[idx]);
if (dev->crcvalid)
{
h5_crc_update(&h5_txmsg_crc, hdr[idx]);
}
}
/* Put h5 payload */
for (idx = 0; idx < len; idx++)
{
length += h5_slip_one_byte(frame, length, payload[idx]);
if (dev->crcvalid)
{
h5_crc_update(&h5_txmsg_crc, payload[idx]);
}
}
if (dev->crcvalid)
{
h5_txmsg_crc = h5_crc_rev16(h5_txmsg_crc);
length += h5_slip_one_byte(frame, length,
(uint8_t)((h5_txmsg_crc >> 8) & 0x00ff));
length += h5_slip_one_byte(frame, length,
(uint8_t)(h5_txmsg_crc & 0x00ff));
}
length += h5_slip_delim(frame, length);
work_cancel(HPWORK, &dev->ackworker);
wlinfo("tx t:%d l:%d s:%d a:%d\n", H5_HDR_PKT_TYPE(hdr), H5_HDR_LEN(hdr),
H5_HDR_SEQ(hdr), H5_HDR_ACK(hdr));
return dev->drv->send(dev->drv, type, frame, length);
}
static ssize_t
uart_bth5_write(FAR struct file *filep, FAR const char *buffer,
size_t buflen)
{
FAR struct inode *inode = filep->f_inode;
FAR struct uart_bth5_s *dev = inode->i_private;
enum bt_buf_type_e type;
size_t reserved = dev->drv->head_reserve;
FAR uint8_t *data;
size_t pktlen;
size_t hdrlen;
int ret;
FAR struct unack_frame_s *frame;
ret = nxmutex_lock(&dev->sendlock);
if (ret < 0)
{
wlerr("%s error, nxmutex_lock", __func__);
return ret;
}
data = dev->sendbuf + reserved + dev->sendlen;
if (dev->sendlen + buflen > CONFIG_UART_BTH5_TXBUFSIZE - reserved)
{
ret = -E2BIG;
goto err;
}
memcpy(data - reserved + dev->sendlen, buffer, buflen);
dev->sendlen += buflen;
for (; ; )
{
ret = h5_uart_header(data, &type, &pktlen, &hdrlen, reserved);
if (ret < 0)
{
goto err;
}
/* Reassembly is incomplete ? */
if (dev->sendlen < hdrlen)
{
goto out;
}
pktlen += hdrlen;
if (dev->sendlen < pktlen)
{
goto out;
}
/* Got the full packet, send out */
if (h5_unack_size(&dev->unackpool) > dev->txwin)
{
work_queue(HPWORK, &dev->ackworker, h5_ack_work, dev, 0);
if (filep->f_oflags & O_NONBLOCK)
{
ret = -EAGAIN;
goto out;
}
else
{
nxmutex_unlock(&dev->sendlock);
nxsem_wait_uninterruptible(&dev->acksem);
nxmutex_lock(&dev->sendlock);
}
}
ret = uart_h5_send(dev, type, data, pktlen);
if (ret < 0)
{
goto err;
}
frame = (FAR struct unack_frame_s *)h5_unack_ctor(&dev->unackpool);
frame->type = type;
frame->pktlen = pktlen;
memcpy(frame->data, data, pktlen);
dev->sendlen -= pktlen + reserved;
if (dev->sendlen > 0)
{
memmove(data - reserved, dev->sendbuf + pktlen, dev->sendlen);
}
}
err:
dev->sendlen = 0;
out:
nxmutex_unlock(&dev->sendlock);
return ret < 0 ? ret : buflen;
}
static int
uart_bth5_ioctl(FAR struct file *filep, int cmd, unsigned long arg)
{
FAR struct inode *inode = filep->f_inode;
FAR struct uart_bth5_s *dev = inode->i_private;
if (!dev->drv->ioctl)
{
return -ENOTTY;
}
return dev->drv->ioctl(dev->drv, cmd, arg);
}
static int
uart_bth5_poll(FAR struct file *filep, FAR struct pollfd *fds, bool setup)
{
FAR struct inode *inode = filep->f_inode;
FAR struct uart_bth5_s *dev = inode->i_private;
pollevent_t eventset = 0;
int ret = 0;
int i;
if (setup)
{
for (i = 0; i < CONFIG_UART_BTH5_NPOLLWAITERS; i++)
{
/* Find an available slot */
if (!dev->fds[i])
{
/* Bind the poll structure and this slot */
dev->fds[i] = fds;
fds->priv = &dev->fds[i];
break;
}
}
if (i >= CONFIG_UART_BTH5_NPOLLWAITERS)
{
fds->priv = NULL;
ret = -EBUSY;
}
nxmutex_lock(&dev->recvlock);
if (!circbuf_is_empty(&dev->circbuf))
{
eventset |= POLLIN;
}
nxmutex_unlock(&dev->recvlock);
eventset |= POLLOUT;
poll_notify(&fds, 1, eventset);
}
else if (fds->priv != NULL)
{
for (i = 0; i < CONFIG_UART_BTH5_NPOLLWAITERS; i++)
{
if (fds == dev->fds[i])
{
dev->fds[i] = NULL;
fds->priv = NULL;
break;
}
}
}
return ret;
}
/****************************************************************************
* Public Functions
****************************************************************************/
int
uart_bth5_register(FAR const char *path, FAR struct bt_driver_s *drv)
{
FAR struct uart_bth5_s *dev;
int ret;
dev = kmm_zalloc(sizeof(struct uart_bth5_s));
if (dev == NULL)
{
return -ENOMEM;
}
ret = circbuf_init(&dev->circbuf, NULL, CONFIG_UART_BTH5_RXBUFSIZE);
if (ret < 0)
{
kmm_free(dev);
return -ENOMEM;
}
dev->drv = drv;
drv->receive = uart_bth5_receive;
drv->priv = dev;
nxsem_init(&dev->opensem, 0, 0);
nxsem_init(&dev->recvsem, 0, 0);
nxsem_init(&dev->acksem, 0, 0);
nxmutex_init(&dev->sendlock);
nxmutex_init(&dev->recvlock);
nxmutex_init(&dev->openlock);
h5_rx_reset(dev);
ret = register_driver(path, &g_uart_bth5_ops, 0666, dev);
if (ret < 0)
{
nxsem_destroy(&dev->recvsem);
nxsem_destroy(&dev->opensem);
nxsem_destroy(&dev->acksem);
nxmutex_destroy(&dev->sendlock);
nxmutex_destroy(&dev->openlock);
nxmutex_destroy(&dev->recvlock);
circbuf_uninit(&dev->circbuf);
kmm_free(dev);
}
return ret;
}