mirror of https://github.com/OpenIPC/firmware.git
				
				
				
			
		
			
				
	
	
		
			613 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Diff
		
	
	
			
		
		
	
	
			613 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Diff
		
	
	
| diff -drupN a/drivers/bluetooth/hci_ldisc.c b/drivers/bluetooth/hci_ldisc.c
 | |
| --- a/drivers/bluetooth/hci_ldisc.c	2018-08-06 17:23:04.000000000 +0300
 | |
| +++ b/drivers/bluetooth/hci_ldisc.c	2022-06-12 05:28:14.000000000 +0300
 | |
| @@ -40,6 +40,7 @@
 | |
|  #include <linux/signal.h>
 | |
|  #include <linux/ioctl.h>
 | |
|  #include <linux/skbuff.h>
 | |
| +#include <linux/version.h>
 | |
|  #include <linux/firmware.h>
 | |
|  
 | |
|  #include <net/bluetooth/bluetooth.h>
 | |
| @@ -49,9 +50,23 @@
 | |
|  #include "btbcm.h"
 | |
|  #include "hci_uart.h"
 | |
|  
 | |
| -#define VERSION "2.3"
 | |
| +#ifdef BTCOEX
 | |
| +#include "rtk_coex.h"
 | |
| +#endif
 | |
|  
 | |
| -static const struct hci_uart_proto *hup[HCI_UART_MAX_PROTO];
 | |
| +#define VERSION "2.2.d448471.20181218-163903"
 | |
| +
 | |
| +#if HCI_VERSION_CODE > KERNEL_VERSION(3, 4, 0)
 | |
| +#define GET_DRV_DATA(x)		hci_get_drvdata(x)
 | |
| +#else
 | |
| +#define GET_DRV_DATA(x)		(struct hci_uart *)(x->driver_data)
 | |
| +#endif
 | |
| +
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0)
 | |
| +static int reset;
 | |
| +#endif
 | |
| +
 | |
| +static struct hci_uart_proto *hup[HCI_UART_MAX_PROTO];
 | |
|  
 | |
|  int hci_uart_register_proto(const struct hci_uart_proto *p)
 | |
|  {
 | |
| @@ -114,8 +129,12 @@ static inline struct sk_buff *hci_uart_d
 | |
|  	struct sk_buff *skb = hu->tx_skb;
 | |
|  
 | |
|  	if (!skb) {
 | |
| +		read_lock(&hu->proto_lock);
 | |
| +
 | |
|  		if (test_bit(HCI_UART_PROTO_READY, &hu->flags))
 | |
|  			skb = hu->proto->dequeue(hu);
 | |
| +
 | |
| +		read_unlock(&hu->proto_lock);
 | |
|  	} else {
 | |
|  		hu->tx_skb = NULL;
 | |
|  	}
 | |
| @@ -123,20 +142,48 @@ static inline struct sk_buff *hci_uart_d
 | |
|  	return skb;
 | |
|  }
 | |
|  
 | |
| +/* This may be called in an IRQ context */
 | |
|  int hci_uart_tx_wakeup(struct hci_uart *hu)
 | |
