xref: /freebsd/sys/arm/ti/ti_cpuid.c (revision fdafd315ad0d0f28a11b9fb4476a9ab059c62b92)
1e53470feSOleksandr Tymoshenko /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3af3dc4a7SPedro F. Giffuni  *
4e53470feSOleksandr Tymoshenko  * Copyright (c) 2011
5e53470feSOleksandr Tymoshenko  *	Ben Gray <ben.r.gray@gmail.com>.
6e53470feSOleksandr Tymoshenko  * All rights reserved.
7e53470feSOleksandr Tymoshenko  *
8e53470feSOleksandr Tymoshenko  * Redistribution and use in source and binary forms, with or without
9e53470feSOleksandr Tymoshenko  * modification, are permitted provided that the following conditions
10e53470feSOleksandr Tymoshenko  * are met:
11e53470feSOleksandr Tymoshenko  * 1. Redistributions of source code must retain the above copyright
12e53470feSOleksandr Tymoshenko  *    notice, this list of conditions and the following disclaimer.
13e53470feSOleksandr Tymoshenko  * 2. Redistributions in binary form must reproduce the above copyright
14e53470feSOleksandr Tymoshenko  *    notice, this list of conditions and the following disclaimer in the
15e53470feSOleksandr Tymoshenko  *    documentation and/or other materials provided with the distribution.
16e53470feSOleksandr Tymoshenko  *
17e53470feSOleksandr Tymoshenko  * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18e53470feSOleksandr Tymoshenko  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19e53470feSOleksandr Tymoshenko  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20e53470feSOleksandr Tymoshenko  * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21e53470feSOleksandr Tymoshenko  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22e53470feSOleksandr Tymoshenko  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23e53470feSOleksandr Tymoshenko  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24e53470feSOleksandr Tymoshenko  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25e53470feSOleksandr Tymoshenko  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26e53470feSOleksandr Tymoshenko  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27e53470feSOleksandr Tymoshenko  * SUCH DAMAGE.
28e53470feSOleksandr Tymoshenko  */
29e53470feSOleksandr Tymoshenko 
30e53470feSOleksandr Tymoshenko #include <sys/param.h>
31e53470feSOleksandr Tymoshenko #include <sys/systm.h>
32e53470feSOleksandr Tymoshenko #include <sys/kernel.h>
33e53470feSOleksandr Tymoshenko #include <sys/module.h>
34e53470feSOleksandr Tymoshenko #include <sys/bus.h>
35e53470feSOleksandr Tymoshenko #include <sys/resource.h>
36e53470feSOleksandr Tymoshenko #include <sys/rman.h>
37e53470feSOleksandr Tymoshenko #include <sys/lock.h>
38e53470feSOleksandr Tymoshenko #include <sys/mutex.h>
39e53470feSOleksandr Tymoshenko 
40e53470feSOleksandr Tymoshenko #include <machine/bus.h>
41cb5ce014SAndrew Turner #include <machine/cpu.h>
42e53470feSOleksandr Tymoshenko #include <machine/fdt.h>
43e53470feSOleksandr Tymoshenko #include <machine/resource.h>
44e53470feSOleksandr Tymoshenko #include <machine/intr.h>
45e53470feSOleksandr Tymoshenko 
46645f6eafSOlivier Houchard #include <dev/fdt/simplebus.h>
47645f6eafSOlivier Houchard #include <dev/fdt/fdt_common.h>
48645f6eafSOlivier Houchard #include <dev/ofw/ofw_bus_subr.h>
49645f6eafSOlivier Houchard 
50e53470feSOleksandr Tymoshenko #include <arm/ti/tivar.h>
51e53470feSOleksandr Tymoshenko #include <arm/ti/ti_cpuid.h>
52e53470feSOleksandr Tymoshenko 
53e53470feSOleksandr Tymoshenko #include <arm/ti/omap4/omap4_reg.h>
54e53470feSOleksandr Tymoshenko #include <arm/ti/am335x/am335x_reg.h>
55e53470feSOleksandr Tymoshenko 
56e53470feSOleksandr Tymoshenko #define OMAP4_STD_FUSE_DIE_ID_0    0x2200
57e53470feSOleksandr Tymoshenko #define OMAP4_ID_CODE              0x2204
58e53470feSOleksandr Tymoshenko #define OMAP4_STD_FUSE_DIE_ID_1    0x2208
59e53470feSOleksandr Tymoshenko #define OMAP4_STD_FUSE_DIE_ID_2    0x220C
60e53470feSOleksandr Tymoshenko #define OMAP4_STD_FUSE_DIE_ID_3    0x2210
61e53470feSOleksandr Tymoshenko #define OMAP4_STD_FUSE_PROD_ID_0   0x2214
62e53470feSOleksandr Tymoshenko #define OMAP4_STD_FUSE_PROD_ID_1   0x2218
63e53470feSOleksandr Tymoshenko 
64e53470feSOleksandr Tymoshenko #define OMAP3_ID_CODE              0xA204
65e53470feSOleksandr Tymoshenko 
66e53470feSOleksandr Tymoshenko static uint32_t chip_revision = 0xffffffff;
67e53470feSOleksandr Tymoshenko 
68e53470feSOleksandr Tymoshenko /**
69e53470feSOleksandr Tymoshenko  *	ti_revision - Returns the revision number of the device
70e53470feSOleksandr Tymoshenko  *
71e53470feSOleksandr Tymoshenko  *	Simply returns an identifier for the revision of the chip we are running
72e53470feSOleksandr Tymoshenko  *	on.
73e53470feSOleksandr Tymoshenko  *
74e53470feSOleksandr Tymoshenko  *	RETURNS
75e53470feSOleksandr Tymoshenko  *	A 32-bit identifier for the current chip
76e53470feSOleksandr Tymoshenko  */
77e53470feSOleksandr Tymoshenko uint32_t
ti_revision(void)78e53470feSOleksandr Tymoshenko ti_revision(void)
79e53470feSOleksandr Tymoshenko {
80e53470feSOleksandr Tymoshenko 	return chip_revision;
81e53470feSOleksandr Tymoshenko }
82e53470feSOleksandr Tymoshenko 
83e53470feSOleksandr Tymoshenko /**
84e53470feSOleksandr Tymoshenko  *	omap4_get_revision - determines omap4 revision
85e53470feSOleksandr Tymoshenko  *
86e53470feSOleksandr Tymoshenko  *	Reads the registers to determine the revision of the chip we are currently
87e53470feSOleksandr Tymoshenko  *	running on.  Stores the information in global variables.
88e53470feSOleksandr Tymoshenko  *
89e53470feSOleksandr Tymoshenko  *
90e53470feSOleksandr Tymoshenko  */
91e53470feSOleksandr Tymoshenko static void
omap4_get_revision(void)92e53470feSOleksandr Tymoshenko omap4_get_revision(void)
93e53470feSOleksandr Tymoshenko {
94e53470feSOleksandr Tymoshenko 	uint32_t id_code;
95e53470feSOleksandr Tymoshenko 	uint32_t revision;
96e53470feSOleksandr Tymoshenko 	uint32_t hawkeye;
97e53470feSOleksandr Tymoshenko 	bus_space_handle_t bsh;
98e53470feSOleksandr Tymoshenko 
99e53470feSOleksandr Tymoshenko 	/* The chip revsion is read from the device identification registers and
100e53470feSOleksandr Tymoshenko 	 * the JTAG (?) tap registers, which are located in address 0x4A00_2200 to
101e53470feSOleksandr Tymoshenko 	 * 0x4A00_2218.  This is part of the L4_CORE memory range and should have
102e53470feSOleksandr Tymoshenko 	 * been mapped in by the machdep.c code.
103e53470feSOleksandr Tymoshenko 	 *
104e53470feSOleksandr Tymoshenko 	 *   STD_FUSE_DIE_ID_0    0x4A00 2200
105e53470feSOleksandr Tymoshenko 	 *   ID_CODE              0x4A00 2204   (this is the only one we need)
106e53470feSOleksandr Tymoshenko 	 *   STD_FUSE_DIE_ID_1    0x4A00 2208
107e53470feSOleksandr Tymoshenko 	 *   STD_FUSE_DIE_ID_2    0x4A00 220C
108e53470feSOleksandr Tymoshenko 	 *   STD_FUSE_DIE_ID_3    0x4A00 2210
109e53470feSOleksandr Tymoshenko 	 *   STD_FUSE_PROD_ID_0   0x4A00 2214
110e53470feSOleksandr Tymoshenko 	 *   STD_FUSE_PROD_ID_1   0x4A00 2218
111e53470feSOleksandr Tymoshenko 	 */
112e5897405SOleksandr Tymoshenko 	/* FIXME Should we map somewhere else? */
113e53470feSOleksandr Tymoshenko 	bus_space_map(fdtbus_bs_tag,OMAP44XX_L4_CORE_HWBASE, 0x4000, 0, &bsh);
114e53470feSOleksandr Tymoshenko 	id_code = bus_space_read_4(fdtbus_bs_tag, bsh, OMAP4_ID_CODE);
115e53470feSOleksandr Tymoshenko 	bus_space_unmap(fdtbus_bs_tag, bsh, 0x4000);
116e53470feSOleksandr Tymoshenko 
117e53470feSOleksandr Tymoshenko 	hawkeye = ((id_code >> 12) & 0xffff);
118e53470feSOleksandr Tymoshenko 	revision = ((id_code >> 28) & 0xf);
119e53470feSOleksandr Tymoshenko 
120e53470feSOleksandr Tymoshenko 	/* Apparently according to the linux code there were some ES2.0 samples that
121e53470feSOleksandr Tymoshenko 	 * have the wrong id code and report themselves as ES1.0 silicon.  So used
122e53470feSOleksandr Tymoshenko 	 * the ARM cpuid to get the correct revision.
123e53470feSOleksandr Tymoshenko 	 */
124e53470feSOleksandr Tymoshenko 	if (revision == 0) {
125cb5ce014SAndrew Turner 		id_code = cp15_midr_get();
126e53470feSOleksandr Tymoshenko 		revision = (id_code & 0xf) - 1;
127e53470feSOleksandr Tymoshenko 	}
128e53470feSOleksandr Tymoshenko 
129e53470feSOleksandr Tymoshenko 	switch (hawkeye) {
130e53470feSOleksandr Tymoshenko 	case 0xB852:
1315c79e1ddSOleksandr Tymoshenko 		switch (revision) {
1325c79e1ddSOleksandr Tymoshenko 		case 0:
133e53470feSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_ES1_0;
134e53470feSOleksandr Tymoshenko 			break;
1355c79e1ddSOleksandr Tymoshenko 		case 1:
136e53470feSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_ES2_1;
1375c79e1ddSOleksandr Tymoshenko 			break;
1385c79e1ddSOleksandr Tymoshenko 		default:
1395c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_UNKNOWN;
1405c79e1ddSOleksandr Tymoshenko 			break;
1415c79e1ddSOleksandr Tymoshenko 		}
1425c79e1ddSOleksandr Tymoshenko 		break;
1435c79e1ddSOleksandr Tymoshenko 
1445c79e1ddSOleksandr Tymoshenko 	case 0xB95C:
1455c79e1ddSOleksandr Tymoshenko 		switch (revision) {
1465c79e1ddSOleksandr Tymoshenko 		case 3:
1475c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_ES2_1;
1485c79e1ddSOleksandr Tymoshenko 			break;
1495c79e1ddSOleksandr Tymoshenko 		case 4:
150e53470feSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_ES2_2;
1515c79e1ddSOleksandr Tymoshenko 			break;
1525c79e1ddSOleksandr Tymoshenko 		case 6:
153e53470feSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_ES2_3;
154e53470feSOleksandr Tymoshenko 			break;
155e53470feSOleksandr Tymoshenko 		default:
1565c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4430_REV_UNKNOWN;
157e53470feSOleksandr Tymoshenko 			break;
158e53470feSOleksandr Tymoshenko 		}
1595c79e1ddSOleksandr Tymoshenko 		break;
1605c79e1ddSOleksandr Tymoshenko 
1615c79e1ddSOleksandr Tymoshenko 	case 0xB94E:
1625c79e1ddSOleksandr Tymoshenko 		switch (revision) {
1635c79e1ddSOleksandr Tymoshenko 		case 0:
1645c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4460_REV_ES1_0;
1655c79e1ddSOleksandr Tymoshenko 			break;
1665c79e1ddSOleksandr Tymoshenko 		case 2:
1675c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4460_REV_ES1_1;
1685c79e1ddSOleksandr Tymoshenko 			break;
1695c79e1ddSOleksandr Tymoshenko 		default:
1705c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4460_REV_UNKNOWN;
1715c79e1ddSOleksandr Tymoshenko 			break;
1725c79e1ddSOleksandr Tymoshenko 		}
1735c79e1ddSOleksandr Tymoshenko 		break;
1745c79e1ddSOleksandr Tymoshenko 
1755c79e1ddSOleksandr Tymoshenko 	case 0xB975:
1765c79e1ddSOleksandr Tymoshenko 		switch (revision) {
1775c79e1ddSOleksandr Tymoshenko 		case 0:
1785c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4470_REV_ES1_0;
1795c79e1ddSOleksandr Tymoshenko 			break;
1805c79e1ddSOleksandr Tymoshenko 		default:
1815c79e1ddSOleksandr Tymoshenko 			chip_revision = OMAP4470_REV_UNKNOWN;
1825c79e1ddSOleksandr Tymoshenko 			break;
1835c79e1ddSOleksandr Tymoshenko 		}
1845c79e1ddSOleksandr Tymoshenko 		break;
1855c79e1ddSOleksandr Tymoshenko 
1865c79e1ddSOleksandr Tymoshenko 	default:
1875c79e1ddSOleksandr Tymoshenko 		/* Default to the latest revision if we can't determine type */
1885c79e1ddSOleksandr Tymoshenko 		chip_revision = OMAP_UNKNOWN_DEV;
1895c79e1ddSOleksandr Tymoshenko 		break;
1905c79e1ddSOleksandr Tymoshenko 	}
1915c79e1ddSOleksandr Tymoshenko 	if (chip_revision != OMAP_UNKNOWN_DEV) {
192e53470feSOleksandr Tymoshenko 		printf("Texas Instruments OMAP%04x Processor, Revision ES%u.%u\n",
193e53470feSOleksandr Tymoshenko 		    OMAP_REV_DEVICE(chip_revision), OMAP_REV_MAJOR(chip_revision),
194e53470feSOleksandr Tymoshenko 		    OMAP_REV_MINOR(chip_revision));
195e53470feSOleksandr Tymoshenko 	}
1965c79e1ddSOleksandr Tymoshenko 	else {
1975c79e1ddSOleksandr Tymoshenko 		printf("Texas Instruments unknown OMAP chip: %04x, rev %d\n",
1985c79e1ddSOleksandr Tymoshenko 		    hawkeye, revision);
1995c79e1ddSOleksandr Tymoshenko 	}
2005c79e1ddSOleksandr Tymoshenko }
201e53470feSOleksandr Tymoshenko 
202e53470feSOleksandr Tymoshenko static void
am335x_get_revision(void)203e53470feSOleksandr Tymoshenko am335x_get_revision(void)
204e53470feSOleksandr Tymoshenko {
205e53470feSOleksandr Tymoshenko 	uint32_t dev_feature;
2061648ac50SWarner Losh 	char cpu_last_char;
207e53470feSOleksandr Tymoshenko 	bus_space_handle_t bsh;
2081648ac50SWarner Losh 	int major;
2091648ac50SWarner Losh 	int minor;
210e53470feSOleksandr Tymoshenko 
211e53470feSOleksandr Tymoshenko 	bus_space_map(fdtbus_bs_tag, AM335X_CONTROL_BASE, AM335X_CONTROL_SIZE, 0, &bsh);
212e53470feSOleksandr Tymoshenko 	chip_revision = bus_space_read_4(fdtbus_bs_tag, bsh, AM335X_CONTROL_DEVICE_ID);
213e53470feSOleksandr Tymoshenko 	dev_feature = bus_space_read_4(fdtbus_bs_tag, bsh, AM335X_CONTROL_DEV_FEATURE);
214e53470feSOleksandr Tymoshenko 	bus_space_unmap(fdtbus_bs_tag, bsh, AM335X_CONTROL_SIZE);
215e53470feSOleksandr Tymoshenko 
216e53470feSOleksandr Tymoshenko 	switch (dev_feature) {
217e53470feSOleksandr Tymoshenko 		case 0x00FF0382:
218e53470feSOleksandr Tymoshenko 			cpu_last_char='2';
219e53470feSOleksandr Tymoshenko 			break;
220e53470feSOleksandr Tymoshenko 		case 0x20FF0382:
221e53470feSOleksandr Tymoshenko 			cpu_last_char='4';
222e53470feSOleksandr Tymoshenko 			break;
223e53470feSOleksandr Tymoshenko 		case 0x00FF0383:
224e53470feSOleksandr Tymoshenko 			cpu_last_char='6';
225e53470feSOleksandr Tymoshenko 			break;
226e53470feSOleksandr Tymoshenko 		case 0x00FE0383:
227e53470feSOleksandr Tymoshenko 			cpu_last_char='7';
228e53470feSOleksandr Tymoshenko 			break;
229e53470feSOleksandr Tymoshenko 		case 0x20FF0383:
230e53470feSOleksandr Tymoshenko 			cpu_last_char='8';
231e53470feSOleksandr Tymoshenko 			break;
232e53470feSOleksandr Tymoshenko 		case 0x20FE0383:
233e53470feSOleksandr Tymoshenko 			cpu_last_char='9';
234e53470feSOleksandr Tymoshenko 			break;
235e53470feSOleksandr Tymoshenko 		default:
236e53470feSOleksandr Tymoshenko 			cpu_last_char='x';
237e53470feSOleksandr Tymoshenko 	}
238e53470feSOleksandr Tymoshenko 
2391648ac50SWarner Losh 	switch(AM335X_DEVREV(chip_revision)) {
2401648ac50SWarner Losh 		case 0:
2411648ac50SWarner Losh 			major = 1;
2421648ac50SWarner Losh 			minor = 0;
2431648ac50SWarner Losh 			break;
2441648ac50SWarner Losh 		case 1:
2451648ac50SWarner Losh 			major = 2;
2461648ac50SWarner Losh 			minor = 0;
2471648ac50SWarner Losh 			break;
2481648ac50SWarner Losh 		case 2:
2491648ac50SWarner Losh 			major = 2;
2501648ac50SWarner Losh 			minor = 1;
2511648ac50SWarner Losh 			break;
2521648ac50SWarner Losh 		default:
2531648ac50SWarner Losh 			major = 0;
2541648ac50SWarner Losh 			minor = AM335X_DEVREV(chip_revision);
2551648ac50SWarner Losh 			break;
2561648ac50SWarner Losh 	}
2571648ac50SWarner Losh 	printf("Texas Instruments AM335%c Processor, Revision ES%u.%u\n",
2581648ac50SWarner Losh 		cpu_last_char, major, minor);
259e53470feSOleksandr Tymoshenko }
260e53470feSOleksandr Tymoshenko 
261e53470feSOleksandr Tymoshenko /**
262e53470feSOleksandr Tymoshenko  *	ti_cpu_ident - attempts to identify the chip we are running on
263e53470feSOleksandr Tymoshenko  *	@dummy: ignored
264e53470feSOleksandr Tymoshenko  *
265e53470feSOleksandr Tymoshenko  *	This function is called before any of the driver are initialised, however
266e53470feSOleksandr Tymoshenko  *	the basic virt to phys maps have been setup in machdep.c so we can still
267e53470feSOleksandr Tymoshenko  *	access the required registers, we just have to use direct register reads
268e53470feSOleksandr Tymoshenko  *	and writes rather than going through the bus stuff.
269e53470feSOleksandr Tymoshenko  *
270e53470feSOleksandr Tymoshenko  *
271e53470feSOleksandr Tymoshenko  */
272e53470feSOleksandr Tymoshenko static void
ti_cpu_ident(void * dummy)273e53470feSOleksandr Tymoshenko ti_cpu_ident(void *dummy)
274e53470feSOleksandr Tymoshenko {
27569d14913SOlivier Houchard 	if (!ti_soc_is_supported())
276645f6eafSOlivier Houchard 		return;
277e53470feSOleksandr Tymoshenko 	switch(ti_chip()) {
278e53470feSOleksandr Tymoshenko 	case CHIP_OMAP_4:
279e53470feSOleksandr Tymoshenko 		omap4_get_revision();
280e53470feSOleksandr Tymoshenko 		break;
281e53470feSOleksandr Tymoshenko 	case CHIP_AM335X:
282e53470feSOleksandr Tymoshenko 		am335x_get_revision();
283e53470feSOleksandr Tymoshenko 		break;
284e53470feSOleksandr Tymoshenko 	default:
285e53470feSOleksandr Tymoshenko 		panic("Unknown chip type, fixme!\n");
286e53470feSOleksandr Tymoshenko 	}
287e53470feSOleksandr Tymoshenko }
288e53470feSOleksandr Tymoshenko 
289e53470feSOleksandr Tymoshenko SYSINIT(ti_cpu_ident, SI_SUB_CPU, SI_ORDER_SECOND, ti_cpu_ident, NULL);
290