Blame view

drivers/bluetooth/btuart_cs.c 13.9 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
  /*
   *
   *  Driver for Bluetooth PCMCIA cards with HCI UART interface
   *
   *  Copyright (C) 2001-2002  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 version 2 as
   *  published by the Free Software Foundation;
   *
   *  Software distributed under the License is distributed on an "AS
   *  IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
   *  implied. See the License for the specific language governing
   *  rights and limitations under the License.
   *
   *  The initial developer of the original code is David A. Hinds
   *  <dahinds@users.sourceforge.net>.  Portions created by David A. Hinds
   *  are Copyright (C) 1999 David A. Hinds.  All Rights Reserved.
   *
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22
23
24
25
26
27
  #include <linux/module.h>
  
  #include <linux/kernel.h>
  #include <linux/init.h>
  #include <linux/slab.h>
  #include <linux/types.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
28
29
30
31
32
33
34
35
36
37
38
39
  #include <linux/delay.h>
  #include <linux/errno.h>
  #include <linux/ptrace.h>
  #include <linux/ioport.h>
  #include <linux/spinlock.h>
  #include <linux/moduleparam.h>
  
  #include <linux/skbuff.h>
  #include <linux/string.h>
  #include <linux/serial.h>
  #include <linux/serial_reg.h>
  #include <linux/bitops.h>
239ede3c0   Prasanna Karthik   Bluetooth: btuart...
40
  #include <linux/io.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
41

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
  #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 ======================== */
  
  
  MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");
  MODULE_DESCRIPTION("Bluetooth driver for Bluetooth PCMCIA cards with HCI UART interface");
  MODULE_LICENSE("GPL");
  
  
  
  /* ======================== Local structures ======================== */