|  {
 | |
| +	/* If acquiring lock fails we assume the tty is being closed because
 | |
| +	 * that is the only time the write lock is acquired. If, however,
 | |
| +	 * at some point in the future the write lock is also acquired in
 | |
| +	 * other situations, then this must be revisited.
 | |
| +	 */
 | |
| +	if (!read_trylock(&hu->proto_lock)) {
 | |
| +		if (in_interrupt())
 | |
| +			return 0;
 | |
| +		read_lock(&hu->proto_lock);
 | |
| +	}
 | |
| +
 | |
| +	/* proto_lock is locked */
 | |
|  	if (!test_bit(HCI_UART_PROTO_READY, &hu->flags))
 | |
| -		return 0;
 | |
| +		goto no_schedule;
 | |
|  
 | |
| +	if (!spin_trylock(&hu->tx_lock)) {
 | |
| +		if (in_interrupt()) {
 | |
| +			schedule_work(&hu->write_work);
 | |
| +			read_unlock(&hu->proto_lock);
 | |
| +			return 0;
 | |
| +		} else {
 | |
| +			spin_lock(&hu->tx_lock);
 | |
| +		}
 | |
| +	}
 | |
| +	/* tx_lock is locked */
 | |
|  	if (test_and_set_bit(HCI_UART_SENDING, &hu->tx_state)) {
 | |
|  		set_bit(HCI_UART_TX_WAKEUP, &hu->tx_state);
 | |
| -		return 0;
 | |
| +		spin_unlock(&hu->tx_lock);
 | |
| +		goto no_schedule;
 | |
|  	}
 | |
| +	spin_unlock(&hu->tx_lock);
 | |
|  
 | |
|  	BT_DBG("");
 | |
|  
 | |
|  	schedule_work(&hu->write_work);
 | |
|  
 | |
| +no_schedule:
 | |
| +	read_unlock(&hu->proto_lock);
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| @@ -171,10 +218,14 @@ restart:
 | |
|  		kfree_skb(skb);
 | |
|  	}
 | |
|  
 | |
| -	if (test_bit(HCI_UART_TX_WAKEUP, &hu->tx_state))
 | |
| +	spin_lock(&hu->tx_lock);
 | |
| +	if (test_bit(HCI_UART_TX_WAKEUP, &hu->tx_state)) {
 | |
| +		spin_unlock(&hu->tx_lock);
 | |
|  		goto restart;
 | |
| +	}
 | |
|  
 | |
|  	clear_bit(HCI_UART_SENDING, &hu->tx_state);
 | |
| +	spin_unlock(&hu->tx_lock);
 | |
|  }
 | |
|  
 | |
|  static void hci_uart_init_work(struct work_struct *work)
 | |
