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