blob: 5d788a619642df828c4656f3762fc28b510718ab [file] [log] [blame]
Franky Lina83369b2011-11-04 22:23:28 +01001/*
2 * Copyright (c) 2011 Broadcom Corporation
3 *
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
11 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
13 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
14 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16/* ***** SDIO interface chip backplane handle functions ***** */
17
18#include <linux/types.h>
19#include <linux/netdevice.h>
20#include <linux/mmc/card.h>
21#include <chipcommon.h>
22#include <brcm_hw_ids.h>
23#include <brcmu_wifi.h>
24#include <brcmu_utils.h>
Franky Lin2d4a9af2011-11-04 22:23:31 +010025#include <soc.h>
Franky Lina83369b2011-11-04 22:23:28 +010026#include "dhd.h"
27#include "dhd_dbg.h"
28#include "sdio_host.h"
29#include "sdio_chip.h"
30
31/* chip core base & ramsize */
32/* bcm4329 */
33/* SDIO device core, ID 0x829 */
34#define BCM4329_CORE_BUS_BASE 0x18011000
35/* internal memory core, ID 0x80e */
36#define BCM4329_CORE_SOCRAM_BASE 0x18003000
37/* ARM Cortex M3 core, ID 0x82a */
38#define BCM4329_CORE_ARM_BASE 0x18002000
39#define BCM4329_RAMSIZE 0x48000
40
41
42/* SB regs */
43/* sbidhigh */
44#define SBIDH_RC_MASK 0x000f /* revision code */
45#define SBIDH_RCE_MASK 0x7000 /* revision code extension field */
46#define SBIDH_RCE_SHIFT 8
47#define SBCOREREV(sbidh) \
48 ((((sbidh) & SBIDH_RCE_MASK) >> SBIDH_RCE_SHIFT) | \
49 ((sbidh) & SBIDH_RC_MASK))
50#define SBIDH_CC_MASK 0x8ff0 /* core code */
51#define SBIDH_CC_SHIFT 4
52#define SBIDH_VC_MASK 0xffff0000 /* vendor code */
53#define SBIDH_VC_SHIFT 16
54
Franky Lin454d2a82011-11-04 22:23:37 +010055static u32
56brcmf_sdio_chip_corerev(struct brcmf_sdio_dev *sdiodev,
57 u32 corebase)
58{
59 u32 regdata;
60
61 regdata = brcmf_sdcard_reg_read(sdiodev,
62 CORE_SB(corebase, sbidhigh), 4);
63 return SBCOREREV(regdata);
64}
65
Franky Lind8f64a42011-11-04 22:23:36 +010066bool
67brcmf_sdio_chip_iscoreup(struct brcmf_sdio_dev *sdiodev,
68 u32 corebase)
69{
70 u32 regdata;
71
72 regdata = brcmf_sdcard_reg_read(sdiodev,
73 CORE_SB(corebase, sbtmstatelow), 4);
74 regdata &= (SBTML_RESET | SBTML_REJ_MASK |
75 (SICF_CLOCK_EN << SBTML_SICF_SHIFT));
76 return ((SICF_CLOCK_EN << SBTML_SICF_SHIFT) == regdata);
77}
78
Franky Lin2d4a9af2011-11-04 22:23:31 +010079void
80brcmf_sdio_chip_coredisable(struct brcmf_sdio_dev *sdiodev, u32 corebase)
81{
82 u32 regdata;
83
84 regdata = brcmf_sdcard_reg_read(sdiodev,
85 CORE_SB(corebase, sbtmstatelow), 4);
86 if (regdata & SBTML_RESET)
87 return;
88
89 regdata = brcmf_sdcard_reg_read(sdiodev,
90 CORE_SB(corebase, sbtmstatelow), 4);
91 if ((regdata & (SICF_CLOCK_EN << SBTML_SICF_SHIFT)) != 0) {
92 /*
93 * set target reject and spin until busy is clear
94 * (preserve core-specific bits)
95 */
96 regdata = brcmf_sdcard_reg_read(sdiodev,
97 CORE_SB(corebase, sbtmstatelow), 4);
98 brcmf_sdcard_reg_write(sdiodev, CORE_SB(corebase, sbtmstatelow),
99 4, regdata | SBTML_REJ);
100
101 regdata = brcmf_sdcard_reg_read(sdiodev,
102 CORE_SB(corebase, sbtmstatelow), 4);
103 udelay(1);
104 SPINWAIT((brcmf_sdcard_reg_read(sdiodev,
105 CORE_SB(corebase, sbtmstatehigh), 4) &
106 SBTMH_BUSY), 100000);
107
108 regdata = brcmf_sdcard_reg_read(sdiodev,
109 CORE_SB(corebase, sbtmstatehigh), 4);
110 if (regdata & SBTMH_BUSY)
111 brcmf_dbg(ERROR, "core state still busy\n");
112
113 regdata = brcmf_sdcard_reg_read(sdiodev,
114 CORE_SB(corebase, sbidlow), 4);
115 if (regdata & SBIDL_INIT) {
116 regdata = brcmf_sdcard_reg_read(sdiodev,
117 CORE_SB(corebase, sbimstate), 4) |
118 SBIM_RJ;
119 brcmf_sdcard_reg_write(sdiodev,
120 CORE_SB(corebase, sbimstate), 4,
121 regdata);
122 regdata = brcmf_sdcard_reg_read(sdiodev,
123 CORE_SB(corebase, sbimstate), 4);
124 udelay(1);
125 SPINWAIT((brcmf_sdcard_reg_read(sdiodev,
126 CORE_SB(corebase, sbimstate), 4) &
127 SBIM_BY), 100000);
128 }
129
130 /* set reset and reject while enabling the clocks */
131 brcmf_sdcard_reg_write(sdiodev,
132 CORE_SB(corebase, sbtmstatelow), 4,
133 (((SICF_FGC | SICF_CLOCK_EN) << SBTML_SICF_SHIFT) |
134 SBTML_REJ | SBTML_RESET));
135 regdata = brcmf_sdcard_reg_read(sdiodev,
136 CORE_SB(corebase, sbtmstatelow), 4);
137 udelay(10);
138
139 /* clear the initiator reject bit */
140 regdata = brcmf_sdcard_reg_read(sdiodev,
141 CORE_SB(corebase, sbidlow), 4);
142 if (regdata & SBIDL_INIT) {
143 regdata = brcmf_sdcard_reg_read(sdiodev,
144 CORE_SB(corebase, sbimstate), 4) &
145 ~SBIM_RJ;
146 brcmf_sdcard_reg_write(sdiodev,
147 CORE_SB(corebase, sbimstate), 4,
148 regdata);
149 }
150 }
151
152 /* leave reset and reject asserted */
153 brcmf_sdcard_reg_write(sdiodev, CORE_SB(corebase, sbtmstatelow), 4,
154 (SBTML_REJ | SBTML_RESET));
155 udelay(1);
156}
157
Franky Lina83369b2011-11-04 22:23:28 +0100158static int brcmf_sdio_chip_recognition(struct brcmf_sdio_dev *sdiodev,
159 struct chip_info *ci, u32 regs)
160{
161 u32 regdata;
162
163 /*
164 * Get CC core rev
165 * Chipid is assume to be at offset 0 from regs arg
166 * For different chiptypes or old sdio hosts w/o chipcommon,
167 * other ways of recognition should be added here.
168 */
169 ci->cccorebase = regs;
170 regdata = brcmf_sdcard_reg_read(sdiodev,
171 CORE_CC_REG(ci->cccorebase, chipid), 4);
172 ci->chip = regdata & CID_ID_MASK;
173 ci->chiprev = (regdata & CID_REV_MASK) >> CID_REV_SHIFT;
174
175 brcmf_dbg(INFO, "chipid=0x%x chiprev=%d\n", ci->chip, ci->chiprev);
176
177 /* Address of cores for new chips should be added here */
178 switch (ci->chip) {
179 case BCM4329_CHIP_ID:
180 ci->buscorebase = BCM4329_CORE_BUS_BASE;
181 ci->ramcorebase = BCM4329_CORE_SOCRAM_BASE;
182 ci->armcorebase = BCM4329_CORE_ARM_BASE;
183 ci->ramsize = BCM4329_RAMSIZE;
184 break;
185 default:
186 brcmf_dbg(ERROR, "chipid 0x%x is not supported\n", ci->chip);
187 return -ENODEV;
188 }
189
Franky Lina83369b2011-11-04 22:23:28 +0100190 return 0;
191}
192
Franky Line63ac6b2011-11-04 22:23:29 +0100193static int
194brcmf_sdio_chip_buscoreprep(struct brcmf_sdio_dev *sdiodev)
195{
196 int err = 0;
197 u8 clkval, clkset;
198
199 /* Try forcing SDIO core to do ALPAvail request only */
200 clkset = SBSDIO_FORCE_HW_CLKREQ_OFF | SBSDIO_ALP_AVAIL_REQ;
201 brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1,
202 SBSDIO_FUNC1_CHIPCLKCSR, clkset, &err);
203 if (err) {
204 brcmf_dbg(ERROR, "error writing for HT off\n");
205 return err;
206 }
207
208 /* If register supported, wait for ALPAvail and then force ALP */
209 /* This may take up to 15 milliseconds */
210 clkval = brcmf_sdcard_cfg_read(sdiodev, SDIO_FUNC_1,
211 SBSDIO_FUNC1_CHIPCLKCSR, NULL);
212
213 if ((clkval & ~SBSDIO_AVBITS) != clkset) {
214 brcmf_dbg(ERROR, "ChipClkCSR access: wrote 0x%02x read 0x%02x\n",
215 clkset, clkval);
216 return -EACCES;
217 }
218
219 SPINWAIT(((clkval = brcmf_sdcard_cfg_read(sdiodev, SDIO_FUNC_1,
220 SBSDIO_FUNC1_CHIPCLKCSR, NULL)),
221 !SBSDIO_ALPAV(clkval)),
222 PMU_MAX_TRANSITION_DLY);
223 if (!SBSDIO_ALPAV(clkval)) {
224 brcmf_dbg(ERROR, "timeout on ALPAV wait, clkval 0x%02x\n",
225 clkval);
226 return -EBUSY;
227 }
228
229 clkset = SBSDIO_FORCE_HW_CLKREQ_OFF | SBSDIO_FORCE_ALP;
230 brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1,
231 SBSDIO_FUNC1_CHIPCLKCSR, clkset, &err);
232 udelay(65);
233
234 /* Also, disable the extra SDIO pull-ups */
235 brcmf_sdcard_cfg_write(sdiodev, SDIO_FUNC_1,
236 SBSDIO_FUNC1_SDIOPULLUP, 0, NULL);
237
238 return 0;
239}
240
Franky Lin5b45e542011-11-04 22:23:30 +0100241static void
242brcmf_sdio_chip_buscoresetup(struct brcmf_sdio_dev *sdiodev,
243 struct chip_info *ci)
244{
245 u32 regdata;
246
247 /* get chipcommon rev */
Franky Lin454d2a82011-11-04 22:23:37 +0100248 ci->ccrev = brcmf_sdio_chip_corerev(sdiodev, ci->cccorebase);
Franky Lin5b45e542011-11-04 22:23:30 +0100249
250 /* get chipcommon capabilites */
251 ci->cccaps = brcmf_sdcard_reg_read(sdiodev,
252 CORE_CC_REG(ci->cccorebase, capabilities), 4);
253
254 /* get pmu caps & rev */
255 if (ci->cccaps & CC_CAP_PMU) {
256 ci->pmucaps = brcmf_sdcard_reg_read(sdiodev,
257 CORE_CC_REG(ci->cccorebase, pmucapabilities), 4);
258 ci->pmurev = ci->pmucaps & PCAP_REV_MASK;
259 }
260
Franky Lin454d2a82011-11-04 22:23:37 +0100261
262 ci->buscorerev = brcmf_sdio_chip_corerev(sdiodev, ci->buscorebase);
Franky Lin5b45e542011-11-04 22:23:30 +0100263 regdata = brcmf_sdcard_reg_read(sdiodev,
264 CORE_SB(ci->buscorebase, sbidhigh), 4);
Franky Lin5b45e542011-11-04 22:23:30 +0100265 ci->buscoretype = (regdata & SBIDH_CC_MASK) >> SBIDH_CC_SHIFT;
266
267 brcmf_dbg(INFO, "ccrev=%d, pmurev=%d, buscore rev/type=%d/0x%x\n",
268 ci->ccrev, ci->pmurev, ci->buscorerev, ci->buscoretype);
Franky Lin966414d2011-11-04 22:23:32 +0100269
270 /*
271 * Make sure any on-chip ARM is off (in case strapping is wrong),
272 * or downloaded code was already running.
273 */
274 brcmf_sdio_chip_coredisable(sdiodev, ci->armcorebase);
Franky Lin5b45e542011-11-04 22:23:30 +0100275}
276
Franky Lina83369b2011-11-04 22:23:28 +0100277int brcmf_sdio_chip_attach(struct brcmf_sdio_dev *sdiodev,
Franky Lina97e4fc2011-11-04 22:23:35 +0100278 struct chip_info **ci_ptr, u32 regs)
Franky Lina83369b2011-11-04 22:23:28 +0100279{
Franky Lina97e4fc2011-11-04 22:23:35 +0100280 int ret;
281 struct chip_info *ci;
282
283 brcmf_dbg(TRACE, "Enter\n");
284
285 /* alloc chip_info_t */
286 ci = kzalloc(sizeof(struct chip_info), GFP_ATOMIC);
287 if (!ci)
288 return -ENOMEM;
Franky Lina83369b2011-11-04 22:23:28 +0100289
Franky Line63ac6b2011-11-04 22:23:29 +0100290 ret = brcmf_sdio_chip_buscoreprep(sdiodev);
291 if (ret != 0)
Franky Lina97e4fc2011-11-04 22:23:35 +0100292 goto err;
Franky Line63ac6b2011-11-04 22:23:29 +0100293
Franky Lina83369b2011-11-04 22:23:28 +0100294 ret = brcmf_sdio_chip_recognition(sdiodev, ci, regs);
295 if (ret != 0)
Franky Lina97e4fc2011-11-04 22:23:35 +0100296 goto err;
Franky Lina83369b2011-11-04 22:23:28 +0100297
Franky Lin5b45e542011-11-04 22:23:30 +0100298 brcmf_sdio_chip_buscoresetup(sdiodev, ci);
299
Franky Lin960908d2011-11-04 22:23:33 +0100300 brcmf_sdcard_reg_write(sdiodev,
301 CORE_CC_REG(ci->cccorebase, gpiopullup), 4, 0);
302 brcmf_sdcard_reg_write(sdiodev,
303 CORE_CC_REG(ci->cccorebase, gpiopulldown), 4, 0);
304
Franky Lina97e4fc2011-11-04 22:23:35 +0100305 *ci_ptr = ci;
306 return 0;
307
308err:
309 kfree(ci);
Franky Lina83369b2011-11-04 22:23:28 +0100310 return ret;
311}