| @@ -213,6 +264,15 @@ static int hci_uart_open(struct hci_dev
 | |
|  	BT_DBG("%s %p", hdev->name, hdev);
 | |
|  
 | |
|  	/* Nothing to do for UART driver */
 | |
| +
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(4, 4, 0)
 | |
| +	set_bit(HCI_RUNNING, &hdev->flags);
 | |
| +#endif
 | |
| +
 | |
| +#ifdef BTCOEX
 | |
| +	rtk_btcoex_open(hdev);
 | |
| +#endif
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| @@ -232,9 +292,13 @@ static int hci_uart_flush(struct hci_dev
 | |
|  	tty_ldisc_flush(tty);
 | |
|  	tty_driver_flush_buffer(tty);
 | |
|  
 | |
| +	read_lock(&hu->proto_lock);
 | |
| +
 | |
|  	if (test_bit(HCI_UART_PROTO_READY, &hu->flags))
 | |
|  		hu->proto->flush(hu);
 | |
|  
 | |
| +	read_unlock(&hu->proto_lock);
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| @@ -243,26 +307,87 @@ static int hci_uart_close(struct hci_dev
 | |
|  {
 | |
|  	BT_DBG("hdev %p", hdev);
 | |
|  
 | |
| +
 | |
| +	/* When in kernel 4.4.0 and greater, the HCI_RUNNING bit is
 | |
| +	 * cleared in hci_dev_do_close(). */
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(4, 4, 0)
 | |
| +	if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags))
 | |
| +		return 0;
 | |
| +#else
 | |
| +	if (test_bit(HCI_RUNNING, &hdev->flags))
 | |
| +		BT_ERR("HCI_RUNNING is not cleared before.");
 | |
| +#endif
 | |
| +
 | |
|  	hci_uart_flush(hdev);
 | |
|  	hdev->flush = NULL;
 | |
| +
 | |
| +#ifdef BTCOEX
 | |
| +	rtk_btcoex_close();
 | |
| +#endif
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
|  /* Send frames from HCI layer */
 | |
| -static int hci_uart_send_frame(struct hci_dev *hdev, struct sk_buff *skb)
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 13, 0)
 | |
| +int hci_uart_send_frame(struct sk_buff *skb)
 | |
| +#else
 | |
| +int hci_uart_send_frame(struct hci_dev *hdev, struct sk_buff *skb)
 | |
| +#endif
 | |
|  {
 | |
| -	struct hci_uart *hu = hci_get_drvdata(hdev);
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 13, 0)
 | |
| +	struct hci_dev *hdev = (struct hci_dev *)skb->dev;
 | |
| +#endif
 | |
| +	struct hci_uart *hu;
 | |
|  
 | |
| -	BT_DBG("%s: type %d len %d", hdev->name, hci_skb_pkt_type(skb),
 | |
| +	if (!hdev) {
 | |
| +		BT_ERR("Frame for unknown device (hdev=NULL)");
 | |
| +		return -ENODEV;
 | |
| +	}
 | |
| +
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(4, 4, 0)
 | |
| +	if (!test_bit(HCI_RUNNING, &hdev->flags))
 | |
| +		return -EBUSY;
 | |
| +#endif
 | |
| +
 | |
| +	hu = GET_DRV_DATA(hdev);	//(struct hci_uart *) hdev->driver_data;
 | |
| +
 | |
| +	BT_DBG("%s: type %d len %d", hdev->name, bt_cb(skb)->pkt_type,
 | |
|  	       skb->len);
 | |
|  
 | |
| +#ifdef BTCOEX
 | |
| +	if (bt_cb(skb)->pkt_type == HCI_COMMAND_PKT)
 | |
| +		rtk_btcoex_parse_cmd(skb->data, skb->len);
 | |
| +	if (bt_cb(skb)->pkt_type == HCI_ACLDATA_PKT)
 | |
| +		rtk_btcoex_parse_l2cap_data_tx(skb->data, skb->len);
 | |
| +#endif
 | |
| +
 | |
| +	read_lock(&hu->proto_lock);
 | |
| +
 | |
| +	if (!test_bit(HCI_UART_PROTO_READY, &hu->flags)) {
 | |
| +		read_unlock(&hu->proto_lock);
 | |
| +		return -EUNATCH;
 | |
| +	}
 | |
| +
 | |
|  	hu->proto->enqueue(hu, skb);
 | |
| +	read_unlock(&hu->proto_lock);
 | |
|  
 | |
|  	hci_uart_tx_wakeup(hu);
 | |
|  
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0)
 | |
| +static void hci_uart_destruct(struct hci_dev *hdev)
 | |
| +{
 | |
| +	if (!hdev)
 | |
| +		return;
 | |
| +
 | |
| +	BT_DBG("%s", hdev->name);
 | |
| +	kfree(hdev->driver_data);
 | |
| +}
 | |
| +#endif
 | |
| +
 | |
|  /* Flow control or un-flow control the device */
 | |
|  void hci_uart_set_flow_control(struct hci_uart *hu, bool enable)
 | |
|  {
 | |
| @@ -445,12 +570,29 @@ done:
 | |
|   */
 | |
|  static int hci_uart_tty_open(struct tty_struct *tty)
 | |
|  {
 | |
| -	struct hci_uart *hu;
 | |
| +	struct hci_uart *hu = (void *)tty->disc_data;
 | |
|  
 | |
|  	BT_DBG("tty %p", tty);
 | |
|  
 | |
| +	/* But nothing ensures disc_data to be NULL. And since ld->ops->open
 | |
| +	 * shall be called only once, we do not need the check at all.
 | |
| +	 * So remove it.
 | |
| +	 *
 | |
| +	 * Note that this is not an issue now, but n_tty will start using the
 | |
| +	 * disc_data pointer and this invalid 'if' would trigger then rendering
 | |
| +	 * TTYs over BT unusable.
 | |
| +	 */
 | |
| +#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 8, 0)
 | |
| +	/* FIXME: This btw is bogus, nothing requires the old ldisc to clear
 | |
| +	 * the pointer
 | |
| +	 */
 | |
| +	if (hu)
 | |
| +		return -EEXIST;
 | |
| +#endif
 | |
| +
 | |
|  	/* Error if the tty has no write op instead of leaving an exploitable
 | |
| -	   hole */
 | |
| +	 * hole
 | |
| +	 */
 | |
|  	if (tty->ops->write == NULL)
 | |
|  		return -EOPNOTSUPP;
 | |
|  
 | |
| @@ -467,7 +609,16 @@ static int hci_uart_tty_open(struct tty_
 | |
|  	INIT_WORK(&hu->init_ready, hci_uart_init_work);
 | |
|  	INIT_WORK(&hu->write_work, hci_uart_write_work);
 | |
|  
 | |
| -	/* Flush any pending characters in the driver */
 | |
| +	rwlock_init(&hu->proto_lock);
 | |
| +	spin_lock_init(&hu->tx_lock);
 | |
| +
 | |
| +	/* Flush any pending characters in the driver and line discipline. */
 | |
| +
 | |
| +	/* FIXME: why is this needed. Note don't use ldisc_ref here as the
 | |
| +	   open path is before the ldisc is referencable */
 | |
| +
 | |
| +	if (tty->ldisc->ops->flush_buffer)
 | |
| +		tty->ldisc->ops->flush_buffer(tty);
 | |
|  	tty_driver_flush_buffer(tty);
 | |
|  
 | |
|  	return 0;
 | |
| @@ -480,8 +631,9 @@ static int hci_uart_tty_open(struct tty_
 | |
|   */
 | |
|  static void hci_uart_tty_close(struct tty_struct *tty)
 | |
|  {
 | |
| -	struct hci_uart *hu = tty->disc_data;
 | |
| +	struct hci_uart *hu = (void *)tty->disc_data;
 | |
|  	struct hci_dev *hdev;
 | |
| +	unsigned long flags;
 | |
|  
 | |
|  	BT_DBG("tty %p", tty);
 | |
|  
 | |
| @@ -495,7 +647,13 @@ static void hci_uart_tty_close(struct tt
 | |
|  	if (hdev)
 | |
|  		hci_uart_close(hdev);
 | |
|  
 | |
| -	cancel_work_sync(&hu->write_work);
 | |
| +	if (test_bit(HCI_UART_PROTO_READY, &hu->flags)) {
 | |
| +		write_lock_irqsave(&hu->proto_lock, flags);
 | |
| +		clear_bit(HCI_UART_PROTO_READY, &hu->flags);
 | |
| +		write_unlock_irqrestore(&hu->proto_lock, flags);
 | |
| +
 | |
| +		cancel_work_sync(&hu->write_work);
 | |
| +	}
 | |
|  
 | |
|  	if (test_and_clear_bit(HCI_UART_PROTO_READY, &hu->flags)) {
 | |
|  		if (hdev) {
 | |
| @@ -512,15 +670,15 @@ static void hci_uart_tty_close(struct tt
 | |
|  
 | |
|  /* hci_uart_tty_wakeup()
 | |
|   *
 | |
| - *    Callback for transmit wakeup. Called when low level
 | |
| - *    device driver can accept more send data.
 | |
| + * Callback for transmit wakeup. Called when low level
 | |
| + * device driver can accept more send data.
 | |
|   *
 | |
|   * Arguments:        tty    pointer to associated tty instance data
 | |
|   * Return Value:    None
 | |
|   */
 | |
|  static void hci_uart_tty_wakeup(struct tty_struct *tty)
 | |
|  {
 | |
| -	struct hci_uart *hu = tty->disc_data;
 | |
| +	struct hci_uart *hu = (void *)tty->disc_data;
 | |
|  
 | |
|  	BT_DBG("");
 | |
|  
 | |
| @@ -538,8 +696,8 @@ static void hci_uart_tty_wakeup(struct t
 | |
|  
 | |
|  /* hci_uart_tty_receive()
 | |
|   *
 | |
| - *     Called by tty low level driver when receive data is
 | |
| - *     available.
 | |
| + * Called by tty low level driver when receive data is
 | |
| + * available.
 | |
|   *
 | |
|   * Arguments:  tty          pointer to tty isntance data
 | |
|   *             data         pointer to received data
 | |
| @@ -551,18 +709,20 @@ static void hci_uart_tty_wakeup(struct t
 | |
|  static void hci_uart_tty_receive(struct tty_struct *tty, const u8 *data,
 | |
|  				 char *flags, int count)
 | |
|  {
 | |
| -	struct hci_uart *hu = tty->disc_data;
 | |
| +	struct hci_uart *hu = (void *)tty->disc_data;
 | |
|  
 | |
|  	if (!hu || tty != hu->tty)
 | |
|  		return;
 | |
|  
 | |
| -	if (!test_bit(HCI_UART_PROTO_READY, &hu->flags))
 | |
| +	read_lock(&hu->proto_lock);
 | |
| +
 | |
| +	if (!test_bit(HCI_UART_PROTO_READY, &hu->flags)) {
 | |
| +		read_unlock(&hu->proto_lock);
 | |
|  		return;
 | |
| +	}
 | |
|  
 | |
| -	/* It does not need a lock here as it is already protected by a mutex in
 | |
| -	 * tty caller
 | |
| -	 */
 | |
| -	hu->proto->recv(hu, data, count);
 | |
| +	hu->proto->recv(hu, (void *)data, count);
 | |
| +	read_unlock(&hu->proto_lock);
 | |
|  
 | |
|  	if (hu->hdev)
 | |
|  		hu->hdev->stat.byte_rx += count;
 | |
| @@ -574,7 +734,7 @@ static int hci_uart_register_dev(struct
 | |
|  {
 | |
|  	struct hci_dev *hdev;
 | |
|  
 | |
| -	BT_DBG("");
 | |
| +	BT_INFO("hci_uart_register_dev");
 | |
|  
 | |
|  	/* Initialize and register HCI device */
 | |
|  	hdev = hci_alloc_dev();
 | |
| @@ -585,39 +745,73 @@ static int hci_uart_register_dev(struct
 | |
|  
 | |
|  	hu->hdev = hdev;
 | |
|  
 | |
| +#if HCI_VERSION_CODE > KERNEL_VERSION(2, 6, 33)
 | |
|  	hdev->bus = HCI_UART;
 | |
| -	hci_set_drvdata(hdev, hu);
 | |
| +#else
 | |
| +	hdev->type = HCI_UART;
 | |
| +#endif
 | |
|  
 | |
| -	/* Only when vendor specific setup callback is provided, consider
 | |
| -	 * the manufacturer information valid. This avoids filling in the
 | |
| -	 * value for Ericsson when nothing is specified.
 | |
| -	 */
 | |
| -	if (hu->proto->setup)
 | |
| -		hdev->manufacturer = hu->proto->manufacturer;
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 4, 0)
 | |
| +	hci_set_drvdata(hdev, hu);
 | |
| +#else
 | |
| +	hdev->driver_data = hu;
 | |
| +#endif
 | |
|  
 | |
|  	hdev->open  = hci_uart_open;
 | |
|  	hdev->close = hci_uart_close;
 | |
|  	hdev->flush = hci_uart_flush;
 | |
| -	hdev->send  = hci_uart_send_frame;
 | |
| -	hdev->setup = hci_uart_setup;
 | |
| +	hdev->send = hci_uart_send_frame;
 | |
| +
 | |
| +	/* NOTE: No hdev->setup setting for Realtek BTUART because
 | |
| +	 * the download procedure is done with rtk_hciattach in userspace
 | |
| +	 * before this function called in hci_uart_set_proto()
 | |
| +	 */
 | |
| +
 | |
|  	SET_HCIDEV_DEV(hdev, hu->tty->dev);
 | |
|  
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0)
 | |
| +	hdev->destruct = hci_uart_destruct;
 | |
| +	hdev->owner = THIS_MODULE;
 | |
| +#endif
 | |
| +
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0)
 | |
| +	if (!reset)
 | |
| +		set_bit(HCI_QUIRK_NO_RESET, &hdev->quirks);
 | |
| +#endif
 | |
| +
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(2, 6, 36)
 | |
|  	if (test_bit(HCI_UART_RAW_DEVICE, &hu->hdev_flags))
 | |
|  		set_bit(HCI_QUIRK_RAW_DEVICE, &hdev->quirks);
 | |
| +#endif
 | |
|  
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 17, 0)
 | |
|  	if (test_bit(HCI_UART_EXT_CONFIG, &hu->hdev_flags))
 | |
|  		set_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks);
 | |
| +#endif
 | |
|  
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 4, 0)
 | |
|  	if (!test_bit(HCI_UART_RESET_ON_INIT, &hu->hdev_flags))
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 6, 0)
 | |
|  		set_bit(HCI_QUIRK_RESET_ON_CLOSE, &hdev->quirks);
 | |
| +#else
 | |
| +		set_bit(HCI_QUIRK_NO_RESET, &hdev->quirks);
 | |
| +#endif
 | |
| +#endif
 | |
|  
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 4, 0)
 | |
|  	if (test_bit(HCI_UART_CREATE_AMP, &hu->hdev_flags))
 | |
|  		hdev->dev_type = HCI_AMP;
 | |
|  	else
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(4, 8, 0)
 | |
| +		hdev->dev_type = HCI_BREDR;
 | |
| +#else
 | |
|  		hdev->dev_type = HCI_PRIMARY;
 | |
| +#endif
 | |
| +#endif
 | |
|  
 | |
| -	if (test_bit(HCI_UART_INIT_PENDING, &hu->hdev_flags))
 | |
| -		return 0;
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(4, 1, 0)
 | |
| +	set_bit(HCI_QUIRK_SIMULTANEOUS_DISCOVERY, &hdev->quirks);
 | |
| +#endif
 | |
|  
 | |
|  	if (hci_register_dev(hdev) < 0) {
 | |
|  		BT_ERR("Can't register HCI device");
 | |
| @@ -627,6 +821,10 @@ static int hci_uart_register_dev(struct
 | |
|  
 | |
|  	set_bit(HCI_UART_REGISTERED, &hu->flags);
 | |
|  
 | |
| +#ifdef BTCOEX
 | |
| +	rtk_btcoex_probe(hdev);
 | |
| +#endif
 | |
| +
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| @@ -656,6 +854,7 @@ static int hci_uart_set_proto(struct hci
 | |
|  	return 0;
 | |
|  }
 | |
|  
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 17, 0)
 | |
|  static int hci_uart_set_flags(struct hci_uart *hu, unsigned long flags)
 | |
|  {
 | |
|  	unsigned long valid_flags = BIT(HCI_UART_RAW_DEVICE) |
 | |
| @@ -672,6 +871,7 @@ static int hci_uart_set_flags(struct hci
 | |
|  
 | |
|  	return 0;
 | |
|  }
 | |
| +#endif
 | |
|  
 | |
|  /* hci_uart_tty_ioctl()
 | |
|   *
 | |
| @@ -689,7 +889,7 @@ static int hci_uart_set_flags(struct hci
 | |
|  static int hci_uart_tty_ioctl(struct tty_struct *tty, struct file *file,
 | |
|  			      unsigned int cmd, unsigned long arg)
 | |
|  {
 | |
| -	struct hci_uart *hu = tty->disc_data;
 | |
| +	struct hci_uart *hu = (void *)tty->disc_data;
 | |
|  	int err = 0;
 | |
|  
 | |
|  	BT_DBG("");
 | |
| @@ -702,41 +902,43 @@ static int hci_uart_tty_ioctl(struct tty
 | |
|  	case HCIUARTSETPROTO:
 | |
|  		if (!test_and_set_bit(HCI_UART_PROTO_SET, &hu->flags)) {
 | |
|  			err = hci_uart_set_proto(hu, arg);
 | |
| -			if (err)
 | |
| +			if (err) {
 | |
|  				clear_bit(HCI_UART_PROTO_SET, &hu->flags);
 | |
| +				return err;
 | |
| +			}
 | |
|  		} else
 | |
| -			err = -EBUSY;
 | |
| +			return -EBUSY;
 | |
|  		break;
 | |
|  
 | |
|  	case HCIUARTGETPROTO:
 | |
|  		if (test_bit(HCI_UART_PROTO_SET, &hu->flags))
 | |
| -			err = hu->proto->id;
 | |
| -		else
 | |
| -			err = -EUNATCH;
 | |
| -		break;
 | |
| +			return hu->proto->id;
 | |
| +		return -EUNATCH;
 | |
|  
 | |
|  	case HCIUARTGETDEVICE:
 | |
|  		if (test_bit(HCI_UART_REGISTERED, &hu->flags))
 | |
| -			err = hu->hdev->id;
 | |
| -		else
 | |
| -			err = -EUNATCH;
 | |
| -		break;
 | |
| +			return hu->hdev->id;
 | |
| +		return -EUNATCH;
 | |
|  
 | |
|  	case HCIUARTSETFLAGS:
 | |
|  		if (test_bit(HCI_UART_PROTO_SET, &hu->flags))
 | |
| -			err = -EBUSY;
 | |
| -		else
 | |
| -			err = hci_uart_set_flags(hu, arg);
 | |
| +			return -EBUSY;
 | |
| +#if HCI_VERSION_CODE >= KERNEL_VERSION(3, 17, 0)
 | |
| +		err = hci_uart_set_flags(hu, arg);
 | |
| +		if (err)
 | |
| +			return err;
 | |
| +#else
 | |
| +		hu->hdev_flags = arg;
 | |
| +#endif
 | |
|  		break;
 | |
|  
 | |
|  	case HCIUARTGETFLAGS:
 | |
| -		err = hu->hdev_flags;
 | |
| -		break;
 | |
| +		return hu->hdev_flags;
 | |
|  
 | |
|  	default:
 | |
|  		err = n_tty_ioctl_helper(tty, file, cmd, arg);
 | |
|  		break;
 | |
| -	}
 | |
| +	};
 | |
|  
 | |
|  	return err;
 | |
|  }
 | |
| @@ -805,6 +1007,9 @@ static int __init hci_uart_init(void)
 | |
|  #ifdef CONFIG_BT_HCIUART_3WIRE
 | |
|  	h5_init();
 | |
|  #endif
 | |
| +#ifdef CONFIG_BT_HCIUART_RTL3WIRE
 | |
| +	h5_rtk_init();
 | |
| +#endif
 | |
|  #ifdef CONFIG_BT_HCIUART_INTEL
 | |
|  	intel_init();
 | |
|  #endif
 | |
| @@ -820,6 +1025,9 @@ static int __init hci_uart_init(void)
 | |
|  #ifdef CONFIG_BT_HCIUART_MRVL
 | |
|  	mrvl_init();
 | |
|  #endif
 | |
| +#ifdef BTCOEX
 | |
| +	rtk_btcoex_init();
 | |
| +#endif
 | |
|  
 | |
|  	return 0;
 | |
|  }
 | |
| @@ -843,6 +1051,9 @@ static void __exit hci_uart_exit(void)
 | |
|  #ifdef CONFIG_BT_HCIUART_3WIRE
 | |
|  	h5_deinit();
 | |
|  #endif
 | |
| +#ifdef CONFIG_BT_HCIUART_RTL3WIRE
 | |
| +	h5_rtk_deinit();
 | |
| +#endif
 | |
|  #ifdef CONFIG_BT_HCIUART_INTEL
 | |
|  	intel_deinit();
 | |
|  #endif
 | |
| @@ -863,11 +1074,20 @@ static void __exit hci_uart_exit(void)
 | |
|  	err = tty_unregister_ldisc(N_HCI);
 | |
|  	if (err)
 | |
|  		BT_ERR("Can't unregister HCI line discipline (%d)", err);
 | |
| +
 | |
| +#ifdef BTCOEX
 | |
| +	rtk_btcoex_exit();
 | |
| +#endif
 | |
|  }
 | |
|  
 | |
|  module_init(hci_uart_init);
 | |
|  module_exit(hci_uart_exit);
 | |
|  
 | |
| +#if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0)
 | |
| +module_param(reset, bool, 0644);
 | |
| +MODULE_PARM_DESC(reset, "Send HCI reset command on initialization");
 | |
| +#endif
 | |
| +
 | |
|  MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");
 | |
|  MODULE_DESCRIPTION("Bluetooth HCI UART driver ver " VERSION);
 | |
|  MODULE_VERSION(VERSION);
 |