亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? hci_usb.c

?? 嵌入式Linux的藍牙模塊驅動
?? C
?? 第 1 頁 / 共 2 頁
字號:
/*    HCI USB driver for Linux Bluetooth protocol stack (BlueZ)   Copyright (C) 2000-2001 Qualcomm Incorporated   Written 2000,2001 by Maxim Krasnyansky <maxk@qualcomm.com>   Copyright (C) 2003 Maxim Krasnyansky <maxk@qualcomm.com>   This program is free software; you can redistribute it and/or modify   it under the terms of the GNU General Public License version 2 as   published by the Free Software Foundation;   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS   OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.   IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY   CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES    WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN    ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF    OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.   ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,    COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS    SOFTWARE IS DISCLAIMED.*//* * Bluetooth HCI USB driver. * Based on original USB Bluetooth driver for Linux kernel *    Copyright (c) 2000 Greg Kroah-Hartman        <greg@kroah.com> *    Copyright (c) 2000 Mark Douglas Corner       <mcorner@umich.edu> * */#include <linux/config.h>#include <linux/module.h>#include <linux/kernel.h>#include <linux/init.h>#include <linux/sched.h>#include <linux/unistd.h>#include <linux/types.h>#include <linux/interrupt.h>#include <linux/slab.h>#include <linux/errno.h>#include <linux/string.h>#include <linux/skbuff.h>#include <linux/usb.h>#include <net/bluetooth/bluetooth.h>#include <net/bluetooth/hci_core.h>#include "hci_usb.h"#ifndef CONFIG_BT_HCIUSB_DEBUG#undef  BT_DBG#define BT_DBG(D...)#undef  BT_DMP#define BT_DMP(D...)#endif#ifndef CONFIG_BT_HCIUSB_ZERO_PACKET#undef  URB_ZERO_PACKET#define URB_ZERO_PACKET 0#endif#define VERSION "2.7"static struct usb_driver hci_usb_driver; static struct usb_device_id bluetooth_ids[] = {	/* Generic Bluetooth USB device */	{ USB_DEVICE_INFO(HCI_DEV_CLASS, HCI_DEV_SUBCLASS, HCI_DEV_PROTOCOL) },	/* AVM BlueFRITZ! USB v2.0 */	{ USB_DEVICE(0x057c, 0x3800) },	/* Bluetooth Ultraport Module from IBM */	{ USB_DEVICE(0x04bf, 0x030a) },	/* ALPS Modules with non-standard id */	{ USB_DEVICE(0x044e, 0x3001) },	{ USB_DEVICE(0x044e, 0x3002) },	/* Ericsson with non-standard id */	{ USB_DEVICE(0x0bdb, 0x1002) },	{ }	/* Terminating entry */};MODULE_DEVICE_TABLE (usb, bluetooth_ids);static struct usb_device_id blacklist_ids[] = {	/* Broadcom BCM2033 without firmware */	{ USB_DEVICE(0x0a5c, 0x2033), .driver_info = HCI_IGNORE },	/* Broadcom BCM2035 */	{ USB_DEVICE(0x0a5c, 0x200a), .driver_info = HCI_RESET },	/* ISSC Bluetooth Adapter v3.1 */	{ USB_DEVICE(0x1131, 0x1001), .driver_info = HCI_RESET },	/* Digianswer device */	{ USB_DEVICE(0x08fd, 0x0001), .driver_info = HCI_DIGIANSWER },	/* RTX Telecom based adapter with buggy SCO support */	{ USB_DEVICE(0x0400, 0x0807), .driver_info = HCI_BROKEN_ISOC },	{ }	/* Terminating entry */};struct _urb *_urb_alloc(int isoc, int gfp){	struct _urb *_urb = kmalloc(sizeof(struct _urb) +				sizeof(struct usb_iso_packet_descriptor) * isoc, gfp);	if (_urb) {		memset(_urb, 0, sizeof(*_urb));		usb_init_urb(&_urb->urb);	}	return _urb;}struct _urb *_urb_dequeue(struct _urb_queue *q){	struct _urb *_urb = NULL;	unsigned long flags;	spin_lock_irqsave(&q->lock, flags);	{		struct list_head *head = &q->head;		struct list_head *next = head->next;		if (next != head) {			_urb = list_entry(next, struct _urb, list);			list_del(next); _urb->queue = NULL;		}	}	spin_unlock_irqrestore(&q->lock, flags);	return _urb;}static void hci_usb_rx_complete(struct urb *urb, struct pt_regs *regs);static void hci_usb_tx_complete(struct urb *urb, struct pt_regs *regs);#define __pending_tx(husb, type)  (&husb->pending_tx[type-1])#define __pending_q(husb, type)   (&husb->pending_q[type-1])#define __completed_q(husb, type) (&husb->completed_q[type-1])#define __transmit_q(husb, type)  (&husb->transmit_q[type-1])#define __reassembly(husb, type)  (husb->reassembly[type-1])static inline struct _urb *__get_completed(struct hci_usb *husb, int type){	return _urb_dequeue(__completed_q(husb, type)); }#ifdef CONFIG_BT_HCIUSB_SCOstatic void __fill_isoc_desc(struct urb *urb, int len, int mtu){	int offset = 0, i;	BT_DBG("len %d mtu %d", len, mtu);	for (i=0; i < HCI_MAX_ISOC_FRAMES && len >= mtu; i++, offset += mtu, len -= mtu) {		urb->iso_frame_desc[i].offset = offset;		urb->iso_frame_desc[i].length = mtu;		BT_DBG("desc %d offset %d len %d", i, offset, mtu);	}	if (len && i < HCI_MAX_ISOC_FRAMES) {		urb->iso_frame_desc[i].offset = offset;		urb->iso_frame_desc[i].length = len;		BT_DBG("desc %d offset %d len %d", i, offset, len);		i++;	}	urb->number_of_packets = i;}#endifstatic int hci_usb_intr_rx_submit(struct hci_usb *husb){	struct _urb *_urb;	struct urb *urb;	int err, pipe, interval, size;	void *buf;	BT_DBG("%s", husb->hdev->name);	size = husb->intr_in_ep->desc.wMaxPacketSize;	buf = kmalloc(size, GFP_ATOMIC);	if (!buf)		return -ENOMEM;	_urb = _urb_alloc(0, GFP_ATOMIC);	if (!_urb) {		kfree(buf);		return -ENOMEM;	}	_urb->type = HCI_EVENT_PKT;	_urb_queue_tail(__pending_q(husb, _urb->type), _urb);	urb = &_urb->urb;	pipe     = usb_rcvintpipe(husb->udev, husb->intr_in_ep->desc.bEndpointAddress);	interval = husb->intr_in_ep->desc.bInterval;	usb_fill_int_urb(urb, husb->udev, pipe, buf, size, hci_usb_rx_complete, husb, interval);		err = usb_submit_urb(urb, GFP_ATOMIC);	if (err) {		BT_ERR("%s intr rx submit failed urb %p err %d",				husb->hdev->name, urb, err);		_urb_unlink(_urb);		_urb_free(_urb);		kfree(buf);	}	return err;}static int hci_usb_bulk_rx_submit(struct hci_usb *husb){	struct _urb *_urb;	struct urb *urb;	int err, pipe, size = HCI_MAX_FRAME_SIZE;	void *buf;	buf = kmalloc(size, GFP_ATOMIC);	if (!buf)		return -ENOMEM;	_urb = _urb_alloc(0, GFP_ATOMIC);	if (!_urb) {		kfree(buf);		return -ENOMEM;	}	_urb->type = HCI_ACLDATA_PKT;	_urb_queue_tail(__pending_q(husb, _urb->type), _urb);	urb  = &_urb->urb;	pipe = usb_rcvbulkpipe(husb->udev, husb->bulk_in_ep->desc.bEndpointAddress);	usb_fill_bulk_urb(urb, husb->udev, pipe, buf, size, hci_usb_rx_complete, husb);	urb->transfer_flags = 0;	BT_DBG("%s urb %p", husb->hdev->name, urb);	err = usb_submit_urb(urb, GFP_ATOMIC);	if (err) {		BT_ERR("%s bulk rx submit failed urb %p err %d",				husb->hdev->name, urb, err);		_urb_unlink(_urb);		_urb_free(_urb);		kfree(buf);	}	return err;}#ifdef CONFIG_BT_HCIUSB_SCOstatic int hci_usb_isoc_rx_submit(struct hci_usb *husb){	struct _urb *_urb;	struct urb *urb;	int err, mtu, size;	void *buf;	mtu  = husb->isoc_in_ep->desc.wMaxPacketSize;	size = mtu * HCI_MAX_ISOC_FRAMES;	buf = kmalloc(size, GFP_ATOMIC);	if (!buf)		return -ENOMEM;	_urb = _urb_alloc(HCI_MAX_ISOC_FRAMES, GFP_ATOMIC);	if (!_urb) {		kfree(buf);		return -ENOMEM;	}	_urb->type = HCI_SCODATA_PKT;	_urb_queue_tail(__pending_q(husb, _urb->type), _urb);	urb = &_urb->urb;	urb->context  = husb;	urb->dev      = husb->udev;	urb->pipe     = usb_rcvisocpipe(husb->udev, husb->isoc_in_ep->desc.bEndpointAddress);	urb->complete = hci_usb_rx_complete;	urb->interval = husb->isoc_in_ep->desc.bInterval;	urb->transfer_buffer_length = size;	urb->transfer_buffer = buf;	urb->transfer_flags  = URB_ISO_ASAP;	__fill_isoc_desc(urb, size, mtu);	BT_DBG("%s urb %p", husb->hdev->name, urb);	err = usb_submit_urb(urb, GFP_ATOMIC);	if (err) {		BT_ERR("%s isoc rx submit failed urb %p err %d",				husb->hdev->name, urb, err);		_urb_unlink(_urb);		_urb_free(_urb);		kfree(buf);	}	return err;}#endif/* Initialize device */static int hci_usb_open(struct hci_dev *hdev){	struct hci_usb *husb = (struct hci_usb *) hdev->driver_data;	int i, err;	unsigned long flags;	BT_DBG("%s", hdev->name);	if (test_and_set_bit(HCI_RUNNING, &hdev->flags))		return 0;	write_lock_irqsave(&husb->completion_lock, flags);	err = hci_usb_intr_rx_submit(husb);	if (!err) {		for (i = 0; i < HCI_MAX_BULK_RX; i++)			hci_usb_bulk_rx_submit(husb);#ifdef CONFIG_BT_HCIUSB_SCO		if (husb->isoc_iface)			for (i = 0; i < HCI_MAX_ISOC_RX; i++)				hci_usb_isoc_rx_submit(husb);#endif	} else {		clear_bit(HCI_RUNNING, &hdev->flags);	}	write_unlock_irqrestore(&husb->completion_lock, flags);	return err;}/* Reset device */static int hci_usb_flush(struct hci_dev *hdev){	struct hci_usb *husb = (struct hci_usb *) hdev->driver_data;	int i;	BT_DBG("%s", hdev->name);	for (i = 0; i < 4; i++)		skb_queue_purge(&husb->transmit_q[i]);	return 0;}static void hci_usb_unlink_urbs(struct hci_usb *husb){	int i;	BT_DBG("%s", husb->hdev->name);	for (i = 0; i < 4; i++) {		struct _urb *_urb;		struct urb *urb;		/* Kill pending requests */		while ((_urb = _urb_dequeue(&husb->pending_q[i]))) {			urb = &_urb->urb;			BT_DBG("%s unlinking _urb %p type %d urb %p", 					husb->hdev->name, _urb, _urb->type, urb);			usb_kill_urb(urb);			_urb_queue_tail(__completed_q(husb, _urb->type), _urb);		}		/* Release completed requests */		while ((_urb = _urb_dequeue(&husb->completed_q[i]))) {			urb = &_urb->urb;			BT_DBG("%s freeing _urb %p type %d urb %p",					husb->hdev->name, _urb, _urb->type, urb);			if (urb->setup_packet)				kfree(urb->setup_packet);			if (urb->transfer_buffer)				kfree(urb->transfer_buffer);			_urb_free(_urb);		}		/* Release reassembly buffers */		if (husb->reassembly[i]) {			kfree_skb(husb->reassembly[i]);			husb->reassembly[i] = NULL;		}	}}/* Close device */static int hci_usb_close(struct hci_dev *hdev){	struct hci_usb *husb = (struct hci_usb *) hdev->driver_data;	unsigned long flags;	if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags))		return 0;	BT_DBG("%s", hdev->name);	/* Synchronize with completion handlers */	write_lock_irqsave(&husb->completion_lock, flags);	write_unlock_irqrestore(&husb->completion_lock, flags);	hci_usb_unlink_urbs(husb);	hci_usb_flush(hdev);	return 0;}static int __tx_submit(struct hci_usb *husb, struct _urb *_urb){	struct urb *urb = &_urb->urb;	int err;	BT_DBG("%s urb %p type %d", husb->hdev->name, urb, _urb->type);	_urb_queue_tail(__pending_q(husb, _urb->type), _urb);	err = usb_submit_urb(urb, GFP_ATOMIC);	if (err) {		BT_ERR("%s tx submit failed urb %p type %d err %d",				husb->hdev->name, urb, _urb->type, err);		_urb_unlink(_urb);		_urb_queue_tail(__completed_q(husb, _urb->type), _urb);	} else		atomic_inc(__pending_tx(husb, _urb->type));	return err;}static inline int hci_usb_send_ctrl(struct hci_usb *husb, struct sk_buff *skb){	struct _urb *_urb = __get_completed(husb, skb->pkt_type);	struct usb_ctrlrequest *dr;	struct urb *urb;	if (!_urb) {		_urb = _urb_alloc(0, GFP_ATOMIC);		if (!_urb)			return -ENOMEM;		_urb->type = skb->pkt_type;		dr = kmalloc(sizeof(*dr), GFP_ATOMIC);		if (!dr) {			_urb_free(_urb);			return -ENOMEM;		}	} else		dr = (void *) _urb->urb.setup_packet;	dr->bRequestType = husb->ctrl_req;	dr->bRequest = 0;	dr->wIndex   = 0;	dr->wValue   = 0;	dr->wLength  = __cpu_to_le16(skb->len);	urb = &_urb->urb;	usb_fill_control_urb(urb, husb->udev, usb_sndctrlpipe(husb->udev, 0),		(void *) dr, skb->data, skb->len, hci_usb_tx_complete, husb);	BT_DBG("%s skb %p len %d", husb->hdev->name, skb, skb->len);		_urb->priv = skb;	return __tx_submit(husb, _urb);}static inline int hci_usb_send_bulk(struct hci_usb *husb, struct sk_buff *skb){	struct _urb *_urb = __get_completed(husb, skb->pkt_type);	struct urb *urb;	int pipe;	if (!_urb) {		_urb = _urb_alloc(0, GFP_ATOMIC);		if (!_urb)			return -ENOMEM;		_urb->type = skb->pkt_type;	}	urb  = &_urb->urb;	pipe = usb_sndbulkpipe(husb->udev, husb->bulk_out_ep->desc.bEndpointAddress);	usb_fill_bulk_urb(urb, husb->udev, pipe, skb->data, skb->len, 			hci_usb_tx_complete, husb);	urb->transfer_flags = URB_ZERO_PACKET;	BT_DBG("%s skb %p len %d", husb->hdev->name, skb, skb->len);	_urb->priv = skb;	return __tx_submit(husb, _urb);}#ifdef CONFIG_BT_HCIUSB_SCOstatic inline int hci_usb_send_isoc(struct hci_usb *husb, struct sk_buff *skb){	struct _urb *_urb = __get_completed(husb, skb->pkt_type);	struct urb *urb;	if (!_urb) {		_urb = _urb_alloc(HCI_MAX_ISOC_FRAMES, GFP_ATOMIC);		if (!_urb)			return -ENOMEM;		_urb->type = skb->pkt_type;	}	BT_DBG("%s skb %p len %d", husb->hdev->name, skb, skb->len);	urb = &_urb->urb;	urb->context  = husb;	urb->dev      = husb->udev;	urb->pipe     = usb_sndisocpipe(husb->udev, husb->isoc_out_ep->desc.bEndpointAddress);	urb->complete = hci_usb_tx_complete;	urb->transfer_flags = URB_ISO_ASAP;	urb->interval = husb->isoc_out_ep->desc.bInterval;

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
奇米色777欧美一区二区| 亚洲区小说区图片区qvod| 欧美日韩综合色| 在线观看一区二区视频| 欧美丝袜自拍制服另类| 欧美精品在欧美一区二区少妇| 在线这里只有精品| 欧美日韩日本视频| 555www色欧美视频| 精品国产乱码久久久久久牛牛| 精品久久久久一区二区国产| 精品久久人人做人人爽| 国产丝袜欧美中文另类| 最近日韩中文字幕| 日韩精品三区四区| 韩国视频一区二区| 国产精品1区2区| a级精品国产片在线观看| 欧美中文字幕不卡| 日韩午夜激情电影| 国产精品视频线看| 亚洲国产综合人成综合网站| 另类中文字幕网| 国产成人精品一区二| 色综合久久综合| 91精品综合久久久久久| 久久精品无码一区二区三区| 亚洲精品国产a久久久久久| 日韩精品一区第一页| 成人做爰69片免费看网站| 欧美日韩极品在线观看一区| 精品国产免费人成电影在线观看四季| 国产精品天干天干在线综合| 午夜伊人狠狠久久| 福利一区二区在线| 69久久夜色精品国产69蝌蚪网| 国产亚洲欧美色| 日韩高清不卡在线| 国产99久久久精品| 在线观看91av| 亚洲特级片在线| 国产麻豆一精品一av一免费| 91久久线看在观草草青青| 国产三级一区二区三区| 日本欧美一区二区| 在线观看亚洲精品视频| 国产婷婷色一区二区三区| 日韩av二区在线播放| 日本韩国精品在线| 国产精品久久久久9999吃药| 蜜臀av一区二区在线免费观看| 色综合久久综合中文综合网| 久久久久久电影| 狠狠v欧美v日韩v亚洲ⅴ| 欧美三级电影精品| 亚洲一区二区三区四区在线观看| 成人小视频免费在线观看| 亚洲精品在线三区| 久久97超碰国产精品超碰| 欧美放荡的少妇| 亚洲自拍偷拍综合| 91成人免费在线| 一区二区在线观看免费视频播放 | 99久久婷婷国产综合精品| 日韩欧美国产1| 蜜桃传媒麻豆第一区在线观看| 欧美日韩视频不卡| 午夜视频在线观看一区二区三区| 99国产精品国产精品毛片| 国产精品成人在线观看| 99re成人精品视频| 亚洲日穴在线视频| 91国在线观看| 亚洲妇女屁股眼交7| 欧美丰满嫩嫩电影| 国产成都精品91一区二区三| 精品国产成人系列| 国产精品一区二区91| 久久毛片高清国产| 国产精品自在在线| 国产日韩欧美电影| 97se狠狠狠综合亚洲狠狠| 国产精品久久777777| 99re热视频精品| 亚洲第一精品在线| 日韩一本二本av| 国产高清精品在线| 国产精品福利电影一区二区三区四区| 不卡的电视剧免费网站有什么| 国产精品乱人伦| 欧美伊人精品成人久久综合97| 亚洲一区二区在线视频| 欧美一区二区高清| 国产一区美女在线| 亚洲视频一区在线| 欧美精选一区二区| 国产精品99久久久久久宅男| 国产欧美va欧美不卡在线 | 亚洲视频一二三| 色婷婷亚洲婷婷| 日韩国产欧美在线观看| 久久久久综合网| 色婷婷香蕉在线一区二区| 奇米在线7777在线精品| 国产精品萝li| 欧美一区二区成人6969| av综合在线播放| 蜜臀久久久久久久| 亚洲天堂av一区| 精品欧美黑人一区二区三区| 白白色亚洲国产精品| 天堂成人免费av电影一区| 国产喂奶挤奶一区二区三区| 欧美少妇bbb| 高清在线成人网| 日韩电影在线观看电影| 中文字幕亚洲视频| 久久综合九色综合欧美98| 欧美午夜精品一区二区三区 | 国产精品国产馆在线真实露脸| 欧美日韩午夜精品| a级高清视频欧美日韩| 毛片av一区二区三区| 亚洲图片欧美色图| 亚洲色图丝袜美腿| 国产亚洲精品中文字幕| 91精品国产综合久久精品图片 | 美腿丝袜在线亚洲一区| 一区二区三区在线视频观看| 国产欧美一区二区在线观看| 日韩免费性生活视频播放| 欧美日韩国产另类不卡| 91影院在线免费观看| 丁香另类激情小说| 国产精品66部| 国内精品免费在线观看| 日本va欧美va精品发布| 亚洲成人资源网| 悠悠色在线精品| 亚洲免费成人av| 亚洲色图制服诱惑 | 亚洲成人激情社区| 亚洲欧美一区二区久久| 亚洲欧美日韩系列| 亚洲免费大片在线观看| 亚洲免费成人av| **欧美大码日韩| 亚洲天堂成人网| 日韩理论电影院| 亚洲精品国产a| 亚洲国产人成综合网站| 亚洲丰满少妇videoshd| 午夜精品久久久| 美女mm1313爽爽久久久蜜臀| 久久aⅴ国产欧美74aaa| 国产精品影视网| 成人h精品动漫一区二区三区| thepron国产精品| 在线免费观看日本一区| 色综合色狠狠综合色| 欧美日韩色一区| 日韩午夜精品视频| 国产日韩欧美麻豆| 亚洲视频网在线直播| 亚洲一区二区三区精品在线| 石原莉奈在线亚洲三区| 久久 天天综合| 成人h精品动漫一区二区三区| 色综合亚洲欧洲| 在线成人小视频| 久久免费偷拍视频| 一区二区三区在线免费播放 | 中文字幕av一区二区三区| 亚洲女人的天堂| 视频一区二区三区中文字幕| 激情另类小说区图片区视频区| 国产成人av影院| 在线免费观看日本一区| 欧美r级电影在线观看| 综合久久给合久久狠狠狠97色| 亚洲一区影音先锋| 国产自产高清不卡| 91蜜桃网址入口| 日韩精品专区在线| 国产精品久久久久久妇女6080| 亚洲成人午夜影院| 成人一区二区三区在线观看| 日本精品裸体写真集在线观看| 日韩精品中文字幕一区二区三区| 国产精品久久影院| 免费观看一级欧美片| 91社区在线播放| 精品久久久久久久一区二区蜜臀| 成人免费在线观看入口| 久久精品99国产精品| 欧美亚一区二区| 国产精品久久毛片a| 免费成人小视频| 欧美日韩久久久久久| 日韩美女啊v在线免费观看|