Return-Path: Date: Wed, 17 Feb 2010 15:25:45 +0530 From: Vikram Kandukuri To: "linux-bluetooth@vger.kernel.org" CC: Subject: Re: [PATCH] Added support for Atheros AR300x UART Bluetooth Chip Message-ID: <20100217095542.GB2138@ATH-LT-538> References: <20100209114231.GA6587@ATH-LT-538> MIME-Version: 1.0 Content-Type: text/plain; charset="us-ascii" In-Reply-To: <20100209114231.GA6587@ATH-LT-538> List-ID: Hi Marcel, Please send your comments Thanks > > Signed-off-by: Suraj > > --- > drivers/bluetooth/Kconfig | 10 + > drivers/bluetooth/Makefile | 1 + > drivers/bluetooth/hci_ath.c | 545 +++++++++++++++++++++++++++++++++++++++++ > drivers/bluetooth/hci_ath.h | 72 ++++++ > drivers/bluetooth/hci_ldisc.c | 6 + > drivers/bluetooth/hci_uart.h | 8 +- > 6 files changed, 641 insertions(+), 1 deletions(-) > create mode 100755 drivers/bluetooth/hci_ath.c > create mode 100755 drivers/bluetooth/hci_ath.h > > diff --git a/drivers/bluetooth/Kconfig b/drivers/bluetooth/Kconfig > index 058fbcc..32b98a4 100644 > --- a/drivers/bluetooth/Kconfig > +++ b/drivers/bluetooth/Kconfig > @@ -58,6 +58,16 @@ config BT_HCIUART_BCSP > > Say Y here to compile support for HCI BCSP protocol. > > +config BT_HCIUART_ATH > + bool "Atheros AR3001 Board support" > + depends on BT_HCIUART > + help > + HCIATH (HCI Atheros) is a serial protocol for communication > + between Bluetooth device and host. This protocol is required for > + serial Bluetooth devices that are based on Atheros AR3001 chips. > + > + Say Y here to compile support for HCIATH protocol. > + > config BT_HCIUART_LL > bool "HCILL protocol support" > depends on BT_HCIUART > diff --git a/drivers/bluetooth/Makefile b/drivers/bluetooth/Makefile > index 7e5aed5..1481faa 100644 > --- a/drivers/bluetooth/Makefile > +++ b/drivers/bluetooth/Makefile > @@ -26,4 +26,5 @@ hci_uart-y := hci_ldisc.o > hci_uart-$(CONFIG_BT_HCIUART_H4) += hci_h4.o > hci_uart-$(CONFIG_BT_HCIUART_BCSP) += hci_bcsp.o > hci_uart-$(CONFIG_BT_HCIUART_LL) += hci_ll.o > +hci_uart-$(CONFIG_BT_HCIUART_ATH) += hci_ath.o > hci_uart-objs := $(hci_uart-y) > diff --git a/drivers/bluetooth/hci_ath.c b/drivers/bluetooth/hci_ath.c > new file mode 100755 > index 0000000..abacc09 > --- /dev/null > +++ b/drivers/bluetooth/hci_ath.c > @@ -0,0 +1,545 @@ > +/* > + * Copyright (c) 2009-2010 Atheros Communications Inc. > + * > + * 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. > + * > + * You should have received a copy of the GNU General Public License > + * along with this program; if not, write to the Free Software > + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA > + * > + */ > + > +#include > +#include > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#include > +#include > + > +#include "hci_uart.h" > +#include "hci_ath.h" > +static void ath_context_switch(struct work_struct *work) > +{ > + int status; > + struct ath_struct *ath; > + struct hci_uart *hu; > + struct tty_struct *tty; > + struct sk_buff *skbuf; > + ath = container_of(work, struct ath_struct, ctxtsw); > + hu = ath->hu; > + tty = hu->tty; > + status = ath_wakeup_ar3001(tty); > + if ((status & TIOCM_CTS)) { > + while ((skbuf = skb_dequeue(&ath->tx_wait_q))) > + skb_queue_tail(&ath->txq, skbuf); > + > + /* Ready to send Data */ > + clear_bit(HCI_UART_SENDING, &hu->tx_state); > + hci_uart_tx_wakeup(hu); > + } > +} > + > +/* Initialize protocol */ > +static int ath_open(struct hci_uart *hu) > +{ > + struct ath_struct *ath; > + BT_DBG("hu %p", hu); > + ath = kzalloc(sizeof(*ath), GFP_ATOMIC); > + if (!ath) > + return -ENOMEM; > + skb_queue_head_init(&ath->txq); > + skb_queue_head_init(&ath->tx_wait_q); > + skb_queue_head_init(&ath->tx_cmd_wait_q); > + spin_lock_init(&ath->hciath_lock); > + ath->cur_sleep = 0; > + ath->num_cmds_complete = 1; > + hu->priv = ath; > + ath->hu = hu; > + init_waitqueue_head(&ath->wqevt); > + INIT_WORK(&ath->ctxtsw, ath_context_switch); > + return 0; > +} > + > +/* Flush protocol data */ > +static int ath_flush(struct hci_uart *hu) > +{ > + struct ath_struct *ath = hu->priv; > + BT_DBG("hu %p", hu); > + skb_queue_purge(&ath->tx_wait_q); > + skb_queue_purge(&ath->txq); > + skb_queue_purge(&ath->tx_cmd_wait_q); > + return 0; > +} > + > +/* Close protocol */ > +static int ath_close(struct hci_uart *hu) > +{ > + struct ath_struct *ath; > + ath = hu->priv; > + BT_DBG("hu %p", hu); > + skb_queue_purge(&ath->tx_wait_q); > + skb_queue_purge(&ath->txq); > + skb_queue_purge(&ath->tx_cmd_wait_q); > + if (ath->rx_skb) > + kfree_skb(ath->rx_skb); > + wake_up_interruptible(&ath->wqevt); > + hu->priv = NULL; > + kfree(ath); > + return 0; > +} > + > +/* Enqueue frame for transmittion (padding, crc, etc) */ > +/* may be called from two simultaneous tasklets */ > +static int ath_enqueue(struct hci_uart *hu, struct sk_buff *skb) > +{ > + struct ath_struct *ath; > + ath = hu->priv; > + if (HCI_SCODATA_PKT == bt_cb(skb)->pkt_type) { > + > + /* Discard SCO packet as AR3001 does not support SCO over HCI */ > + BT_DBG("SCO Packet over HCI received Dropping\n"); > + kfree(skb); > + return 0; > + } > + BT_DBG("hu %p skb %p", hu, skb); > + > + /* Prepend skb with frame type */ > + memcpy(skb_push(skb, 1), &bt_cb(skb)->pkt_type, 1); > + if (ath->num_cmds_complete <= 0 > + && HCI_COMMAND_PKT == bt_cb(skb)->pkt_type) { > + skb_queue_tail(&ath->tx_cmd_wait_q, skb); > + return 0; > + } > + > + skb_queue_tail(&ath->txq, skb); > + set_bit(HCI_UART_SENDING, &hu->tx_state); > + > + schedule_work(&ath->ctxtsw); > + return 0; > +} > + > +static struct sk_buff *ath_dequeue(struct hci_uart *hu) > +{ > + struct ath_struct *ath; > + struct sk_buff *skbuff; > + ath = hu->priv; > + skbuff = skb_dequeue(&ath->txq); > + if (NULL != skbuff) { > + ath_handle_host_data(ath, bt_cb(skbuff)->pkt_type, > + &skbuff->data[1], skbuff->len - 1); > + } > + return skbuff; > +} > + > +static inline int ath_check_data_len(struct ath_struct *ath, int len) > +{ > + register int room = skb_tailroom(ath->rx_skb); > + BT_DBG("len %d room %d", len, room); > + if (!len) { > + hci_recv_frame(ath->rx_skb); > + } else if (len > room) { > + BT_ERR("Data length is too large"); > + kfree_skb(ath->rx_skb); > + } else { > + ath->rx_state = HCIATH_W4_DATA; > + ath->rx_count = len; > + return len; > + } > + ath->rx_state = HCIATH_W4_PACKET_TYPE; > + ath->rx_skb = NULL; > + ath->rx_count = 0; > + return 0; > +} > + > +/* Recv data */ > +static int ath_recv(struct hci_uart *hu, void *data, int count) > +{ > + struct ath_struct *ath = hu->priv; > + register char *ptr; > + struct hci_event_hdr *eh; > + struct hci_acl_hdr *ah; > + struct hci_sco_hdr *sh; > + struct sk_buff *skbuff; > + register int len, type, dlen; > + skbuff = NULL; > + BT_DBG("hu %p count %d rx_state %d rx_count %d", hu, count, > + ath->rx_state, ath->rx_count); > + ptr = data; > + while (count) { > + if (ath->rx_count) { > + len = min_t(unsigned int, ath->rx_count, count); > + memcpy(skb_put(ath->rx_skb, len), ptr, len); > + ath->rx_count -= len; > + count -= len; > + ptr += len; > + if (ath->rx_count) > + continue; > + switch (ath->rx_state) { > + case HCIATH_W4_DATA: > + ath_handle_data_from_controller(ath, > + bt_cb > + (ath->rx_skb)-> > + pkt_type, > + ath->rx_skb-> > + data, > + ath->rx_skb-> > + len); > + if (HCI_EVENT_PKT == > + bt_cb(ath->rx_skb)->pkt_type > + && ath->num_cmds_complete > 0) { > + > + skbuff = > + skb_dequeue(&ath->tx_cmd_wait_q); > + if (skbuff != NULL) { > + skb_queue_tail(&ath->txq, > + skbuff); > + schedule_work(&ath->ctxtsw); > + } > + } > + if (ath_verify_event_discardable > + (hu, bt_cb(ath->rx_skb)->pkt_type, > + ath->rx_skb->data)) { > + kfree(ath->rx_skb); > + ath->rx_skb = NULL; > + } else { > + hci_recv_frame(ath->rx_skb); > + } > + ath->rx_state = HCIATH_W4_PACKET_TYPE; > + ath->rx_skb = NULL; > + ath->rx_count = 0; > + continue; > + case HCIATH_W4_EVENT_HDR: > + eh = (struct hci_event_hdr *)ath->rx_skb->data; > + BT_DBG("Event header: evt 0x%2.2x plen %d", > + eh->evt, eh->plen); > + ath_check_data_len(ath, eh->plen); > + continue; > + case HCIATH_W4_ACL_HDR: > + ah = (struct hci_acl_hdr *)ath->rx_skb->data; > + dlen = __le16_to_cpu(ah->dlen); > + BT_DBG("ACL header: dlen %d", dlen); > + ath_check_data_len(ath, dlen); > + continue; > + case HCIATH_W4_SCO_HDR: > + sh = (struct hci_sco_hdr *)ath->rx_skb->data; > + BT_DBG("SCO header: dlen %d", sh->dlen); > + ath_check_data_len(ath, sh->dlen); > + continue; > + } > + } > + > + /* HCIATH_W4_PACKET_TYPE */ > + switch (*ptr) { > + case HCI_EVENT_PKT: > + BT_DBG("Event packet"); > + ath->rx_state = HCIATH_W4_EVENT_HDR; > + ath->rx_count = HCI_EVENT_HDR_SIZE; > + type = HCI_EVENT_PKT; > + break; > + case HCI_ACLDATA_PKT: > + BT_DBG("ACL packet"); > + ath->rx_state = HCIATH_W4_ACL_HDR; > + ath->rx_count = HCI_ACL_HDR_SIZE; > + type = HCI_ACLDATA_PKT; > + break; > + case HCI_SCODATA_PKT: > + BT_DBG("SCO packet"); > + ath->rx_state = HCIATH_W4_SCO_HDR; > + ath->rx_count = HCI_SCO_HDR_SIZE; > + type = HCI_SCODATA_PKT; > + break; > + default: > + BT_ERR("Unknown HCI packet type %2.2x", (__u8) *ptr); > + hu->hdev->stat.err_rx++; > + ptr++; > + count--; > + continue; > + }; > + ptr++; > + count--; > + > + /* Allocate packet */ > + ath->rx_skb = bt_skb_alloc(HCI_MAX_FRAME_SIZE, GFP_ATOMIC); > + if (!ath->rx_skb) { > + BT_ERR("Can't allocate mem for new packet"); > + ath->rx_state = HCIATH_W4_PACKET_TYPE; > + ath->rx_count = 0; > + return 0; > + } > + ath->rx_skb->dev = (void *)hu->hdev; > + bt_cb(ath->rx_skb)->pkt_type = type; > + } return count; > +} > + > +static void ath_controller_sleep_mode(struct hci_uart *hu, bool enable) > +{ > + unsigned char sleepcmd[] = { 0x01, 0x04, 0xFC, 0x01, 0x00 }; > + sleepcmd[4] = enable; > + ath_write_data_to_cntrlr(hu->hdev, sleepcmd, sizeof(sleepcmd)); > +} > + > +int ath_wakeup_ar3001(struct tty_struct *tty) > +{ > + struct termios settings; > + int status = 0x00; > + mm_segment_t oldfs; > + status = tty->driver->ops->tiocmget(tty, NULL); > + > + if ((status & TIOCM_CTS)) > + return status; > + > + oldfs = get_fs(); > + set_fs(KERNEL_DS); > + n_tty_ioctl_helper(tty, NULL, TCGETS, (unsigned long)&settings); > + > + settings.c_cflag &= ~CRTSCTS; > + n_tty_ioctl_helper(tty, NULL, TCSETS, (unsigned long)&settings); > + set_fs(oldfs); > + status = tty->driver->ops->tiocmget(tty, NULL); > + > + tty->driver->ops->tiocmset(tty, NULL, 0x00, TIOCM_RTS); > + mdelay(20); > + > + status = tty->driver->ops->tiocmget(tty, NULL); > + > + tty->driver->ops->tiocmset(tty, NULL, TIOCM_RTS, 0x00); > + mdelay(20); > + > + status = tty->driver->ops->tiocmget(tty, NULL); > + oldfs = get_fs(); > + set_fs(KERNEL_DS); > + n_tty_ioctl_helper(tty, NULL, TCGETS, (unsigned long)&settings); > + > + settings.c_cflag |= CRTSCTS; > + n_tty_ioctl_helper(tty, NULL, TCSETS, (unsigned long)&settings); > + set_fs(oldfs); > + return status; > +} > + > +int ath_fullboot_config(struct hci_uart *hu, int current_event) > +{ > + struct sk_buff *skbuf; > + struct ath_struct *ath = hu->priv; > + static int autosleep; > + unsigned char rstevt[] = { 0x1, 0x3, 0xc, 0x0 }; > + if (current_event == HCI_RESET) { > + > + if (ath->cur_sleep) { > + autosleep = 1; > + ath_controller_sleep_mode(hu, 1); > + return 1; > + } else { > + return 0; > + } > + } > + > + if (current_event == HCI_SET_SLEEP_MODE) { > + > + if (autosleep == 0) > + return 1; > + > + while ((skbuf = skb_dequeue(&ath->tx_wait_q))) > + skb_queue_tail(&ath->txq, skbuf); > + > + ath_write_data_to_host(hu->hdev, rstevt, sizeof(rstevt)); > + autosleep = 0; > + return 1; > + } > + return 0; > +} > + > +int ath_write_data_to_host(void *dev, unsigned char *data, u8 length) > +{ > + struct sk_buff *skbuf; > + struct hci_dev *hdev; > + hdev = (struct hci_dev *)dev; > + skbuf = bt_skb_alloc(length, GFP_ATOMIC); > + if (!skbuf) { > + BT_ERR("Memory allocation failed"); > + return -1; > + } > + skb_orphan(skbuf); > + > + /* First byte will be added as packet type */ > + bt_cb(skbuf)->pkt_type = data[0]; > + skbuf->dev = (void *)hdev; > + memcpy(skb_put(skbuf, length - 1), &data[1], length - 1); > + hci_recv_frame(skbuf); > + return length; > +} > + > +int ath_write_data_to_cntrlr(void *dev, unsigned char *Data, u8 len) > +{ > + struct sk_buff *skbuff; > + struct ath_struct *ath; > + struct hci_uart *hu; > + struct hci_dev *hdev; > + if (NULL == dev) { > + BT_DBG("NULL handle received %p \n", dev); > + return 0; > + } > + hdev = (struct hci_dev *)dev; > + hu = (struct hci_uart *)hdev->driver_data; > + if (hu == NULL) { > + BT_DBG("NULL handle received %p \n", hdev); > + return 0; > + } > + ath = hu->priv; > + if (ath == NULL) { > + BT_DBG("NULL handle received \n"); > + return 0; > + } > + skbuff = bt_skb_alloc(len, GFP_ATOMIC); > + if (skbuff == NULL) { > + BT_DBG("Malloc Fail memory %p \n", skbuff); > + return 0; > + } > + skb_orphan(skbuff); > + > + if (len != 0) > + memcpy(skb_put(skbuff, len), Data, len); > + > + bt_cb(skbuff)->pkt_type = HCI_COMMAND_PKT; > + skbuff->dev = dev; > + if (ath->num_cmds_complete > 0) { > + skb_queue_tail(&ath->txq, skbuff); > + schedule_work(&ath->ctxtsw); > + } else { > + skb_queue_tail(&ath->tx_cmd_wait_q, skbuff); > + } > + return len; > +} > + > +int ath_check_sleep_cmd(struct ath_struct *ath, unsigned char *packet) > +{ > + if (packet[0] == HCI_EVENT_PKT && packet[1] == 0xFC) > + ath->cur_sleep = packet[3]; > + > + return 0; > +} > + > +int ath_verify_event_discardable(struct hci_uart *hu, unsigned char pkt_type, > + unsigned char *packet) > +{ > + if (pkt_type != HCI_EVENT_PKT) > + return 0; > + > + switch (packet[0]) { > + case 0x0E: /* Command Complete Event */ > + if (packet[3] == 0x03 && packet[4] == 0x0C) { > + > + /* Command complete for HCI Reset Received */ > + return ath_fullboot_config(hu, HCI_RESET); > + } else if (packet[3] == 0x04 && packet[4] == 0xFC) { > + return ath_fullboot_config(hu, HCI_SET_SLEEP_MODE); > + } > + break; > + } > + return 0; > +} > + > +/* Update the number of commands that can be sent to controller */ > +void ath_handle_hci_cmd(struct ath_struct *ath, u8 * packet) > +{ > + spin_lock(&ath->hciath_lock); > + ath->num_cmds_complete--; > + spin_unlock(&ath->hciath_lock); > +} > + > +void ath_handle_hci_event(struct ath_struct *ath, u8 * packet) > +{ > + switch (packet[0]) { > + case 0x05: /* ACL Disconnection Complete Event */ > + break; > + case 0x03: /* ACL Connection Complete Event */ > + break; > + case 0x0E: /* Command Complete Event */ > + spin_lock(&ath->hciath_lock); > + ath->num_cmds_complete = packet[2]; > + spin_unlock(&ath->hciath_lock); > + break; > + case 0x0F: /* Command Status Event */ > + spin_lock(&ath->hciath_lock); > + ath->num_cmds_complete = packet[3]; > + spin_unlock(&ath->hciath_lock); > + break; > + } > +} > + > +void ath_handle_host_data(struct ath_struct *ath, u8 pktType, u8 * packet, > + unsigned int len) > +{ > + switch (pktType) { > + case HCI_ACLDATA_PKT: /* ACL packets */ > + break; > + case HCI_COMMAND_PKT: /* HCI Commands */ > + ath_handle_hci_cmd(ath, packet); > + ath_check_sleep_cmd(ath, packet); > + break; > + } > +} > + > +void ath_handle_data_from_controller(struct ath_struct *ath, u8 pktType, > + u8 *packet, unsigned int len) > +{ > + switch (pktType) { > + case HCI_ACLDATA_PKT: /* ACL packets */ > + break; > + case HCI_EVENT_PKT: /* HCI Events */ > + ath_handle_hci_event(ath, packet); > + break; > + } > +} > + > +static struct hci_uart_proto athp = { > + .id = HCI_UART_ATH, > + .open = ath_open, > + .close = ath_close, > + .recv = ath_recv, > + .enqueue = ath_enqueue, > + .dequeue = ath_dequeue, > + .flush = ath_flush, > +}; > + > +int ath_init(void) > +{ > + int err = hci_uart_register_proto(&athp); > + if (!err) > + BT_INFO("HCIATH protocol initialized"); > + > + else > + BT_ERR("HCIATH protocol registration failed"); > + return err; > +} > + > +int ath_deinit(void) > +{ > + return hci_uart_unregister_proto(&athp); > +} > diff --git a/drivers/bluetooth/hci_ath.h b/drivers/bluetooth/hci_ath.h > new file mode 100755 > index 0000000..7513a89 > --- /dev/null > +++ b/drivers/bluetooth/hci_ath.h > @@ -0,0 +1,72 @@ > +/* > + * Copyright (c) 2009-2010 Atheros Communications Inc. > + * > + * 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. > + * > + * You should have received a copy of the GNU General Public License > + * along with this program; if not, write to the Free Software > + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA > + * > + */ > + > +#ifndef __HCI_ATH_H_ > +#define __HCI_ATH_H_ > + > +#include > +#include > +#include > +#include > +#include > + > +#include > +#include > +#include > +#include > + > +#include > +#include > + > +/* HCIATH receiver States */ > +#define HCIATH_W4_PACKET_TYPE 0 > +#define HCIATH_W4_EVENT_HDR 1 > +#define HCIATH_W4_ACL_HDR 2 > +#define HCIATH_W4_SCO_HDR 3 > +#define HCIATH_W4_DATA 4 > + > +#define HCI_SET_SLEEP_MODE 0xFC04 > +#define HCI_RESET 0x030C > + struct ath_struct { > + struct hci_uart *hu; > + unsigned int rx_state; > + unsigned int rx_count; > + unsigned int cur_sleep; > + int num_cmds_complete; > + spinlock_t hciath_lock; > + struct sk_buff *rx_skb; > + struct sk_buff_head txq; > + struct sk_buff_head tx_wait_q; > + struct sk_buff_head tx_cmd_wait_q; > + wait_queue_head_t wqevt; > + struct work_struct ctxtsw; > +}; > +int ath_write_data_to_cntrlr(void *dev, unsigned char *Data, u8 len); > +int ath_write_data_to_host(void *dev, unsigned char *data, u8 length); > +int ath_wakeup_ar3001(struct tty_struct *tty); > +int ath_verify_event_discardable(struct hci_uart *hu, unsigned char pkt_type, > + unsigned char *packet); > +int ath_check_sleep_cmd(struct ath_struct *ath, unsigned char *packet); > +void ath_handle_host_data(struct ath_struct *ath, u8 pktType, u8 *packet, > + unsigned int len); > +void ath_handle_data_from_controller(struct ath_struct *ath, u8 pktType, > + u8 *packet, unsigned int len); > + > + > +#endif /* __HCI_ATH_H_ */ > diff --git a/drivers/bluetooth/hci_ldisc.c b/drivers/bluetooth/hci_ldisc.c > index aa09193..bd16130 100644 > --- a/drivers/bluetooth/hci_ldisc.c > +++ b/drivers/bluetooth/hci_ldisc.c > @@ -542,6 +542,9 @@ static int __init hci_uart_init(void) > #ifdef CONFIG_BT_HCIUART_LL > ll_init(); > #endif > +#ifdef CONFIG_BT_HCIUART_ATH > + ath_init(); > +#endif > > return 0; > } > @@ -559,6 +562,9 @@ static void __exit hci_uart_exit(void) > #ifdef CONFIG_BT_HCIUART_LL > ll_deinit(); > #endif > +#ifdef CONFIG_BT_HCIUART_ATH > + ath_deinit(); > +#endif > > /* Release tty registration of line discipline */ > if ((err = tty_unregister_ldisc(N_HCI))) > diff --git a/drivers/bluetooth/hci_uart.h b/drivers/bluetooth/hci_uart.h > index 50113db..385537f 100644 > --- a/drivers/bluetooth/hci_uart.h > +++ b/drivers/bluetooth/hci_uart.h > @@ -33,13 +33,14 @@ > #define HCIUARTGETDEVICE _IOR('U', 202, int) > > /* UART protocols */ > -#define HCI_UART_MAX_PROTO 5 > +#define HCI_UART_MAX_PROTO 6 > > #define HCI_UART_H4 0 > #define HCI_UART_BCSP 1 > #define HCI_UART_3WIRE 2 > #define HCI_UART_H4DS 3 > #define HCI_UART_LL 4 > +#define HCI_UART_ATH 5 > > struct hci_uart; > > @@ -91,3 +92,8 @@ int bcsp_deinit(void); > int ll_init(void); > int ll_deinit(void); > #endif > + > +#ifdef CONFIG_BT_HCIUART_ATH > +int ath_init(void); > +int ath_deinit(void); > +#endif > -- > 1.6.3.3 >