blob: 881a7938c4942cfc2e1d2948742f17cb1be18b7c [file] [log] [blame]
Thomas Gleixner2874c5f2019-05-27 08:55:01 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Albert Herranz3dbba8e22009-09-10 19:34:49 +02002/*
3 * Broadcom B43 wireless driver
4 *
5 * SDIO over Sonics Silicon Backplane bus glue for b43.
6 *
7 * Copyright (C) 2009 Albert Herranz
Michael Büscheb032b92011-07-04 20:50:05 +02008 * Copyright (C) 2009 Michael Buesch <m@bues.ch>
Albert Herranz3dbba8e22009-09-10 19:34:49 +02009 */
10
11#include <linux/kernel.h>
12#include <linux/mmc/card.h>
13#include <linux/mmc/sdio_func.h>
14#include <linux/mmc/sdio_ids.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090015#include <linux/slab.h>
Albert Herranz3dbba8e22009-09-10 19:34:49 +020016#include <linux/ssb/ssb.h>
17
18#include "sdio.h"
19#include "b43.h"
20
21
22#define HNBU_CHIPID 0x01 /* vendor & device id */
23
24#define B43_SDIO_BLOCK_SIZE 64 /* rx fifo max size in bytes */
25
26
27static const struct b43_sdio_quirk {
28 u16 vendor;
29 u16 device;
30 unsigned int quirks;
31} b43_sdio_quirks[] = {
32 { 0x14E4, 0x4318, SSB_QUIRK_SDIO_READ_AFTER_WRITE32, },
33 { },
34};
35
36
37static unsigned int b43_sdio_get_quirks(u16 vendor, u16 device)
38{
39 const struct b43_sdio_quirk *q;
40
41 for (q = b43_sdio_quirks; q->quirks; q++) {
42 if (vendor == q->vendor && device == q->device)
43 return q->quirks;
44 }
45
46 return 0;
47}
48
49static void b43_sdio_interrupt_dispatcher(struct sdio_func *func)
50{
51 struct b43_sdio *sdio = sdio_get_drvdata(func);
52 struct b43_wldev *dev = sdio->irq_handler_opaque;
53
Michael Buescha8696c82009-09-11 16:00:19 +020054 if (unlikely(b43_status(dev) < B43_STAT_STARTED))
55 return;
56
57 sdio_release_host(func);
Albert Herranz3dbba8e22009-09-10 19:34:49 +020058 sdio->irq_handler(dev);
Michael Buescha8696c82009-09-11 16:00:19 +020059 sdio_claim_host(func);
Albert Herranz3dbba8e22009-09-10 19:34:49 +020060}
61
62int b43_sdio_request_irq(struct b43_wldev *dev,
63 void (*handler)(struct b43_wldev *dev))
64{
Rafał Miłecki02a0fbe2011-05-19 15:11:25 +020065 struct ssb_bus *bus = dev->dev->sdev->bus;
Albert Herranz3dbba8e22009-09-10 19:34:49 +020066 struct sdio_func *func = bus->host_sdio;
67 struct b43_sdio *sdio = sdio_get_drvdata(func);
68 int err;
69
70 sdio->irq_handler_opaque = dev;
71 sdio->irq_handler = handler;
72 sdio_claim_host(func);
73 err = sdio_claim_irq(func, b43_sdio_interrupt_dispatcher);
74 sdio_release_host(func);
75
76 return err;
77}
78
79void b43_sdio_free_irq(struct b43_wldev *dev)
80{
Rafał Miłecki02a0fbe2011-05-19 15:11:25 +020081 struct ssb_bus *bus = dev->dev->sdev->bus;
Albert Herranz3dbba8e22009-09-10 19:34:49 +020082 struct sdio_func *func = bus->host_sdio;
83 struct b43_sdio *sdio = sdio_get_drvdata(func);
84
85 sdio_claim_host(func);
86 sdio_release_irq(func);
87 sdio_release_host(func);
88 sdio->irq_handler_opaque = NULL;
89 sdio->irq_handler = NULL;
90}
91
Bill Pemberton157c9432012-12-03 09:56:30 -050092static int b43_sdio_probe(struct sdio_func *func,
Hauke Mehrtenscd155982011-06-21 20:57:16 +020093 const struct sdio_device_id *id)
Albert Herranz3dbba8e22009-09-10 19:34:49 +020094{
95 struct b43_sdio *sdio;
96 struct sdio_func_tuple *tuple;
97 u16 vendor = 0, device = 0;
98 int error;
99
100 /* Look for the card chip identifier. */
101 tuple = func->tuples;
102 while (tuple) {
103 switch (tuple->code) {
104 case 0x80:
105 switch (tuple->data[0]) {
106 case HNBU_CHIPID:
107 if (tuple->size != 5)
108 break;
109 vendor = tuple->data[1] | (tuple->data[2]<<8);
110 device = tuple->data[3] | (tuple->data[4]<<8);
111 dev_info(&func->dev, "Chip ID %04x:%04x\n",
112 vendor, device);
113 break;
114 default:
115 break;
116 }
117 break;
118 default:
119 break;
120 }
121 tuple = tuple->next;
122 }
123 if (!vendor || !device) {
124 error = -ENODEV;
125 goto out;
126 }
127
128 sdio_claim_host(func);
129 error = sdio_set_block_size(func, B43_SDIO_BLOCK_SIZE);
130 if (error) {
131 dev_err(&func->dev, "failed to set block size to %u bytes,"
132 " error %d\n", B43_SDIO_BLOCK_SIZE, error);
133 goto err_release_host;
134 }
135 error = sdio_enable_func(func);
136 if (error) {
137 dev_err(&func->dev, "failed to enable func, error %d\n", error);
138 goto err_release_host;
139 }
140 sdio_release_host(func);
141
142 sdio = kzalloc(sizeof(*sdio), GFP_KERNEL);
143 if (!sdio) {
144 error = -ENOMEM;
145 dev_err(&func->dev, "failed to allocate ssb bus\n");
146 goto err_disable_func;
147 }
148 error = ssb_bus_sdiobus_register(&sdio->ssb, func,
149 b43_sdio_get_quirks(vendor, device));
150 if (error) {
151 dev_err(&func->dev, "failed to register ssb sdio bus,"
152 " error %d\n", error);
153 goto err_free_ssb;
154 }
155 sdio_set_drvdata(func, sdio);
156
157 return 0;
158
159err_free_ssb:
160 kfree(sdio);
161err_disable_func:
Guennadi Liakhovetskie476a5a2010-11-23 17:10:24 +0100162 sdio_claim_host(func);
Albert Herranz3dbba8e22009-09-10 19:34:49 +0200163 sdio_disable_func(func);
164err_release_host:
165 sdio_release_host(func);
166out:
167 return error;
168}
169
Bill Pemberton157c9432012-12-03 09:56:30 -0500170static void b43_sdio_remove(struct sdio_func *func)
Albert Herranz3dbba8e22009-09-10 19:34:49 +0200171{
172 struct b43_sdio *sdio = sdio_get_drvdata(func);
173
174 ssb_bus_unregister(&sdio->ssb);
Larry Finger9f2a0fa2010-10-28 10:43:26 -0500175 sdio_claim_host(func);
Albert Herranz3dbba8e22009-09-10 19:34:49 +0200176 sdio_disable_func(func);
Larry Finger9f2a0fa2010-10-28 10:43:26 -0500177 sdio_release_host(func);
Albert Herranz3dbba8e22009-09-10 19:34:49 +0200178 kfree(sdio);
179 sdio_set_drvdata(func, NULL);
180}
181
182static const struct sdio_device_id b43_sdio_ids[] = {
183 { SDIO_DEVICE(0x02d0, 0x044b) }, /* Nintendo Wii WLAN daughter card */
Magnus Damm55d02a42010-06-25 18:32:54 +0900184 { SDIO_DEVICE(0x0092, 0x0004) }, /* C-guys, Inc. EW-CG1102GC */
Albert Herranz3dbba8e22009-09-10 19:34:49 +0200185 { },
186};
187
188static struct sdio_driver b43_sdio_driver = {
189 .name = "b43-sdio",
190 .id_table = b43_sdio_ids,
191 .probe = b43_sdio_probe,
Bill Pemberton157c9432012-12-03 09:56:30 -0500192 .remove = b43_sdio_remove,
Albert Herranz3dbba8e22009-09-10 19:34:49 +0200193};
194
195int b43_sdio_init(void)
196{
197 return sdio_register_driver(&b43_sdio_driver);
198}
199
200void b43_sdio_exit(void)
201{
202 sdio_unregister_driver(&b43_sdio_driver);
203}