ad709d486   Himangi Saraogi   Bluetooth: Remove...
62
  struct btuart_info {
fd238232c   Dominik Brodowski   [PATCH] pcmcia: e...
63
  	struct pcmcia_device *p_dev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
64
65
66
67
68
69
70
71
72
73
74
  
  	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;
ad709d486   Himangi Saraogi   Bluetooth: Remove...
75
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
76

15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
77
  static int btuart_config(struct pcmcia_device *link);
fba395eee   Dominik Brodowski   [PATCH] pcmcia: r...
78
  static void btuart_release(struct pcmcia_device *link);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
79

cc3b4866b   Dominik Brodowski   [PATCH] pcmcia: u...
80
  static void btuart_detach(struct pcmcia_device *p_dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
81

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
  
  /* 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;
  }
ad709d486   Himangi Saraogi   Bluetooth: Remove...
124
  static void btuart_write_wakeup(struct btuart_info *info)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
125
126
127
128
129
130
131
132
133
134
135
136
  {
  	if (!info) {
  		BT_ERR("Unknown device");
  		return;
  	}
  
  	if (test_and_set_bit(XMIT_SENDING, &(info->tx_state))) {
  		set_bit(XMIT_WAKEUP, &(info->tx_state));
  		return;
  	}
  
  	do {
fc5fef615   Gustavo Padovan   Bluetooth: Remove...
137
  		unsigned int iobase = info->p_dev->resource[0]->start;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
138
  		register struct sk_buff *skb;
fc5fef615   Gustavo Padovan   Bluetooth: Remove...
139
  		int len;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
140
141
  
  		clear_bit(XMIT_WAKEUP, &(info->tx_state));
e2d409636   Dominik Brodowski   [PATCH] pcmcia: u...
142
  		if (!pcmcia_dev_present(info->p_dev))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
143
  			return;
a08b15e66   Valentin Ilie   Bluetooth: Remove...
144
145
  		skb = skb_dequeue(&(info->txq));
  		if (!skb)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
  			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));
  }
ad709d486   Himangi Saraogi   Bluetooth: Remove...
165
  static void btuart_receive(struct btuart_info *info)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
166
167
168
169
170
171
172
173
  {
  	unsigned int iobase;
  	int boguscount = 0;
  
  	if (!info) {
  		BT_ERR("Unknown device");
  		return;
  	}
9a017a910   Dominik Brodowski   pcmcia: do not us...
174
  	iobase = info->p_dev->resource[0]->start;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
175
176
177
178
179
  
  	do {
  		info->hdev->stat.byte_rx++;
  
  		/* Allocate packet */
05bd7762f   Prasanna Karthik   Bluetooth: btuart...
180
  		if (!info->rx_skb) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
181
182
  			info->rx_state = RECV_WAIT_PACKET_TYPE;
  			info->rx_count = 0;
a08b15e66   Valentin Ilie   Bluetooth: Remove...
183
184
  			info->rx_skb = bt_skb_alloc(HCI_MAX_FRAME_SIZE, GFP_ATOMIC);
  			if (!info->rx_skb) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
185
186
187
188
189
190
  				BT_ERR("Can't allocate mem for new packet");
  				return;
  			}
  		}
  
  		if (info->rx_state == RECV_WAIT_PACKET_TYPE) {
618e8bc22   Marcel Holtmann   Bluetooth: Use ne...
191
  			hci_skb_pkt_type(info->rx_skb) = inb(iobase + UART_RX);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
192

618e8bc22   Marcel Holtmann   Bluetooth: Use ne...
193
  			switch (hci_skb_pkt_type(info->rx_skb)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
  
  			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 */
618e8bc22   Marcel Holtmann   Bluetooth: Use ne...
212
213
  				BT_ERR("Unknown HCI packet with type 0x%02x received",
  				       hci_skb_pkt_type(info->rx_skb));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
214
  				info->hdev->stat.err_rx++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
  
  				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;
  				struct hci_event_hdr *eh;
  				struct hci_acl_hdr *ah;
  				struct hci_sco_hdr *sh;
  
  
  				switch (info->rx_state) {
  
  				case RECV_WAIT_EVENT_HEADER:
2a123b86e   Arnaldo Carvalho de Melo   [BLUETOOTH]: Intr...
238
  					eh = hci_event_hdr(info->rx_skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
239
240
241
242
243
  					info->rx_state = RECV_WAIT_DATA;
  					info->rx_count = eh->plen;
  					break;
  
  				case RECV_WAIT_ACL_HEADER:
2a123b86e   Arnaldo Carvalho de Melo   [BLUETOOTH]: Intr...
244
  					ah = hci_acl_hdr(info->rx_skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
245
246
247
248
249
250
  					dlen = __le16_to_cpu(ah->dlen);
  					info->rx_state = RECV_WAIT_DATA;
  					info->rx_count = dlen;
  					break;
  
  				case RECV_WAIT_SCO_HEADER:
2a123b86e   Arnaldo Carvalho de Melo   [BLUETOOTH]: Intr...
251
  					sh = hci_sco_hdr(info->rx_skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
252
253
254
255
256
  					info->rx_state = RECV_WAIT_DATA;
  					info->rx_count = sh->dlen;
  					break;
  
  				case RECV_WAIT_DATA:
e1a261706   Marcel Holtmann   Bluetooth: Provid...
257
  					hci_recv_frame(info->hdev, info->rx_skb);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
  					info->rx_skb = NULL;
  					break;
  
  				}
  
  			}
  
  		}
  
  		/* Make sure we don't stay here too long */
  		if (boguscount++ > 16)
  			break;
  
  	} while (inb(iobase + UART_LSR) & UART_LSR_DR);
  }
7d12e780e   David Howells   IRQ: Maintain reg...
273
  static irqreturn_t btuart_interrupt(int irq, void *dev_inst)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
274
  {
ad709d486   Himangi Saraogi   Bluetooth: Remove...
275
  	struct btuart_info *info = dev_inst;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
276
277
278
  	unsigned int iobase;
  	int boguscount = 0;
  	int iir, lsr;
aafcf998c   Alan Cox   pcmcia: IRQ_TYPE_...
279
  	irqreturn_t r = IRQ_NONE;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
280

7427847d2   Mike Frysinger   Bluetooth: Redo c...
281
282
283
  	if (!info || !info->hdev)
  		/* our irq handler is shared */
  		return IRQ_NONE;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
284

9a017a910   Dominik Brodowski   pcmcia: do not us...
285
  	iobase = info->p_dev->resource[0]->start;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
286
287
288
289
290
  
  	spin_lock(&(info->lock));
  
  	iir = inb(iobase + UART_IIR) & UART_IIR_ID;
  	while (iir) {
aafcf998c   Alan Cox   pcmcia: IRQ_TYPE_...
291
  		r = IRQ_HANDLED;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
  
  		/* Clear interrupt */
  		lsr = inb(iobase + UART_LSR);
  
  		switch (iir) {
  		case UART_IIR_RLSI:
  			BT_ERR("RLSI");
  			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:
  			BT_ERR("Unhandled IIR=%#x", iir);
  			break;
  		}
  
  		/* Make sure we don't stay here too long */
  		if (boguscount++ > 100)
  			break;
  
  		iir = inb(iobase + UART_IIR) & UART_IIR_ID;
  
  	}
  
  	spin_unlock(&(info->lock));
aafcf998c   Alan Cox   pcmcia: IRQ_TYPE_...
324
  	return r;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
325
  }
ad709d486   Himangi Saraogi   Bluetooth: Remove...
326
327
  static void btuart_change_speed(struct btuart_info *info,
  				unsigned int speed)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
328
329
330
331
332
333
334
335
336
337
338
  {
  	unsigned long flags;
  	unsigned int iobase;
  	int fcr;		/* FIFO control reg */
  	int lcr;		/* Line control reg */
  	int divisor;
  
  	if (!info) {
  		BT_ERR("Unknown device");
  		return;
  	}
9a017a910   Dominik Brodowski   pcmcia: do not us...
339
  	iobase = info->p_dev->resource[0]->start;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
  
  	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 */
b92b1c572   Joe Perches   drivers/bluetooth...
369
  	/* Turn on interrupts */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
370
371
372
373
374
375
376
377
378
379
380
381
  	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)
  {
ad709d486   Himangi Saraogi   Bluetooth: Remove...
382
  	struct btuart_info *info = hci_get_drvdata(hdev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
383
384
385
386
387
388
389
390
391
392
  
  	/* Drop TX queue */
  	skb_queue_purge(&(info->txq));
  
  	return 0;
  }
  
  
  static int btuart_hci_open(struct hci_dev *hdev)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
393
394
395
396
397
398
  	return 0;
  }
  
  
  static int btuart_hci_close(struct hci_dev *hdev)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
399
400
401
402
  	btuart_hci_flush(hdev);
  
  	return 0;
  }
