2025-05-10 21:58:58 +08:00

2072 lines
48 KiB
C

/*
* Copyright (C) 2013 Realtek Semiconductor Corp.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <errno.h>
#include <unistd.h>
#include <stdlib.h>
#include <termios.h>
#include <time.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/param.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/uio.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <signal.h>
#include <stdint.h>
#include <string.h>
#include <endian.h>
#include <byteswap.h>
#include <netinet/in.h>
#include <poll.h>
#include <sys/timerfd.h>
#include <sys/epoll.h>
#include "rtb_fwc.h"
#include "hciattach.h"
#include "hciattach_h4.h"
#define RTK_VERSION "3.1.390bad8.20220519-142434"
#define TIMESTAMP_PR
#define MAX_EVENTS 10
/* #define SERIAL_NONBLOCK_READ */
#ifdef SERIAL_NONBLOCK_READ
#define FD_BLOCK 0
#define FD_NONBLOCK 1
#endif
/* #define RTL_8703A_SUPPORT */
/* #define RTL8723DSH4_UART_HWFLOWC */ /* 8723DS H4 special */
uint8_t DBG_ON = 1;
#define HCI_EVENT_HDR_SIZE 2
#define PATCH_DATA_FIELD_MAX_SIZE 252
#define HCI_CMD_READ_BD_ADDR 0x1009
#define HCI_VENDOR_CHANGE_BAUD 0xfc17
#define HCI_VENDOR_READ_ROM_VER 0xfc6d
#define HCI_CMD_READ_LOCAL_VER 0x1001
#define HCI_VENDOR_READ_CHIP_TYPE 0xfc61
#define HCI_CMD_RESET 0x0c03
/* HCI data types */
#define H5_ACK_PKT 0x00
#define HCI_COMMAND_PKT 0x01
#define HCI_ACLDATA_PKT 0x02
#define HCI_SCODATA_PKT 0x03
#define HCI_EVENT_PKT 0x04
#define H5_VDRSPEC_PKT 0x0E
#define H5_LINK_CTL_PKT 0x0F
#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) & 0xff) + ((hdr)[2] << 4))
#define H5_HDR_SIZE 4
struct sk_buff {
uint32_t max_len;
uint32_t data_len;
uint8_t *data;
};
struct hci_ev_cmd_complete {
uint8_t ncmd;
uint16_t opcode;
} __attribute__ ((packed));
#define OP_H5_SYNC 0x01
#define OP_H5_CONFIG 0x02
#define OP_ROM_VER ((1 << 24) | HCI_VENDOR_READ_ROM_VER)
#define OP_LMP_VER ((1 << 24) | HCI_CMD_READ_LOCAL_VER)
#define OP_CHIP_TYPE ((1 << 24) | HCI_VENDOR_READ_CHIP_TYPE)
#define OP_SET_BAUD ((1 << 24) | HCI_VENDOR_CHANGE_BAUD)
#define OP_HCI_RESET ((1 << 24) | HCI_CMD_RESET)
struct rtb_struct rtb_cfg;
/* bite reverse in bytes
* 00000001 -> 10000000
* 00000100 -> 00100000
*/
const uint8_t byte_rev_table[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,
};
static __inline uint8_t bit_rev8(uint8_t byte)
{
return byte_rev_table[byte];
}
static __inline uint16_t bit_rev16(uint16_t x)
{
return (bit_rev8(x & 0xff) << 8) | bit_rev8(x >> 8);
}
static const uint16_t crc_table[] = {
0x0000, 0x1081, 0x2102, 0x3183,
0x4204, 0x5285, 0x6306, 0x7387,
0x8408, 0x9489, 0xa50a, 0xb58b,
0xc60c, 0xd68d, 0xe70e, 0xf78f
};
/* Initialise the crc calculator */
#define H5_CRC_INIT(x) x = 0xffff
static __inline struct sk_buff *skb_alloc(unsigned int len)
{
struct sk_buff *skb = NULL;
if ((skb = malloc(len + sizeof(*skb)))) {
skb->max_len = len;
skb->data_len = 0;
skb->data = ((uint8_t *)skb) + sizeof(*skb);
} else {
RS_ERR("Allocate skb fails!");
skb = NULL;
return NULL;
}
memset(skb->data, 0, len);
return skb;
}
static __inline void skb_free(struct sk_buff *skb)
{
free(skb);
return;
}
/*
* Add data to a buffer
* This function extends the used data area of the buffer.
*/
static uint8_t *skb_put(struct sk_buff *skb, uint32_t len)
{
uint32_t old_len = skb->data_len;
if ((skb->data_len + len) > (skb->max_len)) {
RS_ERR("Buffer too small");
exit(EXIT_FAILURE);
}
skb->data_len += len;
return (skb->data + old_len);
}
/*
* Remove end from a buffer
* Cut the length of a buffer down by removing data from the tail
*/
static void skb_trim(struct sk_buff *skb, uint32_t len)
{
if (skb->data_len > len) {
skb->data_len = len;
} else {
RS_ERR("Trim error, data_len %u < len %u", skb->data_len, len);
}
}
/*
* Remove data from the start of a buffer
* This function removes data from the start of a buffer.
* A pointer to the next data in the buffer is returned
*/
static uint8_t *skb_pull(struct sk_buff *skb, uint32_t len)
{
if (len > skb->data_len) {
RS_ERR("Pull error, data_len %u < len %u", skb->data_len, len);
exit(EXIT_FAILURE);
}
skb->data_len -= len;
skb->data += len;
return skb->data;
}
/**
* Add "d" into crc scope, caculate the new crc value
*
* @param crc crc data
* @param d one byte data
*/
static void h5_crc_update(uint16_t * crc, uint8_t d)
{
uint16_t reg = *crc;
reg = (reg >> 4) ^ crc_table[(reg ^ d) & 0x000f];
reg = (reg >> 4) ^ crc_table[(reg ^ (d >> 4)) & 0x000f];
*crc = reg;
}
struct __una_u16 {
uint16_t x;
};
static __inline uint16_t __get_unaligned_cpu16(const void *p)
{
const struct __una_u16 *ptr = (const struct __una_u16 *)p;
return ptr->x;
}
static __inline uint16_t get_unaligned_be16(const void *p)
{
return __get_unaligned_cpu16((const uint8_t *)p);
}
/*
* Get crc data.
*/
static uint16_t h5_get_crc(struct rtb_struct * h5)
{
uint16_t crc = 0;
uint8_t *data = h5->rx_skb->data + h5->rx_skb->data_len - 2;
crc = data[1] + (data[0] << 8);
return crc;
/* return get_unaligned_be16(&h5->rx_skb->data[h5->rx_skb->data_len - 2]); */
}
/*
* Add 0xc0 to buffer.
*/
static void h5_slip_msgdelim(struct sk_buff *skb)
{
const char pkt_delim = 0xc0;
memcpy(skb_put(skb, 1), &pkt_delim, 1);
}
/*
* Encode one byte in h5 proto
* 0xc0 -> 0xdb, 0xdc
* 0xdb -> 0xdb, 0xdd
* 0x11 -> 0xdb, 0xde
* 0x13 -> 0xdb, 0xdf
* others will not change
*/
static void h5_slip_one_byte(struct sk_buff *skb, uint8_t c)
{
const uint8_t esc_c0[2] = { 0xdb, 0xdc };
const uint8_t esc_db[2] = { 0xdb, 0xdd };
const uint8_t esc_11[2] = { 0xdb, 0xde };
const uint8_t esc_13[2] = { 0xdb, 0xdf };
switch (c) {
case 0xc0:
memcpy(skb_put(skb, 2), &esc_c0, 2);
break;
case 0xdb:
memcpy(skb_put(skb, 2), &esc_db, 2);
break;
case 0x11:
memcpy(skb_put(skb, 2), &esc_11, 2);
break;
case 0x13:
memcpy(skb_put(skb, 2), &esc_13, 2);
break;
default:
memcpy(skb_put(skb, 1), &c, 1);
break;
}
}
/*
* Decode one byte in h5 proto
* 0xdb, 0xdc -> 0xc0
* 0xdb, 0xdd -> 0xdb
* 0xdb, 0xde -> 0x11
* 0xdb, 0xdf -> 0x13
* others will not change
*/
static void h5_unslip_one_byte(struct rtb_struct * h5, unsigned char byte)
{
const uint8_t c0 = 0xc0, db = 0xdb;
const uint8_t oof1 = 0x11, oof2 = 0x13;
if (H5_ESCSTATE_NOESC == h5->rx_esc_state) {
if (0xdb == byte) {
h5->rx_esc_state = H5_ESCSTATE_ESC;
} else {
memcpy(skb_put(h5->rx_skb, 1), &byte, 1);
/* Check Pkt Header's CRC enable bit */
if ((h5->rx_skb->data[0] & 0x40) != 0 &&
h5->rx_state != H5_W4_CRC) {
h5_crc_update(&h5->message_crc, byte);
}
h5->rx_count--;
}
} else if (H5_ESCSTATE_ESC == h5->rx_esc_state) {
switch (byte) {
case 0xdc:
memcpy(skb_put(h5->rx_skb, 1), &c0, 1);
if ((h5->rx_skb->data[0] & 0x40) != 0 &&
h5->rx_state != H5_W4_CRC)
h5_crc_update(&h5->message_crc, 0xc0);
h5->rx_esc_state = H5_ESCSTATE_NOESC;
h5->rx_count--;
break;
case 0xdd:
memcpy(skb_put(h5->rx_skb, 1), &db, 1);
if ((h5->rx_skb->data[0] & 0x40) != 0 &&
h5->rx_state != H5_W4_CRC)
h5_crc_update(&h5->message_crc, 0xdb);
h5->rx_esc_state = H5_ESCSTATE_NOESC;
h5->rx_count--;
break;
case 0xde:
memcpy(skb_put(h5->rx_skb, 1), &oof1, 1);
if ((h5->rx_skb->data[0] & 0x40) != 0 &&
h5->rx_state != H5_W4_CRC)
h5_crc_update(&h5->message_crc, oof1);
h5->rx_esc_state = H5_ESCSTATE_NOESC;
h5->rx_count--;
break;
case 0xdf:
memcpy(skb_put(h5->rx_skb, 1), &oof2, 1);
if ((h5->rx_skb->data[0] & 0x40) != 0 &&
h5->rx_state != H5_W4_CRC)
h5_crc_update(&h5->message_crc, oof2);
h5->rx_esc_state = H5_ESCSTATE_NOESC;
h5->rx_count--;
break;
default:
RS_ERR("Error: Invalid byte %02x after esc byte", byte);
skb_free(h5->rx_skb);
h5->rx_skb = NULL;
h5->rx_state = H5_W4_PKT_DELIMITER;
h5->rx_count = 0;
break;
}
}
}
/*
* Prepare h5 packet
* Refer to Core Spec Vol 4, Part D
* Three-wire UART Transport Layer: 4 PACKET HEADER
*/
static struct sk_buff *h5_prepare_pkt(struct rtb_struct * h5, uint8_t *data,
int len, int pkt_type)
{
struct sk_buff *nskb;
uint8_t hdr[4];
uint16_t H5_CRC_INIT(h5_txmsg_crc);
int rel, i;
switch (pkt_type) {
case HCI_ACLDATA_PKT:
case HCI_COMMAND_PKT:
case HCI_EVENT_PKT:
rel = 1; /* reliable */
break;
case H5_ACK_PKT:
case H5_VDRSPEC_PKT:
case H5_LINK_CTL_PKT:
rel = 0; /* unreliable */
break;
default:
RS_ERR("Unknown packet type");
return NULL;
}
/* Max len of packet: (len + 4(h5 hdr) + 2(crc))*2
* Because bytes 0xc0 and 0xdb are escaped, worst case is that the
* packet is only made of 0xc0 and 0xdb
* The additional 2-octets are 0xc0 delimiters at start and end of each
* packet.
*/
nskb = skb_alloc((len + 6) * 2 + 2);
if (!nskb)
return NULL;
/* Add SLIP start byte: 0xc0 */
h5_slip_msgdelim(nskb);
/* Set ack number in SLIP header */
hdr[0] = h5->rxseq_txack << 3;
h5->is_txack_req = 0;
/* RS_DBG("Request packet no(%u) to card", h5->rxseq_txack); */
/* RS_DBG("Sending packet with seqno %u and wait %u", h5->msgq_txseq,
* h5->rxseq_txack);
*/
if (rel) {
/* Set reliable bit and seq number */
hdr[0] |= 0x80 + h5->msgq_txseq;
/* RS_DBG("Sending packet with seqno(%u)", h5->msgq_txseq); */
++(h5->msgq_txseq);
h5->msgq_txseq = (h5->msgq_txseq) & 0x07;
}
/* Set DIC Present bit */
if (h5->use_crc)
hdr[0] |= 0x40;
/* Set packet type and payload length */
hdr[1] = ((len << 4) & 0xff) | pkt_type;
hdr[2] = (uint8_t) (len >> 4);
/* Set header checksum */
hdr[3] = ~(hdr[0] + hdr[1] + hdr[2]);
/* Encode h5 header */
for (i = 0; i < 4; i++) {
h5_slip_one_byte(nskb, hdr[i]);
if (h5->use_crc)
h5_crc_update(&h5_txmsg_crc, hdr[i]);
}
/* Encode payload */
for (i = 0; i < len; i++) {
h5_slip_one_byte(nskb, data[i]);
if (h5->use_crc)
h5_crc_update(&h5_txmsg_crc, data[i]);
}
/* Encode CRC */
if (h5->use_crc) {
h5_txmsg_crc = bit_rev16(h5_txmsg_crc);
h5_slip_one_byte(nskb, (uint8_t) ((h5_txmsg_crc >> 8) & 0x00ff));
h5_slip_one_byte(nskb, (uint8_t) (h5_txmsg_crc & 0x00ff));
}
/* Add 0xc0 at the end of the packet */
h5_slip_msgdelim(nskb);
return nskb;
}
/*
* Remove controller acked packet from host unacked lists
*/
/* static void h5_remove_acked_pkt(struct rtb_struct * h5)
* {
* int pkts_to_be_removed = 0;
* int seqno = 0;
* int i = 0;
*
* seqno = h5->msgq_txseq;
* // pkts_to_be_removed = GetListLength(h5->unacked);
*
* while (pkts_to_be_removed) {
* if (h5->rxack == seqno)
* break;
*
* pkts_to_be_removed--;
* seqno = (seqno - 1) & 0x07;
* }
*
* if (h5->rxack != seqno) {
* RS_DBG("Peer acked invalid packet");
* }
* // skb_queue_walk_safe(&h5->unack, skb, tmp)
* // remove ack'ed packet from h5->unack queue
* for (i = 0; i < 5; ++i) {
* if (i >= pkts_to_be_removed)
* break;
* i++;
* //__skb_unlink(skb, &h5->unack);
* //skb_free(skb);
* }
*
* // if (skb_queue_empty(&h5->unack))
* // del_timer(&h5->th5);
* // spin_unlock_irqrestore(&h5->unack.lock, flags);
*
* if (i != pkts_to_be_removed)
* RS_DBG("Removed only (%u) out of (%u) pkts", i,
* pkts_to_be_removed);
* }
*/
/*
* Send host ack.
*/
static void rtb_send_ack(int fd)
{
int len;
struct sk_buff *nskb = h5_prepare_pkt(&rtb_cfg, NULL, 0, H5_ACK_PKT);
len = write(fd, nskb->data, nskb->data_len);
if (len != nskb->data_len)
RS_ERR("Write pure ack fails");
skb_free(nskb);
return;
}
/*
* Parse hci command complete event in h5 init state.
*/
static void h5_init_hci_cc(struct sk_buff *skb)
{
struct hci_ev_cmd_complete *ev = NULL;
uint16_t opcode = 0;
uint8_t status = 0;
skb_pull(skb, HCI_EVENT_HDR_SIZE);
ev = (struct hci_ev_cmd_complete *)skb->data;
opcode = le16_to_cpu(ev->opcode);
RS_DBG("Receive cmd complete event of command: %04x", opcode);
skb_pull(skb, sizeof(struct hci_ev_cmd_complete));
status = skb->data[0];
if (status) {
RS_ERR("status is %u for cmd %04x", status, opcode);
return;
}
if (rtb_cfg.cmd_state.opcode != opcode) {
RS_ERR("%s: Received unexpected cc for cmd %04x, %04x of cc",
__func__, rtb_cfg.cmd_state.opcode, opcode);
return;
}
rtb_cfg.cmd_state.state = CMD_STATE_SUCCESS;
switch (opcode) {
case HCI_VENDOR_CHANGE_BAUD:
RS_INFO("Received cc of vendor change baud");
break;
case HCI_CMD_READ_BD_ADDR:
RS_INFO("BD Address: %02x:%02x:%02x:%02x:%02x:%02x",
skb->data[5], skb->data[4], skb->data[3],
skb->data[2], skb->data[1], skb->data[0]);
break;
case HCI_CMD_READ_LOCAL_VER:
rtb_cfg.hci_ver = skb->data[1];
rtb_cfg.hci_rev = (skb->data[2] | skb->data[3] << 8);
rtb_cfg.lmp_subver = (skb->data[7] | (skb->data[8] << 8));
RS_INFO("HCI Version 0x%02x", rtb_cfg.hci_ver);
RS_INFO("HCI Revision 0x%04x", rtb_cfg.hci_rev);
RS_INFO("LMP Subversion 0x%04x", rtb_cfg.lmp_subver);
break;
case HCI_VENDOR_READ_ROM_VER:
rtb_cfg.eversion = skb->data[1];
RS_INFO("Read ROM version %02x", rtb_cfg.eversion);
break;
case HCI_VENDOR_READ_CHIP_TYPE:
rtb_cfg.chip_type = (skb->data[1] & 0x0f);
rtb_cfg.chip_ver = (skb->data[2] & 0x0f);
RS_INFO("Read chip type %02x", rtb_cfg.chip_type);
RS_INFO("Read chip ver %02x", rtb_cfg.chip_ver);
break;
default:
return;
}
/* Count the cmd num for makeing the seq number aligned */
rtb_cfg.num_of_cmd_sent++;
}
/*
* Parse hci command complete event in h5 post state.
*/
static void h5_post_hci_cc(struct sk_buff *skb)
{
struct hci_ev_cmd_complete *ev = NULL;
uint16_t opcode = 0;
uint8_t status = 0;
skb_pull(skb, HCI_EVENT_HDR_SIZE);
ev = (struct hci_ev_cmd_complete *)skb->data;
opcode = le16_to_cpu(ev->opcode);
RS_DBG("Receive cmd complete event of command: %04x", opcode);
skb_pull(skb, sizeof(struct hci_ev_cmd_complete));
status = skb->data[0];
if (status) {
RS_ERR("status is %u for cmd %04x", status, opcode);
return;
}
if (rtb_cfg.cmd_state.opcode != opcode) {
RS_ERR("%s: Received unexpected cc for cmd %04x, %04x of cc",
__func__, rtb_cfg.cmd_state.opcode, opcode);
return;
}
rtb_cfg.cmd_state.state = CMD_STATE_SUCCESS;
switch (opcode) {
case HCI_CMD_RESET:
RS_INFO("Received cc of hci reset cmd");
rtb_cfg.link_estab_state = H5_ACTIVE;
break;
default:
break;
}
}
/*
* Process a hci frame
*/
static void hci_recv_frame(struct sk_buff *skb)
{
if (rtb_cfg.link_estab_state == H5_INIT) {
if (skb->data[0] == 0x0e)
h5_init_hci_cc(skb);
/*
* rtb_send_ack(rtb_cfg.serial_fd);
* usleep(10000);
* rtb_send_ack(rtb_cfg.serial_fd);
*/
} else if (rtb_cfg.link_estab_state == H5_PATCH) {
if (skb->data[0] != 0x0e) {
RS_INFO("Received event 0x%x during download patch",
skb->data[0]);
return;
}
rtb_cfg.rx_index = skb->data[6];
/* RS_INFO("rx_index %d", rtb_cfg.rx_index); */
/* Download fw/config done */
if (rtb_cfg.rx_index & 0x80) {
rtb_cfg.rx_index &= ~0x80;
rtb_cfg.link_estab_state = H5_HCI_RESET;
}
} else if (rtb_cfg.link_estab_state == H5_HCI_RESET) {
if (skb->data[0] == 0x0e)
h5_post_hci_cc(skb);
} else {
RS_ERR("receive packets in active state");
}
}
static void h5_handle_internal_rx(struct sk_buff *skb)
{
int len;
uint8_t sync_req[2] = { 0x01, 0x7E };
uint8_t sync_resp[2] = { 0x02, 0x7D };
uint8_t sync_resp_pkt[0x8] = {
0xc0, 0x00, 0x2F, 0x00, 0xD0, 0x02, 0x7D, 0xc0
};
uint8_t conf_req[2] = { 0x03, 0xFC };
uint8_t conf_resp[2] = { 0x04, 0x7B };
uint8_t conf_resp_pkt[0x8] = {
0xc0, 0x00, 0x2F, 0x00, 0xD0, 0x04, 0x7B, 0xc0
};
if (rtb_cfg.link_estab_state == H5_SYNC) {
if (!memcmp(skb->data, sync_req, 2)) {
RS_INFO("[SYNC] Get SYNC Pkt\n");
len = write(rtb_cfg.serial_fd, sync_resp_pkt, 0x8);
if (len != 0x08)
RS_ERR("Send h5 sync resp error, %s",
strerror(errno));
} else if (!memcmp(skb->data, sync_resp, 2)) {
RS_INFO("[SYNC] Get SYNC Resp Pkt");
rtb_cfg.link_estab_state = H5_CONFIG;
}
} else if (rtb_cfg.link_estab_state == H5_CONFIG) {
if (!memcmp(skb->data, sync_req, 0x2)) {
RS_INFO("[CONFIG] Get SYNC pkt");
len = write(rtb_cfg.serial_fd, sync_resp_pkt, 0x8);
if (len != 0x08)
RS_ERR("Send h5 sync resp error, %s",
strerror(errno));
} else if (!memcmp(skb->data, conf_req, 0x2)) {
RS_INFO("[CONFIG] Get CONFG pkt");
len = write(rtb_cfg.serial_fd, conf_resp_pkt, 0x8);
if (len != 0x08)
RS_ERR("Send h5 sync resp to ctl error, %s",
strerror(errno));
} else if (!memcmp(skb->data, conf_resp, 0x2)) {
RS_INFO("[CONFIG] Get CONFG resp pkt");
/* Change state to H5_INIT after receiving a conf resp
*/
rtb_cfg.link_estab_state = H5_INIT;
if (skb->data_len > 2) {
rtb_cfg.use_crc = ((skb->data[2]) >> 4) & 0x01;
RS_INFO("dic is %u, cfg field 0x%02x",
rtb_cfg.use_crc, skb->data[2]);
}
} else {
RS_WARN("[CONFIG] Get unknown pkt");
rtb_send_ack(rtb_cfg.serial_fd);
}
}
}
/*
* Process the received complete h5 packet
*/
static void h5_complete_rx_pkt(struct rtb_struct *h5)
{
int pass_up = 1;
uint8_t *h5_hdr = NULL;
h5_hdr = (uint8_t *) (h5->rx_skb->data);
if (H5_HDR_RELIABLE(h5_hdr)) {
/* RS_DBG("Received reliable seqno %u from card", h5->rxseq_txack);
*/
h5->rxseq_txack = H5_HDR_SEQ(h5_hdr) + 1;
/* h5->rxseq_txack %= 8; */
h5->rxseq_txack &= 0x07;
h5->is_txack_req = 1;
}
h5->rxack = H5_HDR_ACK(h5_hdr);
switch (H5_HDR_PKT_TYPE(h5_hdr)) {
case HCI_ACLDATA_PKT:
case HCI_EVENT_PKT:
case HCI_COMMAND_PKT:
/* h5_remove_acked_pkt(h5); */
pass_up = 1;
break;
case HCI_SCODATA_PKT:
pass_up = 1;
break;
case H5_LINK_CTL_PKT:
pass_up = 0;
skb_pull(h5->rx_skb, H5_HDR_SIZE);
h5_handle_internal_rx(h5->rx_skb);
break;
default: /* Pure ack or other unexpected pkt */
pass_up = 0;
break;
}
if (pass_up) {
skb_pull(h5->rx_skb, H5_HDR_SIZE);
hci_recv_frame(h5->rx_skb);
}
if (h5->is_txack_req) {
rtb_send_ack(rtb_cfg.serial_fd);
h5->is_txack_req = 0;
}
skb_free(h5->rx_skb);
h5->rx_state = H5_W4_PKT_DELIMITER;
h5->rx_skb = NULL;
}
/*
* Parse the receive data in h5 proto.
*/
static int h5_recv(struct rtb_struct *h5, void *data, int count)
{
unsigned char *ptr;
ptr = (unsigned char *)data;
while (count) {
if (h5->rx_count) {
if (*ptr == 0xc0) {
RS_ERR("Short h5 packet");
skb_free(h5->rx_skb);
h5->rx_state = H5_W4_PKT_START;
h5->rx_count = 0;
} else
h5_unslip_one_byte(h5, *ptr);
ptr++;
count--;
continue;
}
switch (h5->rx_state) {
case H5_W4_HDR:
/* Check header checksum */
if ((0xff & (uint8_t)~(h5->rx_skb->data[0] + h5->rx_skb->data[1] +
h5->rx_skb->data[2])) != h5->rx_skb->data[3]) {
RS_ERR("h5 hdr checksum error");
skb_free(h5->rx_skb);
h5->rx_state = H5_W4_PKT_DELIMITER;
h5->rx_count = 0;
continue;
}
/* The received seq number is unexpected */
if (h5->rx_skb->data[0] & 0x80 &&
(h5->rx_skb->data[0] & 0x07) != h5->rxseq_txack) {
uint8_t rxseq_txack = (h5->rx_skb->data[0] & 0x07);
RS_ERR("Out-of-order packet arrived, got(%u)expected(%u)",
h5->rx_skb->data[0] & 0x07,
h5->rxseq_txack);
h5->is_txack_req = 1;
skb_free(h5->rx_skb);
h5->rx_state = H5_W4_PKT_DELIMITER;
h5->rx_count = 0;
/* Depend on whether Controller will reset ack
* number or not
*/
if (rtb_cfg.link_estab_state == H5_PATCH &&
rtb_cfg.tx_index == rtb_cfg.total_num)
rtb_cfg.rxseq_txack = rxseq_txack;
continue;
}
h5->rx_state = H5_W4_DATA;
h5->rx_count =
(h5->rx_skb->data[1] >> 4) +
(h5->rx_skb->data[2] << 4);
continue;
case H5_W4_DATA:
/* Packet with crc */
if (h5->rx_skb->data[0] & 0x40) {
h5->rx_state = H5_W4_CRC;
h5->rx_count = 2;
} else {
h5_complete_rx_pkt(h5);
}
continue;
case H5_W4_CRC:
if (bit_rev16(h5->message_crc) != h5_get_crc(h5)) {
RS_ERR("Checksum failed, computed %04x received %04x",
bit_rev16(h5->message_crc),
h5_get_crc(h5));
skb_free(h5->rx_skb);
h5->rx_state = H5_W4_PKT_DELIMITER;
h5->rx_count = 0;
continue;
}
skb_trim(h5->rx_skb, h5->rx_skb->data_len - 2);
h5_complete_rx_pkt(h5);
continue;
case H5_W4_PKT_DELIMITER:
switch (*ptr) {
case 0xc0:
h5->rx_state = H5_W4_PKT_START;
break;
default:
break;
}
ptr++;
count--;
break;
case H5_W4_PKT_START:
switch (*ptr) {
case 0xc0:
ptr++;
count--;
break;
default:
h5->rx_state = H5_W4_HDR;
h5->rx_count = 4;
h5->rx_esc_state = H5_ESCSTATE_NOESC;
H5_CRC_INIT(h5->message_crc);
/* Do not increment ptr or decrement count
* Allocate packet. Max len of a H5 pkt=
* 0xFFF (payload) +4 (header) +2 (crc)
*/
h5->rx_skb = skb_alloc(0x1005);
if (!h5->rx_skb) {
RS_ERR("Can't alloc skb for new pkt");
h5->rx_state = H5_W4_PKT_DELIMITER;
h5->rx_count = 0;
return 0;
}
break;
}
break;
default:
break;
}
}
return count;
}
static const char *op_string(uint32_t op)
{
switch (op) {
case OP_SET_BAUD:
return "OP_SET_BAUD";
case OP_H5_SYNC:
return "OP_H5_SYNC";
case OP_H5_CONFIG:
return "OP_H5_CONFIG";
case OP_HCI_RESET:
return "OP_HCI_RESET";
case OP_CHIP_TYPE:
return "OP_CHIP_TYPE";
case OP_ROM_VER:
return "OP_ROM_VER";
case OP_LMP_VER:
return "OP_LMP_VER";
default:
return "OP_UNKNOWN";
}
}
static int start_transmit_wait(int fd, struct sk_buff *skb,
uint32_t op, unsigned int msec, int retry)
{
unsigned char buf[128];
ssize_t result;
struct iovec iov;
ssize_t ret;
uint8_t *data;
int len;
int op_result = -1;
uint64_t expired;
int n;
struct epoll_event events[MAX_EVENTS];
int nfds;
uint16_t opcode = 0;
if (fd == -1 || !skb) {
RS_ERR("Invalid parameter");
return -1;
}
data = skb->data;
len = skb->data_len;
if (op & (1 << 24)) {
opcode = (op & 0xffff);
if (opcode != rtb_cfg.cmd_state.opcode ||
rtb_cfg.cmd_state.state != CMD_STATE_UNKNOWN) {
RS_ERR("Invalid opcode or cmd state");
return -1;
}
}
iov.iov_base = data;
iov.iov_len = len;
do {
ret = writev(fd, &iov, 1);
if (ret != len)
RS_WARN("Writev partially, ret %d", (int)ret);
} while (ret < 0 && errno == EINTR);
if (ret < 0) {
RS_ERR("Call writev error, %s", strerror(errno));
return -errno;
}
/* Set timeout */
if (rtb_cfg.timerfd > 0)
timeout_set(rtb_cfg.timerfd, msec);
do {
nfds = epoll_wait(rtb_cfg.epollfd, events, MAX_EVENTS, msec);
if (nfds == -1) {
RS_ERR("epoll_wait, %s (%d)", strerror(errno), errno);
exit(EXIT_FAILURE);
}
for (n = 0; n < nfds; ++n) {
if (events[n].data.fd == rtb_cfg.serial_fd) {
if (events[n].events & (EPOLLERR | EPOLLHUP |
EPOLLRDHUP)) {
RS_ERR("%s: Error happens on serial fd",
__func__);
exit(EXIT_FAILURE);
}
result = read(events[n].data.fd, buf,
sizeof(buf));
if (result <= 0) {
RS_ERR("Read serial error, %s",
strerror(errno));
continue;
} else {
h5_recv(&rtb_cfg, buf, result);
}
} else if (events[n].data.fd == rtb_cfg.timerfd) {
if (events[n].events & (EPOLLERR | EPOLLHUP |
EPOLLRDHUP)) {
RS_ERR("%s: Error happens on timer fd",
__func__);
exit(EXIT_FAILURE);
}
RS_WARN("%s Transmission timeout",
op_string(op));
result = read(events[n].data.fd, &expired,
sizeof(expired));
if (result != sizeof(expired)) {
RS_ERR("Skip retransmit");
break;
}
if (retry <= 0) {
RS_ERR("Retransmission exhausts");
tcflush(fd, TCIOFLUSH);
exit(EXIT_FAILURE);
}
iov.iov_base = data;
iov.iov_len = len;
do {
ret = writev(fd, &iov, 1);
if (ret != len)
RS_WARN("Writev partial, %d",
(int)ret);
} while (ret < 0 && errno == EINTR);
if (ret < 0) {
RS_ERR("ReCall writev error, %s",
strerror(errno));
return -errno;
}
retry--;
timeout_set(rtb_cfg.timerfd, msec);
}
}
if (!(op & (1 << 24))) {
/* h5 sync or config */
if (op == OP_H5_SYNC && rtb_cfg.link_estab_state ==
H5_CONFIG) {
op_result = 0;
break;
}
if (op == OP_H5_CONFIG && rtb_cfg.link_estab_state ==
H5_INIT) {
op_result = 0;
break;
}
continue;
}
if (rtb_cfg.cmd_state.opcode == opcode &&
rtb_cfg.cmd_state.state == CMD_STATE_SUCCESS) {
op_result = 0;
break;
}
} while (1);
/* Disarms timer */
timeout_set(rtb_cfg.timerfd, 0);
return op_result;
}
static int h5_download_patch(int dd, int index, uint8_t *data, int len,
struct termios *ti)
{
unsigned char buf[64];
int retlen;
struct iovec iov;
ssize_t ret;
int nfds;
struct epoll_event events[MAX_EVENTS];
int n;
int timeout;
uint64_t expired;
int retry = 3;
struct sk_buff *nskb;
uint8_t hci_patch[PATCH_DATA_FIELD_MAX_SIZE + 4];
if (index & 0x80) {
rtb_cfg.tx_index = index & 0x7f;
timeout = 1000;
} else {
rtb_cfg.tx_index = index;
timeout = 800;
}
/* download cmd: 0xfc20 */
hci_patch[0] = 0x20;
hci_patch[1] = 0xfc;
hci_patch[2] = len + 1;
hci_patch[3] = (uint8_t)index;
if (data)
memcpy(&hci_patch[4], data, len);
/* length: 2-byte opcode + 1-byte len + 1-byte index + payload */
nskb = h5_prepare_pkt(&rtb_cfg, hci_patch, len + 4, HCI_COMMAND_PKT);
if (!nskb) {
RS_ERR("Prepare command packet for download");
return -1;
}
/* Save pkt address and length for re-transmission */
len = nskb->data_len;
data = nskb->data;
iov.iov_base = nskb->data;
iov.iov_len = nskb->data_len;
do {
ret = writev(dd, &iov, 1);
if (ret != len)
RS_WARN("Writev partially, ret %d", (int)ret);
} while (ret < 0 && errno == EINTR);
if (ret < 0) {
RS_ERR("Call writev error, %s", strerror(errno));
skb_free(nskb);
return -errno;
}
/* RS_INFO("%s: tx_index %d, rx_index %d", __func__,
* rtb_cfg.tx_index, rtb_cfg.rx_index);
*/
if (index & 0x80) {
/* For the last pkt, wait for its complete */
tcdrain(dd);
if (rtb_cfg.uart_flow_ctrl) {
RS_INFO("Enable host hw flow control");
ti->c_cflag |= CRTSCTS;
} else {
RS_INFO("Disable host hw flow control");
ti->c_cflag &= ~CRTSCTS;
}
if (tcsetattr(dd, TCSANOW, ti) < 0) {
RS_ERR("Can't set port settings");
skb_free(nskb);
return -1;
}
/* RS_INFO("Change baud to %d", rtb_cfg.final_speed);
* if (set_speed(dd, ti, rtb_cfg.final_speed) < 0) {
* RS_ERR("Set final speed %d error",
* rtb_cfg.final_speed);
* }
*/
}
if (rtb_cfg.timerfd > 0)
timeout_set(rtb_cfg.timerfd, timeout);
do {
nfds = epoll_wait(rtb_cfg.epollfd, events, MAX_EVENTS, -1);
if (nfds == -1) {
RS_ERR("epoll_wait, %s (%d)", strerror(errno), errno);
exit(EXIT_FAILURE);
}
for (n = 0; n < nfds; ++n) {
if (events[n].data.fd == dd) {
if (events[n].events & (EPOLLERR | EPOLLHUP |
EPOLLRDHUP)) {
RS_ERR("%s: Error happens on serial fd",
__func__);
exit(EXIT_FAILURE);
}
retlen = read(dd, buf, sizeof(buf));
if (retlen <= 0) {
RS_ERR("Read serial error, %s", strerror(errno));
continue;
} else {
h5_recv(&rtb_cfg, buf, retlen);
}
} else if (events[n].data.fd == rtb_cfg.timerfd) {
int fd = events[n].data.fd;
if (events[n].events & (EPOLLERR | EPOLLHUP |
EPOLLRDHUP)) {
RS_ERR("%s: Error happens on timer fd",
__func__);
exit(EXIT_FAILURE);
}
RS_WARN("Patch pkt trans timeout, re-trans");
ret = read(fd, &expired, sizeof(expired));
if (ret != sizeof(expired)) {
RS_ERR("Read expired info error");
exit(EXIT_FAILURE);
}
if (retry <= 0) {
RS_ERR("%s: Retransmission exhausts",
__func__);
tcflush(fd, TCIOFLUSH);
exit(EXIT_FAILURE);
}
iov.iov_base = data;
iov.iov_len = len;
do {
ret = writev(dd, &iov, 1);
if (ret != len)
RS_WARN("Writev partial, %d",
(int)ret);
} while (ret < 0 && errno == EINTR);
if (ret < 0) {
RS_ERR("ReCall writev error, %s",
strerror(errno));
skb_free(nskb);
return -errno;
}
retry--;
timeout_set(fd, timeout);
}
}
} while (rtb_cfg.rx_index != rtb_cfg.tx_index);
/* Disarms timer */
if (rtb_cfg.timerfd > 0)
timeout_set(rtb_cfg.timerfd, 0);
skb_free(nskb);
return 0;
}
/*
* Change the Controller's UART speed.
*/
int h5_vendor_change_speed(int fd, uint32_t baudrate)
{
struct sk_buff *nskb = NULL;
unsigned char cmd[16] = { 0 };
int result;
cmd[0] = 0x17;
cmd[1] = 0xfc;
cmd[2] = 4;
baudrate = cpu_to_le32(baudrate);
#ifdef BAUDRATE_4BYTES
memcpy((uint16_t *) & cmd[3], &baudrate, 4);
#else
memcpy((uint16_t *) & cmd[3], &baudrate, 2);
cmd[5] = 0;
cmd[6] = 0;
#endif
RS_DBG("baudrate in change speed command: 0x%02x 0x%02x 0x%02x 0x%02x",
cmd[3], cmd[4], cmd[5], cmd[6]);
nskb = h5_prepare_pkt(&rtb_cfg, cmd, 7, HCI_COMMAND_PKT);
if (!nskb) {
RS_ERR("Prepare command packet for change speed fail");
return -1;
}
rtb_cfg.cmd_state.opcode = HCI_VENDOR_CHANGE_BAUD;;
rtb_cfg.cmd_state.state = CMD_STATE_UNKNOWN;
result = start_transmit_wait(fd, nskb, OP_SET_BAUD, 1000, 0);
skb_free(nskb);
if (result < 0) {
RS_ERR("OP_SET_BAUD Transmission error");
return result;
}
return 0;
}
/*
* Init realtek Bluetooth h5 proto.
* There are two steps: h5 sync and h5 config.
*/
int rtb_init_h5(int fd, struct termios *ti)
{
struct sk_buff *nskb;
unsigned char h5sync[2] = { 0x01, 0x7E };
/* 16-bit CCITT CRC may be used and the sliding win size is 4 */
unsigned char h5conf[3] = { 0x03, 0xFC, 0x14 };
int result;
/* Disable CRTSCTS by default */
ti->c_cflag &= ~CRTSCTS;
/* set even parity */
ti->c_cflag |= PARENB;
ti->c_cflag &= ~(PARODD);
if (tcsetattr(fd, TCSANOW, ti) < 0) {
RS_ERR("Can't set port settings");
return -1;
}
/* h5 sync */
rtb_cfg.link_estab_state = H5_SYNC;
nskb = h5_prepare_pkt(&rtb_cfg, h5sync, sizeof(h5sync),
H5_LINK_CTL_PKT);
result = start_transmit_wait(fd, nskb, OP_H5_SYNC, 500, 10);
skb_free(nskb);
if (result < 0) {
RS_ERR("OP_H5_SYNC Transmission error");
return -1;
}
/* h5 config */
nskb = h5_prepare_pkt(&rtb_cfg, h5conf, sizeof(h5conf), H5_LINK_CTL_PKT);
result = start_transmit_wait(fd, nskb, OP_H5_CONFIG, 500, 10);
skb_free(nskb);
if (result < 0) {
RS_ERR("OP_H5_CONFIG Transmission error");
return -1;
}
rtb_send_ack(fd);
RS_DBG("H5 init finished\n");
rtb_cfg.cmd_state.state = CMD_STATE_UNKNOWN;
return 0;
}
static int h5_hci_reset(int fd)
{
uint8_t cmd[3] = { 0x03, 0x0c, 0x00};
struct sk_buff *nskb;
int result;
RS_INFO("%s: Issue hci reset cmd", __func__);
nskb = h5_prepare_pkt(&rtb_cfg, cmd, sizeof(cmd), HCI_COMMAND_PKT);
if (!nskb) {
RS_ERR("%s: Failed to alloc mem for hci reset skb", __func__);
return -1;
}
rtb_cfg.cmd_state.opcode = HCI_CMD_RESET;
rtb_cfg.cmd_state.state = CMD_STATE_UNKNOWN;
result = start_transmit_wait(fd, nskb, OP_HCI_RESET, 1500, 1);
skb_free(nskb);
if (result < 0)
RS_ERR("hci reset failed");
return result;
}
#ifdef SERIAL_NONBLOCK_READ
static int set_fd_nonblock(int fd)
{
long arg;
int old_fl;
arg = fcntl(fd, F_GETFL);
if (arg < 0)
return -errno;
/* Return if already nonblock */
if (arg & O_NONBLOCK)
return FD_NONBLOCK;
old_fl = FD_BLOCK;
arg |= O_NONBLOCK;
if (fcntl(fd, F_SETFL, arg) < 0)
return -errno;
return old_fl;
}
static int set_fd_block(int fd)
{
long arg;
arg = fcntl(fd, F_GETFL);
if (arg < 0)
return -errno;
/* Return if already block */
if (!(arg & O_NONBLOCK))
return 0;
arg &= ~O_NONBLOCK;
if (fcntl(fd, F_SETFL, arg) < 0)
return -errno;
return 0;
}
#endif
/*
* Download Realtek Firmware and Config
*/
static int rtb_download_fwc(int fd, uint8_t *buf, int size, int proto,
struct termios *ti)
{
uint8_t curr_idx = 0;
uint8_t curr_len = 0;
uint8_t lp_len = 0;
uint8_t add_pkts = 0;
uint16_t end_idx = 0;
uint16_t total_idx = 0;
uint16_t num;
unsigned char *pkt_buf;
uint16_t i, j;
int result;
#ifdef SERIAL_NONBLOCK_READ
int old_fl;
#endif
end_idx = (uint16_t)((size - 1) / PATCH_DATA_FIELD_MAX_SIZE);
lp_len = size % PATCH_DATA_FIELD_MAX_SIZE;
num = rtb_cfg.num_of_cmd_sent;
num += end_idx + 1;
add_pkts = num % 8 ? (8 - num % 8) : 0;
#ifdef SERIAL_NONBLOCK_READ
old_fl = set_fd_nonblock(fd);
if (old_fl < 0) {
RS_ERR("Set fd nonblock error, %s", strerror(errno));
}
if (old_fl == FD_BLOCK)
RS_INFO("old fd state is block");
#endif
/* Make sure the next seqno is zero after download patch and
* hci reset
*/
if (proto == HCI_UART_3WIRE) {
if (add_pkts)
add_pkts -= 1;
else
add_pkts += 7;
} else
add_pkts = 0; /* No additional packets need */
total_idx = add_pkts + end_idx;
rtb_cfg.total_num = total_idx;
RS_INFO("end_idx: %u, lp_len: %u, additional pkts: %u\n", end_idx,
lp_len, add_pkts);
RS_INFO("Start downloading...");
if (lp_len == 0)
lp_len = PATCH_DATA_FIELD_MAX_SIZE;
pkt_buf = buf;
for (i = 0; i <= total_idx; i++) {
/* Index will roll over when it reaches 0x80
* 0, 1, 2, 3, ..., 126, 127(7f), 1, 2, 3, ...
*/
if (i > 0x7f)
j = (i & 0x7f) + 1;
else
j = i;
if (i < end_idx) {
curr_idx = j;
curr_len = PATCH_DATA_FIELD_MAX_SIZE;
} else if (i == end_idx) {
/* Send last data packets */
if (i == total_idx)
curr_idx = j | 0x80;
else
curr_idx = j;
curr_len = lp_len;
} else if (i < total_idx) {
/* Send additional packets */
curr_idx = j;
pkt_buf = NULL;
curr_len = 0;
RS_INFO("Send additional packet %u", curr_idx);
} else {
/* Send last packet */
curr_idx = j | 0x80;
pkt_buf = NULL;
curr_len = 0;
RS_INFO("Last packet %u", curr_idx);
}
if (curr_idx & 0x80)
RS_INFO("Send last pkt");
if (proto == HCI_UART_H4) {
curr_idx = h4_download_patch(fd, curr_idx, pkt_buf,
curr_len);
if (curr_idx != j && i != total_idx) {
RS_ERR("Index mismatch %u, curr_idx %u", j,
curr_idx);
return -1;
}
} else if (proto == HCI_UART_3WIRE) {
if (h5_download_patch(fd, curr_idx, pkt_buf, curr_len,
ti) < 0)
return -1;
}
if (curr_idx < end_idx) {
pkt_buf += PATCH_DATA_FIELD_MAX_SIZE;
}
}
/* Make hci reset after Controller applies the Firmware and Config */
if (proto == HCI_UART_H4)
result = h4_hci_reset(fd);
else
result = h5_hci_reset(fd);
if (proto == HCI_UART_3WIRE) {
/* Make sure the last pure ack is sent */
tcdrain(fd);
}
if (result)
return result;
#ifdef SERIAL_NONBLOCK_READ
if (old_fl == FD_BLOCK)
set_fd_block(fd);
#endif
return 0;
}
#define ARRAY_SIZE(a) (sizeof(a)/sizeof(a[0]) )
struct rtb_baud {
uint32_t rtb_speed;
int uart_speed;
};
#ifdef BAUDRATE_4BYTES
struct rtb_baud baudrates[] = {
#ifdef RTL_8703A_SUPPORT
{0x00004003, 1500000}, /* for rtl8703as */
#endif
{0x0252C014, 115200},
{0x0252C00A, 230400},
{0x05F75004, 921600},
{0x00005004, 1000000},
{0x04928002, 1500000},
{0x01128002, 1500000}, //8761AT
{0x00005002, 2000000},
{0x0000B001, 2500000},
{0x04928001, 3000000},
{0x052A6001, 3500000},
{0x00005001, 4000000},
};
#else
struct rtb_baud baudrates[] = {
{0x701d, 115200}
{0x6004, 921600},
{0x4003, 1500000},
{0x5002, 2000000},
{0x8001, 3000000},
{0x9001, 3000000},
{0x7001, 3500000},
{0x5001, 4000000},
};
#endif
static void vendor_speed_to_std(uint32_t rtb_speed, uint32_t *uart_speed)
{
*uart_speed = 115200;
unsigned int i;
for (i = 0; i < ARRAY_SIZE(baudrates); i++) {
if (baudrates[i].rtb_speed == rtb_speed) {
*uart_speed = baudrates[i].uart_speed;
return;
}
}
return;
}
static inline void std_speed_to_vendor(int uart_speed, uint32_t *rtb_speed)
{
*rtb_speed = 0x701D;
unsigned int i;
for (i = 0; i < ARRAY_SIZE(baudrates); i++) {
if (baudrates[i].uart_speed == uart_speed) {
*rtb_speed = baudrates[i].rtb_speed;
return;
}
}
return;
}
void rtb_read_chip_type(int dd)
{
/* 0xB000A094 */
unsigned char cmd_buff[] = {
0x61, 0xfc, 0x05, 0x10, 0xA6, 0xAD, 0x00, 0xB0
};
struct sk_buff *nskb;
int result;
nskb = h5_prepare_pkt(&rtb_cfg, cmd_buff, sizeof(cmd_buff),
HCI_COMMAND_PKT);
if (!nskb) {
RS_ERR("Alloc chip type cmd skb buff error");
exit(EXIT_FAILURE);
}
rtb_cfg.cmd_state.opcode = HCI_VENDOR_READ_CHIP_TYPE;
rtb_cfg.cmd_state.state = CMD_STATE_UNKNOWN;
result = start_transmit_wait(dd, nskb, OP_CHIP_TYPE, 250, 3);
skb_free(nskb);
if (result < 0)
RS_ERR("OP_CHIP_TYPE Transmission error");
return;
}
/*
* Read ECO version with vendor cmd 0xfc65
*/
void rtb_read_eversion(int dd)
{
int result;
unsigned char cmd_buf[3] = { 0x6d, 0xfc, 0x00 };
struct sk_buff *nskb;
nskb= h5_prepare_pkt(&rtb_cfg, cmd_buf, 3, HCI_COMMAND_PKT);
if (!nskb) {
RS_ERR("Alloc eversion cmd skb buff error");
exit(EXIT_FAILURE);
}
rtb_cfg.cmd_state.opcode = HCI_VENDOR_READ_ROM_VER;
rtb_cfg.cmd_state.state = CMD_STATE_UNKNOWN;
result = start_transmit_wait(dd, nskb, OP_ROM_VER, 500, 3);
skb_free(nskb);
if (result < 0) {
RS_ERR("OP_ROM_VER Transmit error");
}
return;
}
void rtb_read_local_version(int dd)
{
int result;
unsigned char cmd_buf[3] = { 0x01, 0x10, 0x00 };
struct sk_buff *nskb;
nskb = h5_prepare_pkt(&rtb_cfg, cmd_buf, 3, HCI_COMMAND_PKT);
if (!nskb) {
RS_ERR("Alloc local ver cmd skb buff error");
exit(EXIT_FAILURE);
}
rtb_cfg.cmd_state.state = CMD_STATE_UNKNOWN;
rtb_cfg.cmd_state.opcode = HCI_CMD_READ_LOCAL_VER;
result = start_transmit_wait(dd, nskb, OP_LMP_VER, 500, 3);
skb_free(nskb);
if (result < 0) {
RS_ERR("OP_LMP_VER Transmit error");
}
return;
}
/*
* Config Realtek Bluetooth.
* Config parameters are got from Realtek Config file and FW.
*
* speed is the init_speed in uart struct
* Returns 0 on success
*/
static int rtb_config(int fd, int proto, int speed, struct termios *ti)
{
int final_speed = 0;
int ret = 0;
int max_patch_size = 0;
rtb_cfg.proto = proto;
/* Read Local Version Information and RTK ROM version */
if (proto == HCI_UART_3WIRE) {
RS_INFO("Realtek H5 IC");
rtb_read_local_version(fd);
rtb_read_eversion(fd);
} else {
RS_INFO("Realtek H4 IC");
/* The following set is for special requirement that enables
* flow control before initializing */
#ifdef RTL8723DSH4_UART_HWFLOWC
ti->c_cflag &= ~PARENB;
ti->c_cflag |= CRTSCTS;
if (tcsetattr(fd, TCSANOW, ti) < 0) {
RS_ERR("H4 Can't enable RTSCTS");
return -1;
}
usleep(20 * 1000);
#endif
h4_read_local_ver(fd);
h4_vendor_read_rom_ver(fd);
if (rtb_cfg.lmp_subver == ROM_LMP_8761btc) {
/* 8761B Test Chip */
rtb_cfg.chip_type = CHIP_8761BTC;
rtb_cfg.uart_flow_ctrl = 1;
/* TODO: Change to different uart baud */
std_speed_to_vendor(1500000, &rtb_cfg.vendor_baud);
goto change_baud;
} else if (rtb_cfg.lmp_subver == ROM_LMP_8761a) {
if (rtb_cfg.hci_rev == 0x000b) {
/* 8761B Test Chip without download */
rtb_cfg.chip_type = CHIP_8761BH4;
/* rtb_cfg.uart_flow_ctrl = 1; */
/* TODO: Change to different uart baud */
/* std_speed_to_vendor(1500000, &rtb_cfg.vendor_baud);
* goto change_baud;
*/
} else if (rtb_cfg.hci_rev == 0x000a) {
if (rtb_cfg.eversion == 3)
rtb_cfg.chip_type = CHIP_8761ATF;
else if (rtb_cfg.eversion == 2)
rtb_cfg.chip_type = CHIP_8761AT;
else
rtb_cfg.chip_type = CHIP_UNKNOWN;
}
} else if (rtb_cfg.lmp_subver == ROM_LMP_8723b) {
if (rtb_cfg.hci_ver == 0x08 &&
rtb_cfg.hci_rev == 0x000d) {
rtb_cfg.chip_type = CHIP_8723DS;
} else if (rtb_cfg.hci_ver == 0x06 &&
rtb_cfg.hci_rev == 0x000b) {
rtb_cfg.chip_type = CHIP_8723BS;
} else {
RS_ERR("H4: unknown chip");
return -1;
}
}
}
RS_INFO("LMP Subversion 0x%04x", rtb_cfg.lmp_subver);
RS_INFO("EVersion %u", rtb_cfg.eversion);
switch (rtb_cfg.lmp_subver) {
case ROM_LMP_8723a:
break;
case ROM_LMP_8723b:
#ifdef RTL_8703A_SUPPORT
/* Set chip type for matching fw/config entry */
rtl->chip_type = CHIP_8703AS;
#endif
break;
case ROM_LMP_8821a:
break;
case ROM_LMP_8761a:
rtb_read_chip_type(fd);
break;
case ROM_LMP_8703b:
rtb_read_chip_type(fd);
break;
case ROM_LMP_8852a:
rtb_read_chip_type(fd);
break;
}
if((rtb_cfg.chip_type == CHIP_8852BPE_VR) || (rtb_cfg.chip_type == CHIP_8852BPS)) {
rtb_cfg.chip_type = CHIP_8852BP;
RS_INFO("chip_type: %d", rtb_cfg.chip_type);
}
if(rtb_cfg.chip_type == CHIP_8852BP) {
rtb_cfg.eversion = rtb_cfg.chip_ver;
RS_INFO("eversion: %d", rtb_cfg.eversion);
}
rtb_cfg.patch_ent = get_patch_entry(&rtb_cfg);
if (rtb_cfg.patch_ent) {
RS_INFO("IC: %s", rtb_cfg.patch_ent->ic_name);
RS_INFO("Firmware/config: %s, %s",
rtb_cfg.patch_ent->patch_file,
rtb_cfg.patch_ent->config_file);
} else {
RS_ERR("Can not find firmware/config entry");
return -1;
}
rtb_cfg.config_buf = rtb_read_config(rtb_cfg.patch_ent->config_file,
&rtb_cfg.config_len,
rtb_cfg.patch_ent->chip_type);
if (!rtb_cfg.config_buf) {
RS_ERR("Read Config file error, use eFuse settings");
rtb_cfg.config_len = 0;
}
rtb_cfg.fw_buf = rtb_read_firmware(&rtb_cfg, &rtb_cfg.fw_len);
if (!rtb_cfg.fw_buf) {
RS_ERR("Read Bluetooth firmware error");
rtb_cfg.fw_len = 0;
/* Free config buf */
if (rtb_cfg.config_buf) {
free(rtb_cfg.config_buf);
rtb_cfg.config_buf = NULL;
rtb_cfg.config_len = 0;
}
return -1;
} else {
rtb_cfg.total_buf = rtb_get_final_patch(fd, proto,
&rtb_cfg.total_len);
/* If the above function executes successfully, the Config and
* patch were copied to the total buf */
/* Free config buf */
if (rtb_cfg.config_buf) {
free(rtb_cfg.config_buf);
rtb_cfg.config_buf = NULL;
}
/* Free the fw buf */
free(rtb_cfg.fw_buf);
rtb_cfg.fw_buf = NULL;
rtb_cfg.fw_len = 0;
if (!rtb_cfg.total_buf) {
RS_ERR("Failed to get the final patch");
exit(EXIT_FAILURE);
}
}
switch ((rtb_cfg.patch_ent)->chip_type) {
case CHIP_8822BS:
max_patch_size = 25 * 1024;
break;
case CHIP_8821CS:
case CHIP_8723DS:
case CHIP_8822CS:
case CHIP_8761B:
case CHIP_8725AS:
max_patch_size = 40 * 1024;
break;
case CHIP_8852AS:
max_patch_size = 0x114D0 + 529; /* 69.2KB */
break;
case CHIP_8723FS:
max_patch_size = 0xC4Cf + 529; /* 49.2KB */
break;
case CHIP_8852BS:
case CHIP_8852BP:
max_patch_size = 0x104D0 + 529; /* 65KB */
break;
case CHIP_8852CS:
max_patch_size = 0x130D0 + 529; /* 76.2KB */
break;
default:
max_patch_size = 24 * 1024;
break;
}
if (rtb_cfg.total_len > max_patch_size) {
RS_ERR("Total length of fwc is larger than allowed");
goto buf_free;
}
RS_INFO("Total len %d for fwc", rtb_cfg.total_len);
/* rtl8723ds h4 */
if (rtb_cfg.chip_type == CHIP_8723DS &&
rtb_cfg.proto == HCI_UART_H4) {
if (rtb_cfg.parenb) {
/* set parity */
ti->c_cflag |= PARENB;
if (rtb_cfg.pareven)
ti->c_cflag &= ~(PARODD);
else
ti->c_cflag |= PARODD;
if (tcsetattr(fd, TCSANOW, ti) < 0) {
RS_ERR("8723DSH4 Can't set parity");
goto buf_free;
}
}
}
change_baud:
/* change baudrate if needed
* rtb_cfg.vendor_baud is a __u32/__u16 vendor-specific variable
* parsed from config file
* */
if (rtb_cfg.vendor_baud == 0) {
/* No baud setting in Config file */
std_speed_to_vendor(speed, &rtb_cfg.vendor_baud);
RS_INFO("No baud from Config file, set baudrate: %d, 0x%08x",
speed, rtb_cfg.vendor_baud);
goto start_download;
} else
vendor_speed_to_std(rtb_cfg.vendor_baud,
(uint32_t *)&(rtb_cfg.final_speed));
if (rtb_cfg.final_speed == 115200) {
RS_INFO("Final speed is %d, no baud change needs",
rtb_cfg.final_speed);
goto start_download;
}
if (proto == HCI_UART_3WIRE)
h5_vendor_change_speed(fd, rtb_cfg.vendor_baud);
else
h4_vendor_change_speed(fd, rtb_cfg.vendor_baud);
/* Make sure the ack for cmd complete event is transmitted */
tcdrain(fd);
usleep(50000); /* The same value as before */
final_speed = rtb_cfg.final_speed ? rtb_cfg.final_speed : speed;
RS_INFO("Final speed %d", final_speed);
if (set_speed(fd, ti, final_speed) < 0) {
RS_ERR("Can't set baud rate: %d, %d, %d", final_speed,
rtb_cfg.final_speed, speed);
goto buf_free;
}
start_download:
/* For 8761B Test chip, no patch to download */
if (rtb_cfg.chip_type == CHIP_8761BTC)
goto done;
if (rtb_cfg.total_len > 0 && rtb_cfg.dl_fw_flag) {
rtb_cfg.link_estab_state = H5_PATCH;
rtb_cfg.rx_index = -1;
ret = rtb_download_fwc(fd, rtb_cfg.total_buf, rtb_cfg.total_len,
proto, ti);
free(rtb_cfg.total_buf);
if (ret < 0)
return ret;
}
done:
RS_DBG("Init Process finished");
return 0;
buf_free:
free(rtb_cfg.total_buf);
return -1;
}
int rtb_init(int fd, int proto, int speed, struct termios *ti)
{
struct epoll_event ev;
int result;
RS_INFO("Realtek hciattach version %s \n", RTK_VERSION);
memset(&rtb_cfg, 0, sizeof(rtb_cfg));
rtb_cfg.serial_fd = fd;
rtb_cfg.dl_fw_flag = 1;
rtb_cfg.epollfd = epoll_create(64);
if (rtb_cfg.epollfd == -1) {
RS_ERR("epoll_create1, %s (%d)", strerror(errno), errno);
exit(EXIT_FAILURE);
}
ev.events = EPOLLIN | EPOLLERR | EPOLLHUP | EPOLLRDHUP;
ev.data.fd = fd;
if (epoll_ctl(rtb_cfg.epollfd, EPOLL_CTL_ADD, fd, &ev) == -1) {
RS_ERR("epoll_ctl: epoll ctl add, %s (%d)", strerror(errno),
errno);
exit(EXIT_FAILURE);
}
rtb_cfg.timerfd = timerfd_create(CLOCK_MONOTONIC, 0);
if (rtb_cfg.timerfd == -1) {
RS_ERR("timerfd_create error, %s (%d)", strerror(errno), errno);
return -1;
}
if (rtb_cfg.timerfd > 0) {
ev.events = EPOLLIN | EPOLLERR | EPOLLHUP | EPOLLRDHUP;
ev.data.fd = rtb_cfg.timerfd;
if (epoll_ctl(rtb_cfg.epollfd, EPOLL_CTL_ADD,
rtb_cfg.timerfd, &ev) == -1) {
RS_ERR("epoll_ctl: epoll ctl add, %s (%d)",
strerror(errno), errno);
exit(EXIT_FAILURE);
}
}
RS_INFO("Use epoll");
if (proto == HCI_UART_3WIRE) {
if (rtb_init_h5(fd, ti) < 0)
return -1;;
}
result = rtb_config(fd, proto, speed, ti);
epoll_ctl(rtb_cfg.epollfd, EPOLL_CTL_DEL, fd, NULL);
epoll_ctl(rtb_cfg.epollfd, EPOLL_CTL_DEL, rtb_cfg.timerfd, NULL);
close(rtb_cfg.timerfd);
rtb_cfg.timerfd = -1;
return result;
}
int rtb_post(int fd, int proto, struct termios *ti)
{
/* No need to change baudrate */
/* if (rtb_cfg.final_speed)
* return set_speed(fd, ti, rtb_cfg.final_speed);
*/
return 0;
}