1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-or-later
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Broadcom B43 wireless driver
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * SDIO over Sonics Silicon Backplane bus glue for b43.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Copyright (C) 2009 Albert Herranz
8*4882a593Smuzhiyun * Copyright (C) 2009 Michael Buesch <m@bues.ch>
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <linux/kernel.h>
12*4882a593Smuzhiyun #include <linux/mmc/card.h>
13*4882a593Smuzhiyun #include <linux/mmc/sdio_func.h>
14*4882a593Smuzhiyun #include <linux/mmc/sdio_ids.h>
15*4882a593Smuzhiyun #include <linux/slab.h>
16*4882a593Smuzhiyun #include <linux/ssb/ssb.h>
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun #include "sdio.h"
19*4882a593Smuzhiyun #include "b43.h"
20*4882a593Smuzhiyun
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun #define HNBU_CHIPID 0x01 /* vendor & device id */
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #define B43_SDIO_BLOCK_SIZE 64 /* rx fifo max size in bytes */
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun static const struct b43_sdio_quirk {
28*4882a593Smuzhiyun u16 vendor;
29*4882a593Smuzhiyun u16 device;
30*4882a593Smuzhiyun unsigned int quirks;
31*4882a593Smuzhiyun } b43_sdio_quirks[] = {
32*4882a593Smuzhiyun { 0x14E4, 0x4318, SSB_QUIRK_SDIO_READ_AFTER_WRITE32, },
33*4882a593Smuzhiyun { },
34*4882a593Smuzhiyun };
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun
b43_sdio_get_quirks(u16 vendor,u16 device)37*4882a593Smuzhiyun static unsigned int b43_sdio_get_quirks(u16 vendor, u16 device)
38*4882a593Smuzhiyun {
39*4882a593Smuzhiyun const struct b43_sdio_quirk *q;
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun for (q = b43_sdio_quirks; q->quirks; q++) {
42*4882a593Smuzhiyun if (vendor == q->vendor && device == q->device)
43*4882a593Smuzhiyun return q->quirks;
44*4882a593Smuzhiyun }
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun return 0;
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun
b43_sdio_interrupt_dispatcher(struct sdio_func * func)49*4882a593Smuzhiyun static void b43_sdio_interrupt_dispatcher(struct sdio_func *func)
50*4882a593Smuzhiyun {
51*4882a593Smuzhiyun struct b43_sdio *sdio = sdio_get_drvdata(func);
52*4882a593Smuzhiyun struct b43_wldev *dev = sdio->irq_handler_opaque;
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun if (unlikely(b43_status(dev) < B43_STAT_STARTED))
55*4882a593Smuzhiyun return;
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun sdio_release_host(func);
58*4882a593Smuzhiyun sdio->irq_handler(dev);
59*4882a593Smuzhiyun sdio_claim_host(func);
60*4882a593Smuzhiyun }
61*4882a593Smuzhiyun
b43_sdio_request_irq(struct b43_wldev * dev,void (* handler)(struct b43_wldev * dev))62*4882a593Smuzhiyun int b43_sdio_request_irq(struct b43_wldev *dev,
63*4882a593Smuzhiyun void (*handler)(struct b43_wldev *dev))
64*4882a593Smuzhiyun {
65*4882a593Smuzhiyun struct ssb_bus *bus = dev->dev->sdev->bus;
66*4882a593Smuzhiyun struct sdio_func *func = bus->host_sdio;
67*4882a593Smuzhiyun struct b43_sdio *sdio = sdio_get_drvdata(func);
68*4882a593Smuzhiyun int err;
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun sdio->irq_handler_opaque = dev;
71*4882a593Smuzhiyun sdio->irq_handler = handler;
72*4882a593Smuzhiyun sdio_claim_host(func);
73*4882a593Smuzhiyun err = sdio_claim_irq(func, b43_sdio_interrupt_dispatcher);
74*4882a593Smuzhiyun sdio_release_host(func);
75*4882a593Smuzhiyun
76*4882a593Smuzhiyun return err;
77*4882a593Smuzhiyun }
78*4882a593Smuzhiyun
b43_sdio_free_irq(struct b43_wldev * dev)79*4882a593Smuzhiyun void b43_sdio_free_irq(struct b43_wldev *dev)
80*4882a593Smuzhiyun {
81*4882a593Smuzhiyun struct ssb_bus *bus = dev->dev->sdev->bus;
82*4882a593Smuzhiyun struct sdio_func *func = bus->host_sdio;
83*4882a593Smuzhiyun struct b43_sdio *sdio = sdio_get_drvdata(func);
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun sdio_claim_host(func);
86*4882a593Smuzhiyun sdio_release_irq(func);
87*4882a593Smuzhiyun sdio_release_host(func);
88*4882a593Smuzhiyun sdio->irq_handler_opaque = NULL;
89*4882a593Smuzhiyun sdio->irq_handler = NULL;
90*4882a593Smuzhiyun }
91*4882a593Smuzhiyun
b43_sdio_probe(struct sdio_func * func,const struct sdio_device_id * id)92*4882a593Smuzhiyun static int b43_sdio_probe(struct sdio_func *func,
93*4882a593Smuzhiyun const struct sdio_device_id *id)
94*4882a593Smuzhiyun {
95*4882a593Smuzhiyun struct b43_sdio *sdio;
96*4882a593Smuzhiyun struct sdio_func_tuple *tuple;
97*4882a593Smuzhiyun u16 vendor = 0, device = 0;
98*4882a593Smuzhiyun int error;
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun /* Look for the card chip identifier. */
101*4882a593Smuzhiyun tuple = func->tuples;
102*4882a593Smuzhiyun while (tuple) {
103*4882a593Smuzhiyun switch (tuple->code) {
104*4882a593Smuzhiyun case 0x80:
105*4882a593Smuzhiyun switch (tuple->data[0]) {
106*4882a593Smuzhiyun case HNBU_CHIPID:
107*4882a593Smuzhiyun if (tuple->size != 5)
108*4882a593Smuzhiyun break;
109*4882a593Smuzhiyun vendor = tuple->data[1] | (tuple->data[2]<<8);
110*4882a593Smuzhiyun device = tuple->data[3] | (tuple->data[4]<<8);
111*4882a593Smuzhiyun dev_info(&func->dev, "Chip ID %04x:%04x\n",
112*4882a593Smuzhiyun vendor, device);
113*4882a593Smuzhiyun break;
114*4882a593Smuzhiyun default:
115*4882a593Smuzhiyun break;
116*4882a593Smuzhiyun }
117*4882a593Smuzhiyun break;
118*4882a593Smuzhiyun default:
119*4882a593Smuzhiyun break;
120*4882a593Smuzhiyun }
121*4882a593Smuzhiyun tuple = tuple->next;
122*4882a593Smuzhiyun }
123*4882a593Smuzhiyun if (!vendor || !device) {
124*4882a593Smuzhiyun error = -ENODEV;
125*4882a593Smuzhiyun goto out;
126*4882a593Smuzhiyun }
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun sdio_claim_host(func);
129*4882a593Smuzhiyun error = sdio_set_block_size(func, B43_SDIO_BLOCK_SIZE);
130*4882a593Smuzhiyun if (error) {
131*4882a593Smuzhiyun dev_err(&func->dev, "failed to set block size to %u bytes,"
132*4882a593Smuzhiyun " error %d\n", B43_SDIO_BLOCK_SIZE, error);
133*4882a593Smuzhiyun goto err_release_host;
134*4882a593Smuzhiyun }
135*4882a593Smuzhiyun error = sdio_enable_func(func);
136*4882a593Smuzhiyun if (error) {
137*4882a593Smuzhiyun dev_err(&func->dev, "failed to enable func, error %d\n", error);
138*4882a593Smuzhiyun goto err_release_host;
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun sdio_release_host(func);
141*4882a593Smuzhiyun
142*4882a593Smuzhiyun sdio = kzalloc(sizeof(*sdio), GFP_KERNEL);
143*4882a593Smuzhiyun if (!sdio) {
144*4882a593Smuzhiyun error = -ENOMEM;
145*4882a593Smuzhiyun dev_err(&func->dev, "failed to allocate ssb bus\n");
146*4882a593Smuzhiyun goto err_disable_func;
147*4882a593Smuzhiyun }
148*4882a593Smuzhiyun error = ssb_bus_sdiobus_register(&sdio->ssb, func,
149*4882a593Smuzhiyun b43_sdio_get_quirks(vendor, device));
150*4882a593Smuzhiyun if (error) {
151*4882a593Smuzhiyun dev_err(&func->dev, "failed to register ssb sdio bus,"
152*4882a593Smuzhiyun " error %d\n", error);
153*4882a593Smuzhiyun goto err_free_ssb;
154*4882a593Smuzhiyun }
155*4882a593Smuzhiyun sdio_set_drvdata(func, sdio);
156*4882a593Smuzhiyun
157*4882a593Smuzhiyun return 0;
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun err_free_ssb:
160*4882a593Smuzhiyun kfree(sdio);
161*4882a593Smuzhiyun err_disable_func:
162*4882a593Smuzhiyun sdio_claim_host(func);
163*4882a593Smuzhiyun sdio_disable_func(func);
164*4882a593Smuzhiyun err_release_host:
165*4882a593Smuzhiyun sdio_release_host(func);
166*4882a593Smuzhiyun out:
167*4882a593Smuzhiyun return error;
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun
b43_sdio_remove(struct sdio_func * func)170*4882a593Smuzhiyun static void b43_sdio_remove(struct sdio_func *func)
171*4882a593Smuzhiyun {
172*4882a593Smuzhiyun struct b43_sdio *sdio = sdio_get_drvdata(func);
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun ssb_bus_unregister(&sdio->ssb);
175*4882a593Smuzhiyun sdio_claim_host(func);
176*4882a593Smuzhiyun sdio_disable_func(func);
177*4882a593Smuzhiyun sdio_release_host(func);
178*4882a593Smuzhiyun kfree(sdio);
179*4882a593Smuzhiyun sdio_set_drvdata(func, NULL);
180*4882a593Smuzhiyun }
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun static const struct sdio_device_id b43_sdio_ids[] = {
183*4882a593Smuzhiyun { SDIO_DEVICE(SDIO_VENDOR_ID_BROADCOM, SDIO_DEVICE_ID_BROADCOM_NINTENDO_WII) },
184*4882a593Smuzhiyun { SDIO_DEVICE(SDIO_VENDOR_ID_CGUYS, SDIO_DEVICE_ID_CGUYS_EW_CG1102GC) },
185*4882a593Smuzhiyun { },
186*4882a593Smuzhiyun };
187*4882a593Smuzhiyun
188*4882a593Smuzhiyun static struct sdio_driver b43_sdio_driver = {
189*4882a593Smuzhiyun .name = "b43-sdio",
190*4882a593Smuzhiyun .id_table = b43_sdio_ids,
191*4882a593Smuzhiyun .probe = b43_sdio_probe,
192*4882a593Smuzhiyun .remove = b43_sdio_remove,
193*4882a593Smuzhiyun };
194*4882a593Smuzhiyun
b43_sdio_init(void)195*4882a593Smuzhiyun int b43_sdio_init(void)
196*4882a593Smuzhiyun {
197*4882a593Smuzhiyun return sdio_register_driver(&b43_sdio_driver);
198*4882a593Smuzhiyun }
199*4882a593Smuzhiyun
b43_sdio_exit(void)200*4882a593Smuzhiyun void b43_sdio_exit(void)
201*4882a593Smuzhiyun {
202*4882a593Smuzhiyun sdio_unregister_driver(&b43_sdio_driver);
203*4882a593Smuzhiyun }
204