7bd8f09f6   Marcel Holtmann   Bluetooth: Add hd...
403
  static int btuart_hci_send_frame(struct hci_dev *hdev, struct sk_buff *skb)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
404
  {
ad709d486   Himangi Saraogi   Bluetooth: Remove...
405
  	struct btuart_info *info = hci_get_drvdata(hdev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
406

618e8bc22   Marcel Holtmann   Bluetooth: Use ne...
407
  	switch (hci_skb_pkt_type(skb)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
408
409
410
411
412
413
414
415
416
  	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;
5ad777958   Peter Senna Tschudin   Bluetooth: btuart...
417
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
418
419
  
  	/* Prepend skb with frame type */
618e8bc22   Marcel Holtmann   Bluetooth: Use ne...
420
  	memcpy(skb_push(skb, 1), &hci_skb_pkt_type(skb), 1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
421
422
423
424
425
426
  	skb_queue_tail(&(info->txq), skb);
  
  	btuart_write_wakeup(info);
  
  	return 0;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
427
428
  
  /* ======================== Card services HCI interaction ======================== */
ad709d486   Himangi Saraogi   Bluetooth: Remove...
429
  static int btuart_open(struct btuart_info *info)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
430
431
  {
  	unsigned long flags;
9a017a910   Dominik Brodowski   pcmcia: do not us...
432
  	unsigned int iobase = info->p_dev->resource[0]->start;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
  	struct hci_dev *hdev;
  
  	spin_lock_init(&(info->lock));
  
  	skb_queue_head_init(&(info->txq));
  
  	info->rx_state = RECV_WAIT_PACKET_TYPE;
  	info->rx_count = 0;
  	info->rx_skb = NULL;
  
  	/* Initialize HCI device */
  	hdev = hci_alloc_dev();
  	if (!hdev) {
  		BT_ERR("Can't allocate HCI device");
  		return -ENOMEM;
  	}
  
  	info->hdev = hdev;
c13854cef   Marcel Holtmann   Bluetooth: Conver...
451
  	hdev->bus = HCI_PCCARD;
155961e80   David Herrmann   Bluetooth: Remove...
452
  	hci_set_drvdata(hdev, info);
27d352842   Marcel Holtmann   [Bluetooth] Add p...
453
  	SET_HCIDEV_DEV(hdev, &info->p_dev->dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
454

71f390304   Marcel Holtmann   Bluetooth: Remove...
455
456
457
458
  	hdev->open  = btuart_hci_open;
  	hdev->close = btuart_hci_close;
  	hdev->flush = btuart_hci_flush;
  	hdev->send  = btuart_hci_send_frame;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
459

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
  	spin_lock_irqsave(&(info->lock), flags);
  
  	/* Reset UART */
  	outb(0, iobase + UART_MCR);
  
  	/* Turn off interrupts */
  	outb(0, iobase + UART_IER);
  
  	/* Initialize UART */
  	outb(UART_LCR_WLEN8, iobase + UART_LCR);	/* Reset DLAB */
  	outb((UART_MCR_DTR | UART_MCR_RTS | UART_MCR_OUT2), iobase + UART_MCR);
  
  	/* Turn on interrupts */
  	// outb(UART_IER_RLSI | UART_IER_RDI | UART_IER_THRI, iobase + UART_IER);
  
  	spin_unlock_irqrestore(&(info->lock), flags);
  
  	btuart_change_speed(info, DEFAULT_BAUD_RATE);
  
  	/* Timeout before it is safe to send the first HCI packet */
  	msleep(1000);
  
  	/* Register HCI device */
  	if (hci_register_dev(hdev) < 0) {
  		BT_ERR("Can't register HCI device");
  		info->hdev = NULL;
  		hci_free_dev(hdev);
  		return -ENODEV;
  	}
  
  	return 0;
  }
ad709d486   Himangi Saraogi   Bluetooth: Remove...
492
  static int btuart_close(struct btuart_info *info)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
493
494
  {
  	unsigned long flags;
9a017a910   Dominik Brodowski   pcmcia: do not us...
495
  	unsigned int iobase = info->p_dev->resource[0]->start;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
  	struct hci_dev *hdev = info->hdev;
  
  	if (!hdev)
  		return -ENODEV;
  
  	btuart_hci_close(hdev);
  
  	spin_lock_irqsave(&(info->lock), flags);
  
  	/* Reset UART */
  	outb(0, iobase + UART_MCR);
  
  	/* Turn off interrupts */
  	outb(0, iobase + UART_IER);
  
  	spin_unlock_irqrestore(&(info->lock), flags);
13ea4015d   David Herrmann   Bluetooth: Ignore...
512
  	hci_unregister_dev(hdev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
513
514
515
516
  	hci_free_dev(hdev);
  
  	return 0;
  }
15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
517
  static int btuart_probe(struct pcmcia_device *link)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
518
  {
ad709d486   Himangi Saraogi   Bluetooth: Remove...
519
  	struct btuart_info *info;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
520
521
  
  	/* Create new info device */
fdefa118e   Sachin Kamat   Bluetooth: Use de...
522
  	info = devm_kzalloc(&link->dev, sizeof(*info), GFP_KERNEL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
523
  	if (!info)
f8cfa618d   Dominik Brodowski   [PATCH] pcmcia: u...
524
  		return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
525

fba395eee   Dominik Brodowski   [PATCH] pcmcia: r...
526
  	info->p_dev = link;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527
  	link->priv = info;
00990e7ce   Dominik Brodowski   pcmcia: use autoc...
528
529
  	link->config_flags |= CONF_ENABLE_IRQ | CONF_AUTO_SET_VPP |
  		CONF_AUTO_SET_IO;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
530

15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
531
  	return btuart_config(link);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
532
  }
fba395eee   Dominik Brodowski   [PATCH] pcmcia: r...
533
  static void btuart_detach(struct pcmcia_device *link)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
534
  {
e2d409636   Dominik Brodowski   [PATCH] pcmcia: u...
535
  	btuart_release(link);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
536
  }
00990e7ce   Dominik Brodowski   pcmcia: use autoc...
537
  static int btuart_check_config(struct pcmcia_device *p_dev, void *priv_data)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
538
  {
ad913c119   Dominik Brodowski   pcmcia: pcmcia_co...
539
  	int *try = priv_data;
90abdc3b9   Dominik Brodowski   pcmcia: do not us...
540

510df2515   Andrei Emeltchenko   Bluetooth: Fix wa...
541
  	if (!try)
00990e7ce   Dominik Brodowski   pcmcia: use autoc...
542
543
544
545
546
547
548
549
550
551
  		p_dev->io_lines = 16;
  
  	if ((p_dev->resource[0]->end != 8) || (p_dev->resource[0]->start == 0))
  		return -EINVAL;
  
  	p_dev->resource[0]->end = 8;
  	p_dev->resource[0]->flags &= ~IO_DATA_PATH_WIDTH;
  	p_dev->resource[0]->flags |= IO_DATA_PATH_WIDTH_8;
  
  	return pcmcia_request_io(p_dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
552
  }
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
553
  static int btuart_check_config_notpicky(struct pcmcia_device *p_dev,
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
554
  					void *priv_data)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
555
  {
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
556
557
  	static unsigned int base[5] = { 0x3f8, 0x2f8, 0x3e8, 0x2e8, 0x0 };
  	int j;
00990e7ce   Dominik Brodowski   pcmcia: use autoc...
558
559
560
561
562
563
564
565
566
567
568
569
  	if (p_dev->io_lines > 3)
  		return -ENODEV;
  
  	p_dev->resource[0]->flags &= ~IO_DATA_PATH_WIDTH;
  	p_dev->resource[0]->flags |= IO_DATA_PATH_WIDTH_8;
  	p_dev->resource[0]->end = 8;
  
  	for (j = 0; j < 5; j++) {
  		p_dev->resource[0]->start = base[j];
  		p_dev->io_lines = base[j] ? 16 : 3;
  		if (!pcmcia_request_io(p_dev))
  			return 0;
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
570
571
  	}
  	return -ENODEV;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
572
  }
15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
573
  static int btuart_config(struct pcmcia_device *link)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
574
  {
ad709d486   Himangi Saraogi   Bluetooth: Remove...
575
  	struct btuart_info *info = link->priv;
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
576
  	int i;
ad913c119   Dominik Brodowski   pcmcia: pcmcia_co...
577
  	int try;
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
578
579
580
581
  
  	/* First pass: look for a config entry that looks normal.
  	   Two tries: without IO aliases, then with aliases */
  	for (try = 0; try < 2; try++)
ad913c119   Dominik Brodowski   pcmcia: pcmcia_co...
582
  		if (!pcmcia_loop_config(link, btuart_check_config, &try))
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
583
  			goto found_port;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
584
585
586
587
  
  	/* Second pass: try to find an entry that isn't picky about
  	   its base address, then try to grab any standard serial port
  	   address, and finally try to get any free port. */
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
588
589
  	if (!pcmcia_loop_config(link, btuart_check_config_notpicky, NULL))
  		goto found_port;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
590

ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
591
  	BT_ERR("No usable port range found");
ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
592
  	goto failed;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
593

ed58872aa   Dominik Brodowski   pcmcia: use pcmci...
594
  found_port:
eb14120f7   Dominik Brodowski   pcmcia: re-work p...
595
  	i = pcmcia_request_irq(link, btuart_interrupt);
9ac3e58ce   Dominik Brodowski   pcmcia: deprecate...
596
  	if (i != 0)
eb14120f7   Dominik Brodowski   pcmcia: re-work p...
597
  		goto failed;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
598

1ac71e5a3   Dominik Brodowski   pcmcia: convert p...
599
  	i = pcmcia_enable_device(link);
9ac3e58ce   Dominik Brodowski   pcmcia: deprecate...
600
  	if (i != 0)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
601
  		goto failed;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
602
603
604
  
  	if (btuart_open(info) != 0)
  		goto failed;
15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
605
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
606

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
607
608
  failed:
  	btuart_release(link);
15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
609
  	return -ENODEV;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
610
  }
fba395eee   Dominik Brodowski   [PATCH] pcmcia: r...
611
  static void btuart_release(struct pcmcia_device *link)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
612
  {
ad709d486   Himangi Saraogi   Bluetooth: Remove...
613
  	struct btuart_info *info = link->priv;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
614

e2d409636   Dominik Brodowski   [PATCH] pcmcia: u...
615
  	btuart_close(info);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
616

fba395eee   Dominik Brodowski   [PATCH] pcmcia: r...
617
  	pcmcia_disable_device(link);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
618
  }
25f8f54f6   Joe Perches   pcmcia: Convert p...
619
  static const struct pcmcia_device_id btuart_ids[] = {
279c93615   Dominik Brodowski   [PATCH] pcmcia: i...
620
621
622
623
  	/* don't use this driver. Use serial_cs + hci_uart instead */
  	PCMCIA_DEVICE_NULL
  };
  MODULE_DEVICE_TABLE(pcmcia, btuart_ids);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
624
625
  static struct pcmcia_driver btuart_driver = {
  	.owner		= THIS_MODULE,
2e9b981a7   Dominik Brodowski   pcmcia: move driv...
626
  	.name		= "btuart_cs",
15b99ac17   Dominik Brodowski   [PATCH] pcmcia: a...
627
  	.probe		= btuart_probe,
cc3b4866b   Dominik Brodowski   [PATCH] pcmcia: u...
628
  	.remove		= btuart_detach,
279c93615   Dominik Brodowski   [PATCH] pcmcia: i...
629
  	.id_table	= btuart_ids,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
630
  };
e0c005f4b   H Hartley Sweeten   drivers/bluetooth...
631
  module_pcmcia_driver(btuart_driver);