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

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

?? hci_ldisc.c

?? 嵌入式Linux的藍牙模塊驅動
?? C
字號:
/*    BlueZ - Bluetooth protocol stack for Linux   Copyright (C) 2000-2001 Qualcomm Incorporated   Written 2000,2001 by 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 UART driver. * * $Id: hci_ldisc.c,v 1.5 2002/10/02 18:37:20 maxk Exp $     */#define VERSION "2.1"#include <linux/config.h>#include <linux/module.h>#include <linux/kernel.h>#include <linux/init.h>#include <linux/sched.h>#include <linux/types.h>#include <linux/fcntl.h>#include <linux/interrupt.h>#include <linux/ptrace.h>#include <linux/poll.h>#include <linux/slab.h>#include <linux/tty.h>#include <linux/errno.h>#include <linux/string.h>#include <linux/signal.h>#include <linux/ioctl.h>#include <linux/skbuff.h>#include <net/bluetooth/bluetooth.h>#include <net/bluetooth/hci_core.h>#include "hci_uart.h"#ifndef CONFIG_BT_HCIUART_DEBUG#undef  BT_DBG#define BT_DBG( A... )#undef  BT_DMP#define BT_DMP( A... )#endifstatic struct hci_uart_proto *hup[HCI_UART_MAX_PROTO];int hci_uart_register_proto(struct hci_uart_proto *p){	if (p->id >= HCI_UART_MAX_PROTO)		return -EINVAL;	if (hup[p->id])		return -EEXIST;	hup[p->id] = p;	return 0;}int hci_uart_unregister_proto(struct hci_uart_proto *p){	if (p->id >= HCI_UART_MAX_PROTO)		return -EINVAL;	if (!hup[p->id])		return -EINVAL;	hup[p->id] = NULL;	return 0;}static struct hci_uart_proto *hci_uart_get_proto(unsigned int id){	if (id >= HCI_UART_MAX_PROTO)		return NULL;	return hup[id];}static inline void hci_uart_tx_complete(struct hci_uart *hu, int pkt_type){	struct hci_dev *hdev = hu->hdev;		/* Update HCI stat counters */	switch (pkt_type) {	case HCI_COMMAND_PKT:		hdev->stat.cmd_tx++;		break;	case HCI_ACLDATA_PKT:		hdev->stat.acl_tx++;		break;	case HCI_SCODATA_PKT:		hdev->stat.cmd_tx++;		break;	}}static inline struct sk_buff *hci_uart_dequeue(struct hci_uart *hu){	struct sk_buff *skb = hu->tx_skb;	if (!skb)		skb = hu->proto->dequeue(hu);	else		hu->tx_skb = NULL;	return skb;}int hci_uart_tx_wakeup(struct hci_uart *hu){	struct tty_struct *tty = hu->tty;	struct hci_dev *hdev = hu->hdev;	struct sk_buff *skb;		if (test_and_set_bit(HCI_UART_SENDING, &hu->tx_state)) {		set_bit(HCI_UART_TX_WAKEUP, &hu->tx_state);		return 0;	}	BT_DBG("");restart:	clear_bit(HCI_UART_TX_WAKEUP, &hu->tx_state);	while ((skb = hci_uart_dequeue(hu))) {		int len;			set_bit(TTY_DO_WRITE_WAKEUP, &tty->flags);		len = tty->driver->write(tty, 0, skb->data, skb->len);		hdev->stat.byte_tx += len;		skb_pull(skb, len);		if (skb->len) {			hu->tx_skb = skb;			break;		}			hci_uart_tx_complete(hu, skb->pkt_type);		kfree_skb(skb);	} 		if (test_bit(HCI_UART_TX_WAKEUP, &hu->tx_state))		goto restart;	clear_bit(HCI_UART_SENDING, &hu->tx_state);	return 0;}/* ------- Interface to HCI layer ------ *//* Initialize device */static int hci_uart_open(struct hci_dev *hdev){	BT_DBG("%s %p", hdev->name, hdev);	/* Nothing to do for UART driver */	set_bit(HCI_RUNNING, &hdev->flags);	return 0;}/* Reset device */static int hci_uart_flush(struct hci_dev *hdev){	struct hci_uart *hu  = (struct hci_uart *) hdev->driver_data;	struct tty_struct *tty = hu->tty;	BT_DBG("hdev %p tty %p", hdev, tty);	if (hu->tx_skb) {		kfree_skb(hu->tx_skb); hu->tx_skb = NULL;	}	/* Flush any pending characters in the driver and discipline. */	if (tty->ldisc.flush_buffer)		tty->ldisc.flush_buffer(tty);	if (tty->driver->flush_buffer)		tty->driver->flush_buffer(tty);	if (test_bit(HCI_UART_PROTO_SET, &hu->flags))		hu->proto->flush(hu);	return 0;}/* Close device */static int hci_uart_close(struct hci_dev *hdev){	BT_DBG("hdev %p", hdev);	if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags))		return 0;	hci_uart_flush(hdev);	return 0;}/* Send frames from HCI layer */static int hci_uart_send_frame(struct sk_buff *skb){	struct hci_dev* hdev = (struct hci_dev *) skb->dev;	struct tty_struct *tty;	struct hci_uart *hu;	if (!hdev) {		BT_ERR("Frame for uknown device (hdev=NULL)");		return -ENODEV;	}	if (!test_bit(HCI_RUNNING, &hdev->flags))		return -EBUSY;	hu = (struct hci_uart *) hdev->driver_data;	tty = hu->tty;	BT_DBG("%s: type %d len %d", hdev->name, skb->pkt_type, skb->len);	hu->proto->enqueue(hu, skb);	hci_uart_tx_wakeup(hu);	return 0;}static void hci_uart_destruct(struct hci_dev *hdev){	struct hci_uart *hu;	if (!hdev) return;	BT_DBG("%s", hdev->name);	hu = (struct hci_uart *) hdev->driver_data;	kfree(hu);}/* ------ LDISC part ------ *//* hci_uart_tty_open *  *     Called when line discipline changed to HCI_UART. * * Arguments: *     tty    pointer to tty info structure * Return Value:     *     0 if success, otherwise error code */static int hci_uart_tty_open(struct tty_struct *tty){	struct hci_uart *hu = (void *) tty->disc_data;	BT_DBG("tty %p", tty);	if (hu)		return -EEXIST;	if (!(hu = kmalloc(sizeof(struct hci_uart), GFP_KERNEL))) {		BT_ERR("Can't allocate controll structure");		return -ENFILE;	}	memset(hu, 0, sizeof(struct hci_uart));	tty->disc_data = hu;	hu->tty = tty;	spin_lock_init(&hu->rx_lock);	/* Flush any pending characters in the driver and line discipline */	if (tty->ldisc.flush_buffer)		tty->ldisc.flush_buffer(tty);	if (tty->driver->flush_buffer)		tty->driver->flush_buffer(tty);	return 0;}/* hci_uart_tty_close() * *    Called when the line discipline is changed to something *    else, the tty is closed, or the tty detects a hangup. */static void hci_uart_tty_close(struct tty_struct *tty){	struct hci_uart *hu = (void *)tty->disc_data;	BT_DBG("tty %p", tty);	/* Detach from the tty */	tty->disc_data = NULL;	if (hu) {		struct hci_dev *hdev = hu->hdev;		hci_uart_close(hdev);		if (test_and_clear_bit(HCI_UART_PROTO_SET, &hu->flags)) {			hu->proto->close(hu);			hci_unregister_dev(hdev);			hci_free_dev(hdev);		}	}}/* hci_uart_tty_wakeup() * *    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 = (void *)tty->disc_data;	BT_DBG("");	if (!hu)		return;	clear_bit(TTY_DO_WRITE_WAKEUP, &tty->flags);	if (tty != hu->tty)		return;	if (test_bit(HCI_UART_PROTO_SET, &hu->flags))		hci_uart_tx_wakeup(hu);}/* hci_uart_tty_room() *  *    Callback function from tty driver. Return the amount of  *    space left in the receiver's buffer to decide if remote *    transmitter is to be throttled. * * Arguments:        tty    pointer to associated tty instance data * Return Value:    number of bytes left in receive buffer */static int hci_uart_tty_room (struct tty_struct *tty){	return 65536;}/* hci_uart_tty_receive() *  *     Called by tty low level driver when receive data is *     available. *      * Arguments:  tty          pointer to tty isntance data *             data         pointer to received data *             flags        pointer to flags for data *             count        count of received data in bytes *      * Return Value:    None */static void hci_uart_tty_receive(struct tty_struct *tty, const __u8 *data, char *flags, int count){	struct hci_uart *hu = (void *)tty->disc_data;		if (!hu || tty != hu->tty)		return;	if (!test_bit(HCI_UART_PROTO_SET, &hu->flags))		return;		spin_lock(&hu->rx_lock);	hu->proto->recv(hu, (void *) data, count);	hu->hdev->stat.byte_rx += count;	spin_unlock(&hu->rx_lock);	if (test_and_clear_bit(TTY_THROTTLED,&tty->flags) && tty->driver->unthrottle)		tty->driver->unthrottle(tty);}static int hci_uart_register_dev(struct hci_uart *hu){	struct hci_dev *hdev;	BT_DBG("");	/* Initialize and register HCI device */	hdev = hci_alloc_dev();	if (!hdev) {		BT_ERR("Can't allocate HCI device");		return -ENOMEM;	}	hu->hdev = hdev;	hdev->type = HCI_UART;	hdev->driver_data = hu;	hdev->open  = hci_uart_open;	hdev->close = hci_uart_close;	hdev->flush = hci_uart_flush;	hdev->send  = hci_uart_send_frame;	hdev->destruct = hci_uart_destruct;	hdev->owner = THIS_MODULE;		if (hci_register_dev(hdev) < 0) {		BT_ERR("Can't register HCI device");		hci_free_dev(hdev);		return -ENODEV;	}	return 0;}static int hci_uart_set_proto(struct hci_uart *hu, int id){	struct hci_uart_proto *p;	int err;			p = hci_uart_get_proto(id);	if (!p)		return -EPROTONOSUPPORT;	err = p->open(hu);	if (err)		return err;	hu->proto = p;	err = hci_uart_register_dev(hu);	if (err) {		p->close(hu);		return err;	}	return 0;}/* hci_uart_tty_ioctl() * *    Process IOCTL system call for the tty device. * * Arguments: * *    tty        pointer to tty instance data *    file       pointer to open file object for device *    cmd        IOCTL command code *    arg        argument for IOCTL call (cmd dependent) * * Return Value:    Command dependent */static int hci_uart_tty_ioctl(struct tty_struct *tty, struct file * file,                            unsigned int cmd, unsigned long arg){	struct hci_uart *hu = (void *)tty->disc_data;	int err = 0;	BT_DBG("");	/* Verify the status of the device */	if (!hu)		return -EBADF;	switch (cmd) {	case HCIUARTSETPROTO:		if (!test_and_set_bit(HCI_UART_PROTO_SET, &hu->flags)) {			err = hci_uart_set_proto(hu, arg);			if (err) {				clear_bit(HCI_UART_PROTO_SET, &hu->flags);				return err;			}			tty->low_latency = 1;		} else				return -EBUSY;	case HCIUARTGETPROTO:		if (test_bit(HCI_UART_PROTO_SET, &hu->flags))			return hu->proto->id;		return -EUNATCH;			default:		err = n_tty_ioctl(tty, file, cmd, arg);		break;	};	return err;}/* * We don't provide read/write/poll interface for user space. */static ssize_t hci_uart_tty_read(struct tty_struct *tty, struct file *file, unsigned char __user *buf, size_t nr){	return 0;}static ssize_t hci_uart_tty_write(struct tty_struct *tty, struct file *file, const unsigned char __user *data, size_t count){	return 0;}static unsigned int hci_uart_tty_poll(struct tty_struct *tty, struct file *filp, poll_table *wait){	return 0;}#ifdef CONFIG_BT_HCIUART_H4int h4_init(void);int h4_deinit(void);#endif#ifdef CONFIG_BT_HCIUART_BCSPint bcsp_init(void);int bcsp_deinit(void);#endifstatic int __init hci_uart_init(void){	static struct tty_ldisc hci_uart_ldisc;	int err;	BT_INFO("HCI UART driver ver %s", VERSION);	/* Register the tty discipline */	memset(&hci_uart_ldisc, 0, sizeof (hci_uart_ldisc));	hci_uart_ldisc.magic       = TTY_LDISC_MAGIC;	hci_uart_ldisc.name        = "n_hci";	hci_uart_ldisc.open        = hci_uart_tty_open;	hci_uart_ldisc.close       = hci_uart_tty_close;	hci_uart_ldisc.read        = hci_uart_tty_read;	hci_uart_ldisc.write       = hci_uart_tty_write;	hci_uart_ldisc.ioctl       = hci_uart_tty_ioctl;	hci_uart_ldisc.poll        = hci_uart_tty_poll;	hci_uart_ldisc.receive_room= hci_uart_tty_room;	hci_uart_ldisc.receive_buf = hci_uart_tty_receive;	hci_uart_ldisc.write_wakeup= hci_uart_tty_wakeup;	hci_uart_ldisc.owner       = THIS_MODULE;	if ((err = tty_register_ldisc(N_HCI, &hci_uart_ldisc))) {		BT_ERR("HCI line discipline registration failed. (%d)", err);		return err;	}#ifdef CONFIG_BT_HCIUART_H4	h4_init();#endif#ifdef CONFIG_BT_HCIUART_BCSP	bcsp_init();#endif		return 0;}static void __exit hci_uart_exit(void){	int err;#ifdef CONFIG_BT_HCIUART_H4	h4_deinit();#endif#ifdef CONFIG_BT_HCIUART_BCSP	bcsp_deinit();#endif	/* Release tty registration of line discipline */	if ((err = tty_register_ldisc(N_HCI, NULL)))		BT_ERR("Can't unregister HCI line discipline (%d)", err);}module_init(hci_uart_init);module_exit(hci_uart_exit);MODULE_AUTHOR("Maxim Krasnyansky <maxk@qualcomm.com>");MODULE_DESCRIPTION("Bluetooth HCI UART driver ver " VERSION);MODULE_VERSION(VERSION);MODULE_LICENSE("GPL");MODULE_ALIAS_LDISC(N_HCI);

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
久久精品国产免费| 大桥未久av一区二区三区中文| 久久99久久99小草精品免视看| 国产成人亚洲综合色影视| 欧美午夜片在线看| 亚洲国产精品成人久久综合一区| 亚洲国产精品一区二区尤物区| 国产91清纯白嫩初高中在线观看 | 国产综合久久久久久鬼色| 91小视频在线免费看| 国产欧美日韩综合精品一区二区 | 日韩毛片精品高清免费| 美腿丝袜亚洲一区| 91久久线看在观草草青青| 欧美经典一区二区| 国精产品一区一区三区mba视频| 欧美中文字幕一区| 亚洲色图欧洲色图| a美女胸又www黄视频久久| 国产日本亚洲高清| 国产一区二区三区| 精品久久一区二区三区| 首页国产欧美久久| 欧美精品第1页| 亚洲mv在线观看| 欧美亚洲一区二区在线| 亚洲男同1069视频| 日本伦理一区二区| 亚洲激情图片小说视频| 91久久精品国产91性色tv | 成人午夜在线播放| 国产日韩精品视频一区| 国产在线国偷精品产拍免费yy | 欧美三级视频在线| 亚洲国产aⅴ成人精品无吗| 色视频欧美一区二区三区| 亚洲色图自拍偷拍美腿丝袜制服诱惑麻豆 | 亚洲国产中文字幕| 欧美亚日韩国产aⅴ精品中极品| 亚洲欧美另类久久久精品| www.爱久久.com| 亚洲欧美精品午睡沙发| 欧美日韩视频专区在线播放| 天天综合网 天天综合色| 欧美高清精品3d| 免播放器亚洲一区| 久久久精品天堂| 一本大道av伊人久久综合| 亚洲综合另类小说| 欧美美女视频在线观看| 久久精品国产77777蜜臀| 欧美精品一区二区在线播放| 国产成人欧美日韩在线电影| 久久久久久久精| 99re这里都是精品| 五月天精品一区二区三区| 欧美一级生活片| 岛国精品一区二区| 亚洲大片精品永久免费| 欧美大片国产精品| 成人黄色电影在线| 污片在线观看一区二区| 日韩精品中文字幕在线一区| 成人做爰69片免费看网站| 亚洲夂夂婷婷色拍ww47| 欧美一级高清大全免费观看| 国产成人精品影院| 亚洲香肠在线观看| 久久精品欧美一区二区三区不卡| 91亚洲精品久久久蜜桃| 免费av成人在线| 自拍偷拍欧美激情| 精品电影一区二区| 91国产视频在线观看| 国产综合色在线视频区| 一区二区三区日韩精品视频| 欧美tk丨vk视频| 在线日韩国产精品| 国产成人在线视频播放| 性欧美大战久久久久久久久| 久久婷婷成人综合色| 欧美性大战久久久久久久| 国模无码大尺度一区二区三区| 有码一区二区三区| 国产午夜亚洲精品羞羞网站| 欧美日韩国产一区| eeuss影院一区二区三区| 久久se精品一区二区| 亚洲国产综合色| 国产精品嫩草影院av蜜臀| 日韩女优制服丝袜电影| 欧美日韩国产bt| 色欲综合视频天天天| 国产福利一区在线| 精品一区二区三区视频| 石原莉奈在线亚洲二区| 亚洲精选视频在线| 中文字幕一区二区三区不卡在线| 精品国产乱码久久久久久1区2区 | 一区二区三区在线免费播放| 久久久久久久久久久久久久久99 | 久久精品国产久精国产| 亚洲福利视频一区| 一区二区免费看| 亚洲男女一区二区三区| 综合亚洲深深色噜噜狠狠网站| 久久精品人人做人人爽97 | 欧美日韩精品系列| 色综合久久久久久久久久久| 成人三级伦理片| 大胆欧美人体老妇| 成人综合婷婷国产精品久久| 久久99精品国产.久久久久久| 视频一区欧美日韩| 日韩在线观看一区二区| 日精品一区二区三区| 日韩高清一区在线| 日本不卡视频在线观看| 麻豆成人久久精品二区三区小说| 日韩av在线发布| 男女男精品视频网| 美女视频免费一区| 国产福利一区二区| 成人免费av资源| 91麻豆高清视频| 欧美影院一区二区三区| 6080国产精品一区二区| 欧美电影免费观看高清完整版| 欧美变态口味重另类| 久久久精品国产99久久精品芒果 | 亚洲网友自拍偷拍| 日韩精品视频网| 国产一区在线观看麻豆| 国产不卡视频在线播放| 91麻豆自制传媒国产之光| 欧美在线播放高清精品| 91麻豆精品国产91久久久更新时间 | 亚洲v日本v欧美v久久精品| 偷拍一区二区三区四区| 久久se这里有精品| 成人天堂资源www在线| 一道本成人在线| 欧美一区二区三区精品| 日韩美女主播在线视频一区二区三区| ww亚洲ww在线观看国产| 最新久久zyz资源站| 午夜国产精品影院在线观看| 久久国内精品视频| 99精品视频免费在线观看| 欧美日韩一区二区三区四区| 精品少妇一区二区三区在线视频| 欧美激情一区二区三区四区| 一区二区三区在线播| 久久国产精品色| 91美女片黄在线观看91美女| 91精品国产综合久久婷婷香蕉| 久久综合久久99| 伊人一区二区三区| 国产精品一级在线| 欧美日韩在线综合| 国产精品每日更新在线播放网址| 亚洲va欧美va人人爽| 国产91精品免费| 91精品国产福利| 亚洲蜜桃精久久久久久久| 国精产品一区一区三区mba视频| 色综合久久六月婷婷中文字幕| 日韩视频一区二区三区在线播放| 亚洲图片欧美激情| 国产精品99久久久久| 91精品国产综合久久精品性色 | 亚洲va欧美va人人爽午夜| 成人午夜精品一区二区三区| 欧美一区二区三区啪啪| 亚洲美女免费视频| 成人一区在线看| 久久精品人人做人人爽人人| 日韩av二区在线播放| 在线区一区二视频| 国产精品女同一区二区三区| 久久国产生活片100| 欧美精品黑人性xxxx| 亚洲综合色在线| 一本一道波多野结衣一区二区| 国产欧美日韩不卡免费| 精品一区二区av| 91精品国产入口| 免费人成精品欧美精品| 欧美群妇大交群中文字幕| 亚洲综合男人的天堂| 91久久精品日日躁夜夜躁欧美| 国产精品久久久久久亚洲毛片 | 欧美亚洲国产怡红院影院| 欧美激情综合在线| 成人在线视频一区二区| 中文字幕第一区二区| 成人综合在线观看| 国产精品乱码妇女bbbb| 不卡高清视频专区| √…a在线天堂一区|