⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 btuart.c

📁 基于arm的蓝牙通讯协议的设计
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * *  Driver for Bluetooth PCMCIA cards with HCI UART interface * * */#include <linux/config.h>#include <linux/module.h>#include <linux/kernel.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/types.h>#include <linux/sched.h>#include <linux/timer.h>#include <linux/errno.h>#include <linux/ptrace.h>#include <linux/ioport.h>#include <linux/spinlock.h>#include <linux/skbuff.h>#include <linux/string.h>#include <linux/serial.h>#include <linux/serial_reg.h>#include <asm/system.h>#include <asm/bitops.h>#include <asm/io.h>#include <pcmcia/version.h>#include <pcmcia/cs_types.h>#include <pcmcia/cs.h>#include <pcmcia/cistpl.h>#include <pcmcia/ciscode.h>#include <pcmcia/ds.h>#include <pcmcia/cisreg.h>#include <net/bluetooth/bluetooth.h>#include <net/bluetooth/hci_core.h>/* ======================== Module parameters ======================== *//* Bit map of interrupts to choose from */static u_int irq_mask = 0xffff;static int irq_list[4] = { -1 };MODULE_PARM(irq_mask, "i");MODULE_PARM(irq_list, "1-4i");MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");MODULE_DESCRIPTION("BlueZ driver for Bluetooth PCMCIA cards with HCI UART interface");MODULE_LICENSE("GPL");/* ======================== Local structures ======================== */typedef struct btuart_info_t {	dev_link_t link;	dev_node_t node;	struct hci_dev hdev;	spinlock_t lock;	/* For serializing operations */	struct sk_buff_head txq;	unsigned long tx_state;	unsigned long rx_state;	unsigned long rx_count;	struct sk_buff *rx_skb;} btuart_info_t;void btuart_config(dev_link_t *link);void btuart_release(u_long arg);int btuart_event(event_t event, int priority, event_callback_args_t *args);static dev_info_t dev_info = "btuart_cs";dev_link_t *btuart_attach(void);void btuart_detach(dev_link_t *);static dev_link_t *dev_list = NULL;/* Maximum baud rate */#define SPEED_MAX  115200/* Default baud rate: 57600, 115200, 230400 or 460800 */#define DEFAULT_BAUD_RATE  115200/* Transmit states  */#define XMIT_SENDING	1#define XMIT_WAKEUP	2#define XMIT_WAITING	8/* Receiver states */#define RECV_WAIT_PACKET_TYPE	0#define RECV_WAIT_EVENT_HEADER	1#define RECV_WAIT_ACL_HEADER	2#define RECV_WAIT_SCO_HEADER	3#define RECV_WAIT_DATA		4/* ======================== Interrupt handling ======================== */static int btuart_write(unsigned int iobase, int fifo_size, __u8 *buf, int len){	int actual = 0;	/* Tx FIFO should be empty */	if (!(inb(iobase + UART_LSR) & UART_LSR_THRE))		return 0;	/* Fill FIFO with current frame */	while ((fifo_size-- > 0) && (actual < len)) {		/* Transmit next byte */		outb(buf[actual], iobase + UART_TX);		actual++;	}	return actual;}static void btuart_write_wakeup(btuart_info_t *info){	if (!info) {		printk(KERN_WARNING "btuart_cs: Call of write_wakeup for unknown device.\n");		return;	}	if (test_and_set_bit(XMIT_SENDING, &(info->tx_state))) {		set_bit(XMIT_WAKEUP, &(info->tx_state));		return;	}	do {		register unsigned int iobase = info->link.io.BasePort1;		register struct sk_buff *skb;		register int len;		clear_bit(XMIT_WAKEUP, &(info->tx_state));		if (!(info->link.state & DEV_PRESENT))			return;		if (!(skb = skb_dequeue(&(info->txq))))			break;		/* Send frame */		len = btuart_write(iobase, 16, skb->data, skb->len);		set_bit(XMIT_WAKEUP, &(info->tx_state));		if (len == skb->len) {			kfree_skb(skb);		} else {			skb_pull(skb, len);			skb_queue_head(&(info->txq), skb);		}		info->hdev.stat.byte_tx += len;	} while (test_bit(XMIT_WAKEUP, &(info->tx_state)));	clear_bit(XMIT_SENDING, &(info->tx_state));}static void btuart_receive(btuart_info_t *info){	unsigned int iobase;	int boguscount = 0;	if (!info) {		printk(KERN_WARNING "btuart_cs: Call of receive for unknown device.\n");		return;	}	iobase = info->link.io.BasePort1;	do {		info->hdev.stat.byte_rx++;		/* Allocate packet */		if (info->rx_skb == NULL) {			info->rx_state = RECV_WAIT_PACKET_TYPE;			info->rx_count = 0;			if (!(info->rx_skb = bluez_skb_alloc(HCI_MAX_FRAME_SIZE, GFP_ATOMIC))) {				printk(KERN_WARNING "btuart_cs: Can't allocate mem for new packet.\n");				return;			}		}		if (info->rx_state == RECV_WAIT_PACKET_TYPE) {			info->rx_skb->dev = (void *)&(info->hdev);			info->rx_skb->pkt_type = inb(iobase + UART_RX);			switch (info->rx_skb->pkt_type) {			case HCI_EVENT_PKT:				info->rx_state = RECV_WAIT_EVENT_HEADER;				info->rx_count = HCI_EVENT_HDR_SIZE;				break;			case HCI_ACLDATA_PKT:				info->rx_state = RECV_WAIT_ACL_HEADER;				info->rx_count = HCI_ACL_HDR_SIZE;				break;			case HCI_SCODATA_PKT:				info->rx_state = RECV_WAIT_SCO_HEADER;				info->rx_count = HCI_SCO_HDR_SIZE;				break;			default:				/* Unknown packet */				printk(KERN_WARNING "btuart_cs: Unknown HCI packet with type 0x%02x received.\n", info->rx_skb->pkt_type);				info->hdev.stat.err_rx++;				clear_bit(HCI_RUNNING, &(info->hdev.flags));				kfree_skb(info->rx_skb);				info->rx_skb = NULL;				break;			}		} else {			*skb_put(info->rx_skb, 1) = inb(iobase + UART_RX);			info->rx_count--;			if (info->rx_count == 0) {				int dlen;				hci_event_hdr *eh;				hci_acl_hdr *ah;				hci_sco_hdr *sh;				switch (info->rx_state) {				case RECV_WAIT_EVENT_HEADER:					eh = (hci_event_hdr *)(info->rx_skb->data);					info->rx_state = RECV_WAIT_DATA;					info->rx_count = eh->plen;					break;				case RECV_WAIT_ACL_HEADER:					ah = (hci_acl_hdr *)(info->rx_skb->data);					dlen = __le16_to_cpu(ah->dlen);					info->rx_state = RECV_WAIT_DATA;					info->rx_count = dlen;					break;				case RECV_WAIT_SCO_HEADER:					sh = (hci_sco_hdr *)(info->rx_skb->data);					info->rx_state = RECV_WAIT_DATA;					info->rx_count = sh->dlen;					break;				case RECV_WAIT_DATA:					hci_recv_frame(info->rx_skb);					info->rx_skb = NULL;					break;				}			}		}		/* Make sure we don't stay here to long */		if (boguscount++ > 16)			break;	} while (inb(iobase + UART_LSR) & UART_LSR_DR);}void btuart_interrupt(int irq, void *dev_inst, struct pt_regs *regs){	btuart_info_t *info = dev_inst;	unsigned int iobase;	int boguscount = 0;	int iir, lsr;	if (!info) {		printk(KERN_WARNING "btuart_cs: Call of irq %d for unknown device.\n", irq);		return;	}	iobase = info->link.io.BasePort1;	spin_lock(&(info->lock));	iir = inb(iobase + UART_IIR) & UART_IIR_ID;	while (iir) {		/* Clear interrupt */		lsr = inb(iobase + UART_LSR);		switch (iir) {		case UART_IIR_RLSI:			printk(KERN_NOTICE "btuart_cs: RLSI\n");			break;		case UART_IIR_RDI:			/* Receive interrupt */			btuart_receive(info);			break;		case UART_IIR_THRI:			if (lsr & UART_LSR_THRE) {				/* Transmitter ready for data */				btuart_write_wakeup(info);			}			break;		default:			printk(KERN_NOTICE "btuart_cs: Unhandled IIR=%#x\n", iir);			break;		}		/* Make sure we don't stay here to long */		if (boguscount++ > 100)			break;		iir = inb(iobase + UART_IIR) & UART_IIR_ID;	}	spin_unlock(&(info->lock));}static void btuart_change_speed(btuart_info_t *info, unsigned int speed){	unsigned long flags;	unsigned int iobase;	int fcr;		/* FIFO control reg */	int lcr;		/* Line control reg */	int divisor;	if (!info) {		printk(KERN_WARNING "btuart_cs: Call of change speed for unknown device.\n");		return;	}	iobase = info->link.io.BasePort1;	spin_lock_irqsave(&(info->lock), flags);	/* Turn off interrupts */	outb(0, iobase + UART_IER);	divisor = SPEED_MAX / speed;	fcr = UART_FCR_ENABLE_FIFO | UART_FCR_CLEAR_RCVR | UART_FCR_CLEAR_XMIT;	/* 	 * Use trigger level 1 to avoid 3 ms. timeout delay at 9600 bps, and	 * almost 1,7 ms at 19200 bps. At speeds above that we can just forget	 * about this timeout since it will always be fast enough. 	 */	if (speed < 38400)		fcr |= UART_FCR_TRIGGER_1;	else		fcr |= UART_FCR_TRIGGER_14;	/* Bluetooth cards use 8N1 */	lcr = UART_LCR_WLEN8;	outb(UART_LCR_DLAB | lcr, iobase + UART_LCR);	/* Set DLAB */	outb(divisor & 0xff, iobase + UART_DLL);	/* Set speed */	outb(divisor >> 8, iobase + UART_DLM);	outb(lcr, iobase + UART_LCR);	/* Set 8N1  */	outb(fcr, iobase + UART_FCR);	/* Enable FIFO's */	/* Turn on interrups */	outb(UART_IER_RLSI | UART_IER_RDI | UART_IER_THRI, iobase + UART_IER);	spin_unlock_irqrestore(&(info->lock), flags);}/* ======================== HCI interface ======================== */static int btuart_hci_flush(struct hci_dev *hdev){	btuart_info_t *info = (btuart_info_t *)(hdev->driver_data);	/* Drop TX queue */	skb_queue_purge(&(info->txq));	return 0;}static int btuart_hci_open(struct hci_dev *hdev){	set_bit(HCI_RUNNING, &(hdev->flags));	return 0;}static int btuart_hci_close(struct hci_dev *hdev){	if (!test_and_clear_bit(HCI_RUNNING, &(hdev->flags)))		return 0;	btuart_hci_flush(hdev);	return 0;}static int btuart_hci_send_frame(struct sk_buff *skb){	btuart_info_t *info;	struct hci_dev *hdev = (struct hci_dev *)(skb->dev);	if (!hdev) {		printk(KERN_WARNING "btuart_cs: Frame for unknown HCI device (hdev=NULL).");		return -ENODEV;	}	info = (btuart_info_t *)(hdev->driver_data);	switch (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++;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -