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

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

?? btsdio.c

?? Linux環境下的藍牙模塊驅動程序源代碼
?? C
字號:
/* * *  Generic Bluetooth SDIO driver * *  Copyright (C) 2007  Cambridge Silicon Radio Ltd. *  Copyright (C) 2007  Marcel Holtmann <marcel@holtmann.org> * * *  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 <linux/kernel.h>#include <linux/module.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/types.h>#include <linux/sched.h>#include <linux/errno.h>#include <linux/skbuff.h>#include <linux/mmc/sdio_ids.h>#include <linux/mmc/sdio_func.h>#include <net/bluetooth/bluetooth.h>#include <net/bluetooth/hci_core.h>#define VERSION "0.1"static const struct sdio_device_id btsdio_table[] = {	/* Generic Bluetooth Type-A SDIO device */	{ SDIO_DEVICE_CLASS(SDIO_CLASS_BT_A) },	/* Generic Bluetooth Type-B SDIO device */	{ SDIO_DEVICE_CLASS(SDIO_CLASS_BT_B) },	{ }	/* Terminating entry */};MODULE_DEVICE_TABLE(sdio, btsdio_table);struct btsdio_data {	struct hci_dev   *hdev;	struct sdio_func *func;	struct work_struct work;	struct sk_buff_head txq;};#define REG_RDAT     0x00	/* Receiver Data */#define REG_TDAT     0x00	/* Transmitter Data */#define REG_PC_RRT   0x10	/* Read Packet Control */#define REG_PC_WRT   0x11	/* Write Packet Control */#define REG_RTC_STAT 0x12	/* Retry Control Status */#define REG_RTC_SET  0x12	/* Retry Control Set */#define REG_INTRD    0x13	/* Interrupt Indication */#define REG_CL_INTRD 0x13	/* Interrupt Clear */#define REG_EN_INTRD 0x14	/* Interrupt Enable */#define REG_MD_STAT  0x20	/* Bluetooth Mode Status */static int btsdio_tx_packet(struct btsdio_data *data, struct sk_buff *skb){	int err;	BT_DBG("%s", data->hdev->name);	/* Prepend Type-A header */	skb_push(skb, 4);	skb->data[0] = (skb->len & 0x0000ff);	skb->data[1] = (skb->len & 0x00ff00) >> 8;	skb->data[2] = (skb->len & 0xff0000) >> 16;	skb->data[3] = bt_cb(skb)->pkt_type;	err = sdio_writesb(data->func, REG_TDAT, skb->data, skb->len);	if (err < 0) {		skb_pull(skb, 4);		sdio_writeb(data->func, 0x01, REG_PC_WRT, NULL);		return err;	}	data->hdev->stat.byte_tx += skb->len;	kfree_skb(skb);	return 0;}static void btsdio_work(struct work_struct *work){	struct btsdio_data *data = container_of(work, struct btsdio_data, work);	struct sk_buff *skb;	int err;	BT_DBG("%s", data->hdev->name);	sdio_claim_host(data->func);	while ((skb = skb_dequeue(&data->txq))) {		err = btsdio_tx_packet(data, skb);		if (err < 0) {			data->hdev->stat.err_tx++;			skb_queue_head(&data->txq, skb);			break;		}	}	sdio_release_host(data->func);}static int btsdio_rx_packet(struct btsdio_data *data){	u8 hdr[4] __attribute__ ((aligned(4)));	struct sk_buff *skb;	int err, len;	BT_DBG("%s", data->hdev->name);	err = sdio_readsb(data->func, hdr, REG_RDAT, 4);	if (err < 0)		return err;	len = hdr[0] | (hdr[1] << 8) | (hdr[2] << 16);	if (len < 4 || len > 65543)		return -EILSEQ;	skb = bt_skb_alloc(len - 4, GFP_KERNEL);	if (!skb) {		/* Out of memory. Prepare a read retry and just		 * return with the expectation that the next time		 * we're called we'll have more memory. */		return -ENOMEM;	}	skb_put(skb, len - 4);	err = sdio_readsb(data->func, skb->data, REG_RDAT, len - 4);	if (err < 0) {		kfree_skb(skb);		return err;	}	data->hdev->stat.byte_rx += len;	skb->dev = (void *) data->hdev;	bt_cb(skb)->pkt_type = hdr[3];	err = hci_recv_frame(skb);	if (err < 0)		return err;	sdio_writeb(data->func, 0x00, REG_PC_RRT, NULL);	return 0;}static void btsdio_interrupt(struct sdio_func *func){	struct btsdio_data *data = sdio_get_drvdata(func);	int intrd;	BT_DBG("%s", data->hdev->name);	intrd = sdio_readb(func, REG_INTRD, NULL);	if (intrd & 0x01) {		sdio_writeb(func, 0x01, REG_CL_INTRD, NULL);		if (btsdio_rx_packet(data) < 0) {			data->hdev->stat.err_rx++;			sdio_writeb(data->func, 0x01, REG_PC_RRT, NULL);		}	}}static int btsdio_open(struct hci_dev *hdev){	struct btsdio_data *data = hdev->driver_data;	int err;	BT_DBG("%s", hdev->name);	if (test_and_set_bit(HCI_RUNNING, &hdev->flags))		return 0;	sdio_claim_host(data->func);	err = sdio_enable_func(data->func);	if (err < 0) {		clear_bit(HCI_RUNNING, &hdev->flags);		goto release;	}	err = sdio_claim_irq(data->func, btsdio_interrupt);	if (err < 0) {		sdio_disable_func(data->func);		clear_bit(HCI_RUNNING, &hdev->flags);		goto release;	}	if (data->func->class == SDIO_CLASS_BT_B)		sdio_writeb(data->func, 0x00, REG_MD_STAT, NULL);	sdio_writeb(data->func, 0x01, REG_EN_INTRD, NULL);release:	sdio_release_host(data->func);	return err;}static int btsdio_close(struct hci_dev *hdev){	struct btsdio_data *data = hdev->driver_data;	BT_DBG("%s", hdev->name);	if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags))		return 0;	sdio_claim_host(data->func);	sdio_writeb(data->func, 0x00, REG_EN_INTRD, NULL);	sdio_release_irq(data->func);	sdio_disable_func(data->func);	sdio_release_host(data->func);	return 0;}static int btsdio_flush(struct hci_dev *hdev){	struct btsdio_data *data = hdev->driver_data;	BT_DBG("%s", hdev->name);	skb_queue_purge(&data->txq);	return 0;}static int btsdio_send_frame(struct sk_buff *skb){	struct hci_dev *hdev = (struct hci_dev *) skb->dev;	struct btsdio_data *data = hdev->driver_data;	BT_DBG("%s", hdev->name);	if (!test_bit(HCI_RUNNING, &hdev->flags))		return -EBUSY;	switch (bt_cb(skb)->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.sco_tx++;		break;	default:		return -EILSEQ;	}	skb_queue_tail(&data->txq, skb);	schedule_work(&data->work);	return 0;}static void btsdio_destruct(struct hci_dev *hdev){	struct btsdio_data *data = hdev->driver_data;	BT_DBG("%s", hdev->name);	kfree(data);}static int btsdio_probe(struct sdio_func *func,				const struct sdio_device_id *id){	struct btsdio_data *data;	struct hci_dev *hdev;	struct sdio_func_tuple *tuple = func->tuples;	int err;	BT_DBG("func %p id %p class 0x%04x", func, id, func->class);	while (tuple) {		BT_DBG("code 0x%x size %d", tuple->code, tuple->size);		tuple = tuple->next;	}	data = kzalloc(sizeof(*data), GFP_KERNEL);	if (!data)		return -ENOMEM;	data->func = func;	INIT_WORK(&data->work, btsdio_work);	skb_queue_head_init(&data->txq);	hdev = hci_alloc_dev();	if (!hdev) {		kfree(data);		return -ENOMEM;	}	hdev->type = HCI_SDIO;	hdev->driver_data = data;	data->hdev = hdev;	SET_HCIDEV_DEV(hdev, &func->dev);	hdev->open     = btsdio_open;	hdev->close    = btsdio_close;	hdev->flush    = btsdio_flush;	hdev->send     = btsdio_send_frame;	hdev->destruct = btsdio_destruct;	hdev->owner = THIS_MODULE;	err = hci_register_dev(hdev);	if (err < 0) {		hci_free_dev(hdev);		kfree(data);		return err;	}	sdio_set_drvdata(func, data);	return 0;}static void btsdio_remove(struct sdio_func *func){	struct btsdio_data *data = sdio_get_drvdata(func);	struct hci_dev *hdev;	BT_DBG("func %p", func);	if (!data)		return;	hdev = data->hdev;	sdio_set_drvdata(func, NULL);	hci_unregister_dev(hdev);	hci_free_dev(hdev);}static struct sdio_driver btsdio_driver = {	.name		= "btsdio",	.probe		= btsdio_probe,	.remove		= btsdio_remove,	.id_table	= btsdio_table,};static int __init btsdio_init(void){	BT_INFO("Generic Bluetooth SDIO driver ver %s", VERSION);	return sdio_register_driver(&btsdio_driver);}static void __exit btsdio_exit(void){	sdio_unregister_driver(&btsdio_driver);}module_init(btsdio_init);module_exit(btsdio_exit);MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");MODULE_DESCRIPTION("Generic Bluetooth SDIO driver ver " VERSION);MODULE_VERSION(VERSION);MODULE_LICENSE("GPL");

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
亚洲免费成人av| 欧美午夜理伦三级在线观看| 欧美一卡二卡三卡四卡| 性久久久久久久久| 欧美日韩一区二区三区视频| 亚洲成av人片在www色猫咪| 欧美亚洲一区二区在线| 午夜精品一区二区三区免费视频| 欧美人与z0zoxxxx视频| 亚洲国产精品一区二区久久 | 亚洲国产综合色| 成人v精品蜜桃久久一区| 国产亚洲va综合人人澡精品| 国产伦精品一区二区三区免费迷| 91碰在线视频| 亚洲最新在线观看| 欧美肥妇bbw| 国产精品一线二线三线精华| 中文字幕欧美三区| 色香蕉久久蜜桃| 亚洲国产视频一区二区| 日韩欧美国产一区在线观看| 国产精品538一区二区在线| 最好看的中文字幕久久| 欧美人妖巨大在线| 成人性生交大片免费看中文网站| 欧美久久久久久久久中文字幕| 久久久久国产免费免费| jlzzjlzz亚洲日本少妇| 天堂影院一区二区| 久久亚洲精精品中文字幕早川悠里| 亚洲三级免费电影| 26uuu久久综合| 91在线观看下载| 三级久久三级久久| 亚洲国产成人在线| 日韩亚洲欧美中文三级| 国产成人免费视频一区| 亚洲第一主播视频| 国产日产欧美一区| 欧美区视频在线观看| 成人午夜精品一区二区三区| 亚洲va国产天堂va久久en| 久久精品亚洲麻豆av一区二区| 麻豆精品一二三| 欧美极品少妇xxxxⅹ高跟鞋 | 亚洲午夜免费电影| 国产亚洲精品免费| 欧美区一区二区三区| 成人免费看视频| 日av在线不卡| 亚洲九九爱视频| 久久久久久久久久电影| 欧美精品丝袜中出| 97se狠狠狠综合亚洲狠狠| 蜜臀久久99精品久久久久宅男 | 成人永久免费视频| 日韩电影网1区2区| 伊人婷婷欧美激情| 国产校园另类小说区| 欧美一级电影网站| 欧美色图片你懂的| 色婷婷精品久久二区二区蜜臀av| 亚洲人成亚洲人成在线观看图片| 不卡一卡二卡三乱码免费网站| 中文字幕免费在线观看视频一区| 国产成人小视频| 国内精品伊人久久久久av一坑| 精品久久久久香蕉网| 这里只有精品电影| 欧美亚洲动漫制服丝袜| 色欧美88888久久久久久影院| 樱花草国产18久久久久| 亚洲国产精品成人综合色在线婷婷| 不卡大黄网站免费看| 国产一区二区按摩在线观看| 免费在线看成人av| 天天色天天操综合| 香蕉乱码成人久久天堂爱免费| 欧美一区二区久久久| 欧美日韩色一区| 在线日韩一区二区| 欧美影片第一页| 欧美综合视频在线观看| 91久久香蕉国产日韩欧美9色| 亚洲图片欧美色图| 亚洲亚洲精品在线观看| 国产麻豆精品95视频| 久久超碰97中文字幕| 蜜桃av一区二区在线观看| 青青国产91久久久久久| 久久国产人妖系列| 国产九色sp调教91| 成人激情图片网| 91香蕉国产在线观看软件| 91在线精品一区二区| 欧美少妇bbb| 日韩欧美高清一区| 国产欧美日韩精品一区| 国产精品嫩草影院av蜜臀| 亚洲人成影院在线观看| 爽好久久久欧美精品| 韩国成人精品a∨在线观看| 国产福利一区在线| 一本色道久久综合亚洲91| 欧美日韩一级片在线观看| 日韩三级视频在线观看| 欧美国产亚洲另类动漫| 亚洲黄色片在线观看| 奇米色一区二区| 国产成都精品91一区二区三| 色婷婷精品大视频在线蜜桃视频| 国产美女一区二区三区| aa级大片欧美| 欧美日韩在线播放| 精品毛片乱码1区2区3区| 国产精品免费视频网站| 亚洲一区二区精品视频| 天天色天天操综合| 国产福利精品一区| 欧美亚洲综合一区| 久久综合色婷婷| 亚洲精品成人精品456| 奇米精品一区二区三区在线观看 | 国产女人水真多18毛片18精品视频| 欧美日韩高清在线播放| 精品欧美一区二区久久| 国产精品久久免费看| 三级欧美在线一区| 欧美成人精品1314www| 亚洲国产精品成人综合色在线婷婷 | 欧美日韩一本到| 精品久久五月天| 亚洲少妇中出一区| 久久精品理论片| 在线观看亚洲精品视频| 久久久五月婷婷| 五月天久久比比资源色| 成人高清视频免费观看| 日韩欧美一二区| 亚洲精品欧美二区三区中文字幕| 国产精品午夜在线观看| 欧美a级一区二区| 91麻豆国产香蕉久久精品| 精品电影一区二区三区 | 欧美一二三区在线观看| 亚洲婷婷国产精品电影人久久| 国产欧美1区2区3区| 午夜电影一区二区| 色综合色狠狠综合色| 国产婷婷一区二区| 免费的成人av| 欧美日韩精品一区二区在线播放| 欧美日韩你懂的| 亚洲欧美另类图片小说| 国产精品一二三四区| 日韩三级高清在线| 肉肉av福利一精品导航| 欧美又粗又大又爽| 日韩一区中文字幕| 成人午夜电影久久影院| 日韩欧美国产一区在线观看| 亚洲成av人片| 在线看国产一区二区| 亚洲男人的天堂在线aⅴ视频| 亚洲欧洲成人自拍| 成人免费黄色在线| 亚洲国产经典视频| 懂色av一区二区在线播放| 精品国产乱码久久久久久夜甘婷婷| 欧美电影精品一区二区| 美腿丝袜一区二区三区| 日韩一区二区三区在线观看| 日韩精品成人一区二区在线| 欧美日韩你懂得| 青青草国产精品亚洲专区无| 欧美一区二区三区视频免费| 亚洲电影欧美电影有声小说| 精品婷婷伊人一区三区三| 丝袜美腿亚洲综合| 日韩欧美中文字幕精品| 经典三级在线一区| 久久精品一二三| 成人av电影免费观看| 国产精品水嫩水嫩| 91麻豆蜜桃一区二区三区| 亚洲蜜臀av乱码久久精品蜜桃| 视频在线观看一区二区三区| 国产精品网曝门| 国产一区 二区| 日本一区二区不卡视频| 色综合久久88色综合天天6| 亚洲综合图片区| 欧美一区二区三区日韩| 精品一区二区精品| 国产精品国产馆在线真实露脸 | 国内精品伊人久久久久av影院 | 久久精品国产网站| 国产日韩欧美精品电影三级在线| 亚洲最色的网站|