xref: /freebsd/sys/dev/bhnd/cores/chipc/chipc.c (revision cdf63a700c77204252e3c2e38d7106965559f3c6)
1 /*-
2  * Copyright (c) 2015 Landon Fuller <landon@landonf.org>
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer,
10  *    without modification.
11  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
12  *    similar to the "NO WARRANTY" disclaimer below ("Disclaimer") and any
13  *    redistribution must be conditioned upon including a substantially
14  *    similar Disclaimer requirement for further binary redistribution.
15  *
16  * NO WARRANTY
17  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19  * LIMITED TO, THE IMPLIED WARRANTIES OF NONINFRINGEMENT, MERCHANTIBILITY
20  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
21  * THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY,
22  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25  * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
27  * THE POSSIBILITY OF SUCH DAMAGES.
28  */
29 
30 #include <sys/cdefs.h>
31 __FBSDID("$FreeBSD$");
32 
33 /*
34  * Broadcom ChipCommon driver.
35  *
36  * With the exception of some very early chipsets, the ChipCommon core
37  * has been included in all HND SoCs and chipsets based on the siba(4)
38  * and bcma(4) interconnects, providing a common interface to chipset
39  * identification, bus enumeration, UARTs, clocks, watchdog interrupts, GPIO,
40  * flash, etc.
41  */
42 
43 #include <sys/param.h>
44 #include <sys/kernel.h>
45 #include <sys/bus.h>
46 #include <sys/module.h>
47 #include <sys/systm.h>
48 
49 #include <machine/bus.h>
50 #include <sys/rman.h>
51 #include <machine/resource.h>
52 
53 #include <dev/bhnd/bhnd.h>
54 
55 #include "bhnd_nvram_if.h"
56 
57 #include "chipcreg.h"
58 #include "chipcvar.h"
59 
60 devclass_t bhnd_chipc_devclass;	/**< bhnd(4) chipcommon device class */
61 
62 static const struct resource_spec chipc_rspec[CHIPC_MAX_RSPEC] = {
63 	{ SYS_RES_MEMORY,	0,	RF_ACTIVE },
64 	{ -1, -1, 0 }
65 };
66 
67 static struct bhnd_device_quirk chipc_quirks[];
68 
69 /* Supported device identifiers */
70 static const struct bhnd_device chipc_devices[] = {
71 	BHND_DEVICE(CC, "CC", chipc_quirks),
72 	BHND_DEVICE_END
73 };
74 
75 
76 /* Device quirks table */
77 static struct bhnd_device_quirk chipc_quirks[] = {
78 	{ BHND_HWREV_GTE	(32),	CHIPC_QUIRK_SUPPORTS_SPROM },
79 	{ BHND_HWREV_GTE	(35),	CHIPC_QUIRK_SUPPORTS_NFLASH },
80 	BHND_DEVICE_QUIRK_END
81 };
82 
83 /* Chip-specific quirks table */
84 static struct bhnd_chip_quirk chipc_chip_quirks[] = {
85 	/* 4331 12x9 packages */
86 	{{ BHND_CHIP_IP(4331, 4331TN) },
87 		CHIPC_QUIRK_4331_GPIO2_5_MUX_SPROM
88 	},
89 	{{ BHND_CHIP_IP(4331, 4331TNA0) },
90 		CHIPC_QUIRK_4331_GPIO2_5_MUX_SPROM
91 	},
92 
93 	/* 4331 12x12 packages */
94 	{{ BHND_CHIP_IPR(4331, 4331TT, HWREV_GTE(1)) },
95 		CHIPC_QUIRK_4331_EXTPA2_MUX_SPROM
96 	},
97 
98 	/* 4331 (all packages/revisions) */
99 	{{ BHND_CHIP_ID(4331) },
100 		CHIPC_QUIRK_4331_EXTPA_MUX_SPROM
101 	},
102 
103 	/* 4360 family (all revs <= 2) */
104 	{{ BHND_CHIP_IR(4352, HWREV_LTE(2)) },
105 		CHIPC_QUIRK_4360_FEM_MUX_SPROM },
106 	{{ BHND_CHIP_IR(43460, HWREV_LTE(2)) },
107 		CHIPC_QUIRK_4360_FEM_MUX_SPROM },
108 	{{ BHND_CHIP_IR(43462, HWREV_LTE(2)) },
109 		CHIPC_QUIRK_4360_FEM_MUX_SPROM },
110 	{{ BHND_CHIP_IR(43602, HWREV_LTE(2)) },
111 		CHIPC_QUIRK_4360_FEM_MUX_SPROM },
112 
113 	BHND_CHIP_QUIRK_END
114 };
115 
116 /* quirk and capability flag convenience macros */
117 #define	CHIPC_QUIRK(_sc, _name)	\
118     ((_sc)->quirks & CHIPC_QUIRK_ ## _name)
119 
120 #define CHIPC_CAP(_sc, _name)	\
121     ((_sc)->caps & CHIPC_ ## _name)
122 
123 #define	CHIPC_ASSERT_QUIRK(_sc, name)	\
124     KASSERT(CHIPC_QUIRK((_sc), name), ("quirk " __STRING(_name) " not set"))
125 
126 #define	CHIPC_ASSERT_CAP(_sc, name)	\
127     KASSERT(CHIPC_CAP((_sc), name), ("capability " __STRING(_name) " not set"))
128 
129 static bhnd_nvram_src_t	chipc_nvram_identify(struct chipc_softc *sc);
130 static int		chipc_sprom_init(struct chipc_softc *);
131 static int		chipc_enable_sprom_pins(struct chipc_softc *);
132 static int		chipc_disable_sprom_pins(struct chipc_softc *);
133 
134 
135 static int
136 chipc_probe(device_t dev)
137 {
138 	const struct bhnd_device *id;
139 
140 	id = bhnd_device_lookup(dev, chipc_devices, sizeof(chipc_devices[0]));
141 	if (id == NULL)
142 		return (ENXIO);
143 
144 	bhnd_set_default_core_desc(dev);
145 	return (BUS_PROBE_DEFAULT);
146 }
147 
148 static int
149 chipc_attach(device_t dev)
150 {
151 	struct chipc_softc		*sc;
152 	bhnd_addr_t			 enum_addr;
153 	uint32_t			 ccid_reg;
154 	uint8_t				 chip_type;
155 	int				 error;
156 
157 	sc = device_get_softc(dev);
158 	sc->dev = dev;
159 	sc->quirks = bhnd_device_quirks(dev, chipc_devices,
160 	    sizeof(chipc_devices[0]));
161 	sc->quirks |= bhnd_chip_quirks(dev, chipc_chip_quirks);
162 
163 	CHIPC_LOCK_INIT(sc);
164 
165 	/* Allocate bus resources */
166 	memcpy(sc->rspec, chipc_rspec, sizeof(sc->rspec));
167 	if ((error = bhnd_alloc_resources(dev, sc->rspec, sc->res)))
168 		return (error);
169 
170 	sc->core = sc->res[0];
171 
172 	/* Fetch our chipset identification data */
173 	ccid_reg = bhnd_bus_read_4(sc->core, CHIPC_ID);
174 	chip_type = CHIPC_GET_ATTR(ccid_reg, ID_BUS);
175 
176 	switch (chip_type) {
177 	case BHND_CHIPTYPE_SIBA:
178 		/* enumeration space starts at the ChipCommon register base. */
179 		enum_addr = rman_get_start(sc->core->res);
180 		break;
181 	case BHND_CHIPTYPE_BCMA:
182 	case BHND_CHIPTYPE_BCMA_ALT:
183 		enum_addr = bhnd_bus_read_4(sc->core, CHIPC_EROMPTR);
184 		break;
185 	default:
186 		device_printf(dev, "unsupported chip type %hhu\n", chip_type);
187 		error = ENODEV;
188 		goto cleanup;
189 	}
190 
191 	sc->ccid = bhnd_parse_chipid(ccid_reg, enum_addr);
192 
193 	/* Fetch capability and status register values */
194 	sc->caps = bhnd_bus_read_4(sc->core, CHIPC_CAPABILITIES);
195 	sc->cst = bhnd_bus_read_4(sc->core, CHIPC_CHIPST);
196 
197 	/* Identify NVRAM source */
198 	sc->nvram_src = chipc_nvram_identify(sc);
199 
200 	/* Read NVRAM data */
201 	switch (sc->nvram_src) {
202 	case BHND_NVRAM_SRC_OTP:
203 		// TODO (requires access to OTP hardware)
204 		device_printf(sc->dev, "NVRAM-OTP unsupported\n");
205 		break;
206 
207 	case BHND_NVRAM_SRC_NFLASH:
208 		// TODO (requires access to NFLASH hardware)
209 		device_printf(sc->dev, "NVRAM-NFLASH unsupported\n");
210 		break;
211 
212 	case BHND_NVRAM_SRC_SPROM:
213 		if ((error = chipc_sprom_init(sc)))
214 			goto cleanup;
215 		break;
216 
217 	case BHND_NVRAM_SRC_UNKNOWN:
218 		/* Handled externally */
219 		break;
220 	}
221 
222 	return (0);
223 
224 cleanup:
225 	bhnd_release_resources(dev, sc->rspec, sc->res);
226 	CHIPC_LOCK_DESTROY(sc);
227 	return (error);
228 }
229 
230 static int
231 chipc_detach(device_t dev)
232 {
233 	struct chipc_softc	*sc;
234 
235 	sc = device_get_softc(dev);
236 	bhnd_release_resources(dev, sc->rspec, sc->res);
237 	bhnd_sprom_fini(&sc->sprom);
238 
239 	CHIPC_LOCK_DESTROY(sc);
240 
241 	return (0);
242 }
243 
244 static int
245 chipc_suspend(device_t dev)
246 {
247 	return (0);
248 }
249 
250 static int
251 chipc_resume(device_t dev)
252 {
253 	return (0);
254 }
255 
256 /**
257  * Initialize local SPROM shadow, if required.
258  *
259  * @param sc chipc driver state.
260  */
261 static int
262 chipc_sprom_init(struct chipc_softc *sc)
263 {
264 	int	error;
265 
266 	KASSERT(sc->nvram_src == BHND_NVRAM_SRC_SPROM,
267 	    ("non-SPROM source (%u)\n", sc->nvram_src));
268 
269 	/* Enable access to the SPROM */
270 	CHIPC_LOCK(sc);
271 	if ((error = chipc_enable_sprom_pins(sc)))
272 		goto failed;
273 
274 	/* Initialize SPROM parser */
275 	error = bhnd_sprom_init(&sc->sprom, sc->core, CHIPC_SPROM_OTP);
276 	if (error) {
277 		device_printf(sc->dev, "SPROM identification failed: %d\n",
278 			error);
279 
280 		chipc_disable_sprom_pins(sc);
281 		goto failed;
282 	}
283 
284 	/* Drop access to the SPROM lines */
285 	if ((error = chipc_disable_sprom_pins(sc))) {
286 		bhnd_sprom_fini(&sc->sprom);
287 		goto failed;
288 	}
289 	CHIPC_UNLOCK(sc);
290 
291 	return (0);
292 
293 failed:
294 	CHIPC_UNLOCK(sc);
295 	return (error);
296 }
297 
298 /**
299  * Determine the NVRAM data source for this device.
300  *
301  * @param sc chipc driver state.
302  */
303 static bhnd_nvram_src_t
304 chipc_nvram_identify(struct chipc_softc *sc)
305 {
306 	uint32_t		 srom_ctrl;
307 
308 	/* Very early devices vend SPROM/OTP/CIS (if at all) via the
309 	 * host bridge interface instead of ChipCommon. */
310 	if (!CHIPC_QUIRK(sc, SUPPORTS_SPROM))
311 		return (BHND_NVRAM_SRC_UNKNOWN);
312 
313 	/*
314 	 * Later chipset revisions standardized the SPROM capability flags and
315 	 * register interfaces.
316 	 *
317 	 * We check for hardware presence in order of precedence. For example,
318 	 * SPROM is is always used in preference to internal OTP if found.
319 	 */
320 	if (CHIPC_CAP(sc, CAP_SPROM)) {
321 		srom_ctrl = bhnd_bus_read_4(sc->core, CHIPC_SPROM_CTRL);
322 		if (srom_ctrl & CHIPC_SRC_PRESENT)
323 			return (BHND_NVRAM_SRC_SPROM);
324 	}
325 
326 	/* Check for OTP */
327 	if (CHIPC_CAP(sc, CAP_OTP_SIZE))
328 		return (BHND_NVRAM_SRC_OTP);
329 
330 	/*
331 	 * Finally, Northstar chipsets (and possibly other chipsets?) support
332 	 * external NAND flash.
333 	 */
334 	if (CHIPC_QUIRK(sc, SUPPORTS_NFLASH) && CHIPC_CAP(sc, CAP_NFLASH))
335 		return (BHND_NVRAM_SRC_NFLASH);
336 
337 	/* No NVRAM hardware capability declared */
338 	return (BHND_NVRAM_SRC_UNKNOWN);
339 }
340 
341 
342 /**
343  * If required by this device, enable access to the SPROM.
344  *
345  * @param sc chipc driver state.
346  */
347 static int
348 chipc_enable_sprom_pins(struct chipc_softc *sc)
349 {
350 	uint32_t cctrl;
351 
352 	CHIPC_LOCK_ASSERT(sc, MA_OWNED);
353 
354 	/* Nothing to do? */
355 	if (!CHIPC_QUIRK(sc, MUX_SPROM))
356 		return (0);
357 
358 	cctrl = bhnd_bus_read_4(sc->core, CHIPC_CHIPCTRL);
359 
360 	/* 4331 devices */
361 	if (CHIPC_QUIRK(sc, 4331_EXTPA_MUX_SPROM)) {
362 		cctrl &= ~CHIPC_CCTRL4331_EXTPA_EN;
363 
364 		if (CHIPC_QUIRK(sc, 4331_GPIO2_5_MUX_SPROM))
365 			cctrl &= ~CHIPC_CCTRL4331_EXTPA_ON_GPIO2_5;
366 
367 		if (CHIPC_QUIRK(sc, 4331_EXTPA2_MUX_SPROM))
368 			cctrl &= ~CHIPC_CCTRL4331_EXTPA_EN2;
369 
370 		bhnd_bus_write_4(sc->core, CHIPC_CHIPCTRL, cctrl);
371 		return (0);
372 	}
373 
374 	/* 4360 devices */
375 	if (CHIPC_QUIRK(sc, 4360_FEM_MUX_SPROM)) {
376 		/* Unimplemented */
377 	}
378 
379 	/* Refuse to proceed on unsupported devices with muxed SPROM pins */
380 	device_printf(sc->dev, "muxed sprom lines on unrecognized device\n");
381 	return (ENXIO);
382 }
383 
384 /**
385  * If required by this device, revert any GPIO/pin configuration applied
386  * to allow SPROM access.
387  *
388  * @param sc chipc driver state.
389  */
390 static int
391 chipc_disable_sprom_pins(struct chipc_softc *sc)
392 {
393 	uint32_t cctrl;
394 
395 	CHIPC_LOCK_ASSERT(sc, MA_OWNED);
396 
397 	/* Nothing to do? */
398 	if (!CHIPC_QUIRK(sc, MUX_SPROM))
399 		return (0);
400 
401 	cctrl = bhnd_bus_read_4(sc->core, CHIPC_CHIPCTRL);
402 
403 	/* 4331 devices */
404 	if (CHIPC_QUIRK(sc, 4331_EXTPA_MUX_SPROM)) {
405 		cctrl |= CHIPC_CCTRL4331_EXTPA_EN;
406 
407 		if (CHIPC_QUIRK(sc, 4331_GPIO2_5_MUX_SPROM))
408 			cctrl |= CHIPC_CCTRL4331_EXTPA_ON_GPIO2_5;
409 
410 		if (CHIPC_QUIRK(sc, 4331_EXTPA2_MUX_SPROM))
411 			cctrl |= CHIPC_CCTRL4331_EXTPA_EN2;
412 
413 		bhnd_bus_write_4(sc->core, CHIPC_CHIPCTRL, cctrl);
414 		return (0);
415 	}
416 
417 	/* 4360 devices */
418 	if (CHIPC_QUIRK(sc, 4360_FEM_MUX_SPROM)) {
419 		/* Unimplemented */
420 	}
421 
422 	/* Refuse to proceed on unsupported devices with muxed SPROM pins */
423 	device_printf(sc->dev, "muxed sprom lines on unrecognized device\n");
424 	return (ENXIO);
425 }
426 
427 static bhnd_nvram_src_t
428 chipc_nvram_src(device_t dev)
429 {
430 	struct chipc_softc *sc = device_get_softc(dev);
431 	return (sc->nvram_src);
432 }
433 
434 static int
435 chipc_nvram_getvar(device_t dev, const char *name, void *buf, size_t *len)
436 {
437 	struct chipc_softc	*sc;
438 	int			 error;
439 
440 	sc = device_get_softc(dev);
441 
442 	switch (sc->nvram_src) {
443 	case BHND_NVRAM_SRC_SPROM:
444 		CHIPC_LOCK(sc);
445 		error = bhnd_sprom_getvar(&sc->sprom, name, buf, len);
446 		CHIPC_UNLOCK(sc);
447 		return (error);
448 
449 	case BHND_NVRAM_SRC_OTP:
450 	case BHND_NVRAM_SRC_NFLASH:
451 		/* Currently unsupported */
452 		return (ENXIO);
453 
454 	case BHND_NVRAM_SRC_UNKNOWN:
455 		return (ENODEV);
456 	}
457 
458 	/* Unknown NVRAM source */
459 	return (ENODEV);
460 }
461 
462 static int
463 chipc_nvram_setvar(device_t dev, const char *name, const void *buf,
464     size_t len)
465 {
466 	struct chipc_softc	*sc;
467 	int			 error;
468 
469 	sc = device_get_softc(dev);
470 
471 	switch (sc->nvram_src) {
472 	case BHND_NVRAM_SRC_SPROM:
473 		CHIPC_LOCK(sc);
474 		error = bhnd_sprom_setvar(&sc->sprom, name, buf, len);
475 		CHIPC_UNLOCK(sc);
476 		return (error);
477 
478 	case BHND_NVRAM_SRC_OTP:
479 	case BHND_NVRAM_SRC_NFLASH:
480 		/* Currently unsupported */
481 		return (ENXIO);
482 
483 	case BHND_NVRAM_SRC_UNKNOWN:
484 	default:
485 		return (ENODEV);
486 	}
487 
488 	/* Unknown NVRAM source */
489 	return (ENODEV);
490 }
491 
492 static device_method_t chipc_methods[] = {
493 	/* Device interface */
494 	DEVMETHOD(device_probe,		chipc_probe),
495 	DEVMETHOD(device_attach,	chipc_attach),
496 	DEVMETHOD(device_detach,	chipc_detach),
497 	DEVMETHOD(device_suspend,	chipc_suspend),
498 	DEVMETHOD(device_resume,	chipc_resume),
499 
500 	/* ChipCommon interface */
501 	DEVMETHOD(bhnd_chipc_nvram_src,	chipc_nvram_src),
502 
503 	/* NVRAM interface */
504 	DEVMETHOD(bhnd_nvram_getvar,	chipc_nvram_getvar),
505 	DEVMETHOD(bhnd_nvram_setvar,	chipc_nvram_setvar),
506 
507 	DEVMETHOD_END
508 };
509 
510 DEFINE_CLASS_0(bhnd_chipc, chipc_driver, chipc_methods, sizeof(struct chipc_softc));
511 DRIVER_MODULE(bhnd_chipc, bhnd, chipc_driver, bhnd_chipc_devclass, 0, 0);
512 MODULE_DEPEND(bhnd_chipc, bhnd, 1, 1, 1);
513 MODULE_VERSION(bhnd_chipc, 1);
514