xref: /OK3568_Linux_fs/kernel/arch/arm/mach-omap2/id.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * linux/arch/arm/mach-omap2/id.c
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * OMAP2 CPU identification code
6*4882a593Smuzhiyun  *
7*4882a593Smuzhiyun  * Copyright (C) 2005 Nokia Corporation
8*4882a593Smuzhiyun  * Written by Tony Lindgren <tony@atomide.com>
9*4882a593Smuzhiyun  *
10*4882a593Smuzhiyun  * Copyright (C) 2009-11 Texas Instruments
11*4882a593Smuzhiyun  * Added OMAP4 support - Santosh Shilimkar <santosh.shilimkar@ti.com>
12*4882a593Smuzhiyun  */
13*4882a593Smuzhiyun 
14*4882a593Smuzhiyun #include <linux/module.h>
15*4882a593Smuzhiyun #include <linux/kernel.h>
16*4882a593Smuzhiyun #include <linux/init.h>
17*4882a593Smuzhiyun #include <linux/io.h>
18*4882a593Smuzhiyun #include <linux/random.h>
19*4882a593Smuzhiyun #include <linux/slab.h>
20*4882a593Smuzhiyun 
21*4882a593Smuzhiyun #ifdef CONFIG_SOC_BUS
22*4882a593Smuzhiyun #include <linux/sys_soc.h>
23*4882a593Smuzhiyun #endif
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun #include <asm/cputype.h>
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun #include "common.h"
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun #include "id.h"
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun #include "soc.h"
32*4882a593Smuzhiyun #include "control.h"
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #define OMAP4_SILICON_TYPE_STANDARD		0x01
35*4882a593Smuzhiyun #define OMAP4_SILICON_TYPE_PERFORMANCE		0x02
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun #define OMAP_SOC_MAX_NAME_LENGTH		16
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun static unsigned int omap_revision;
40*4882a593Smuzhiyun static char soc_name[OMAP_SOC_MAX_NAME_LENGTH];
41*4882a593Smuzhiyun static char soc_rev[OMAP_SOC_MAX_NAME_LENGTH];
42*4882a593Smuzhiyun u32 omap_features;
43*4882a593Smuzhiyun 
omap_rev(void)44*4882a593Smuzhiyun unsigned int omap_rev(void)
45*4882a593Smuzhiyun {
46*4882a593Smuzhiyun 	return omap_revision;
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun EXPORT_SYMBOL(omap_rev);
49*4882a593Smuzhiyun 
omap_type(void)50*4882a593Smuzhiyun int omap_type(void)
51*4882a593Smuzhiyun {
52*4882a593Smuzhiyun 	static u32 val = OMAP2_DEVICETYPE_MASK;
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun 	if (val < OMAP2_DEVICETYPE_MASK)
55*4882a593Smuzhiyun 		return val;
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun 	if (soc_is_omap24xx()) {
58*4882a593Smuzhiyun 		val = omap_ctrl_readl(OMAP24XX_CONTROL_STATUS);
59*4882a593Smuzhiyun 	} else if (soc_is_ti81xx()) {
60*4882a593Smuzhiyun 		val = omap_ctrl_readl(TI81XX_CONTROL_STATUS);
61*4882a593Smuzhiyun 	} else if (soc_is_am33xx() || soc_is_am43xx()) {
62*4882a593Smuzhiyun 		val = omap_ctrl_readl(AM33XX_CONTROL_STATUS);
63*4882a593Smuzhiyun 	} else if (soc_is_omap34xx()) {
64*4882a593Smuzhiyun 		val = omap_ctrl_readl(OMAP343X_CONTROL_STATUS);
65*4882a593Smuzhiyun 	} else if (soc_is_omap44xx()) {
66*4882a593Smuzhiyun 		val = omap_ctrl_readl(OMAP4_CTRL_MODULE_CORE_STATUS);
67*4882a593Smuzhiyun 	} else if (soc_is_omap54xx() || soc_is_dra7xx()) {
68*4882a593Smuzhiyun 		val = omap_ctrl_readl(OMAP5XXX_CONTROL_STATUS);
69*4882a593Smuzhiyun 		val &= OMAP5_DEVICETYPE_MASK;
70*4882a593Smuzhiyun 		val >>= 6;
71*4882a593Smuzhiyun 		goto out;
72*4882a593Smuzhiyun 	} else {
73*4882a593Smuzhiyun 		pr_err("Cannot detect omap type!\n");
74*4882a593Smuzhiyun 		goto out;
75*4882a593Smuzhiyun 	}
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 	val &= OMAP2_DEVICETYPE_MASK;
78*4882a593Smuzhiyun 	val >>= 8;
79*4882a593Smuzhiyun 
80*4882a593Smuzhiyun out:
81*4882a593Smuzhiyun 	return val;
82*4882a593Smuzhiyun }
83*4882a593Smuzhiyun EXPORT_SYMBOL(omap_type);
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun 
86*4882a593Smuzhiyun /*----------------------------------------------------------------------------*/
87*4882a593Smuzhiyun 
88*4882a593Smuzhiyun #define OMAP_TAP_IDCODE		0x0204
89*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_0	0x0218
90*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_1	0x021C
91*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_2	0x0220
92*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_3	0x0224
93*4882a593Smuzhiyun 
94*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_44XX_0	0x0200
95*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_44XX_1	0x0208
96*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_44XX_2	0x020c
97*4882a593Smuzhiyun #define OMAP_TAP_DIE_ID_44XX_3	0x0210
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun #define read_tap_reg(reg)	readl_relaxed(tap_base  + (reg))
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun struct omap_id {
102*4882a593Smuzhiyun 	u16	hawkeye;	/* Silicon type (Hawkeye id) */
103*4882a593Smuzhiyun 	u8	dev;		/* Device type from production_id reg */
104*4882a593Smuzhiyun 	u32	type;		/* Combined type id copied to omap_revision */
105*4882a593Smuzhiyun };
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun /* Register values to detect the OMAP version */
108*4882a593Smuzhiyun static struct omap_id omap_ids[] __initdata = {
109*4882a593Smuzhiyun 	{ .hawkeye = 0xb5d9, .dev = 0x0, .type = 0x24200024 },
110*4882a593Smuzhiyun 	{ .hawkeye = 0xb5d9, .dev = 0x1, .type = 0x24201024 },
111*4882a593Smuzhiyun 	{ .hawkeye = 0xb5d9, .dev = 0x2, .type = 0x24202024 },
112*4882a593Smuzhiyun 	{ .hawkeye = 0xb5d9, .dev = 0x4, .type = 0x24220024 },
113*4882a593Smuzhiyun 	{ .hawkeye = 0xb5d9, .dev = 0x8, .type = 0x24230024 },
114*4882a593Smuzhiyun 	{ .hawkeye = 0xb68a, .dev = 0x0, .type = 0x24300024 },
115*4882a593Smuzhiyun };
116*4882a593Smuzhiyun 
117*4882a593Smuzhiyun static void __iomem *tap_base;
118*4882a593Smuzhiyun static u16 tap_prod_id;
119*4882a593Smuzhiyun 
omap_get_die_id(struct omap_die_id * odi)120*4882a593Smuzhiyun void omap_get_die_id(struct omap_die_id *odi)
121*4882a593Smuzhiyun {
122*4882a593Smuzhiyun 	if (soc_is_omap44xx() || soc_is_omap54xx() || soc_is_dra7xx()) {
123*4882a593Smuzhiyun 		odi->id_0 = read_tap_reg(OMAP_TAP_DIE_ID_44XX_0);
124*4882a593Smuzhiyun 		odi->id_1 = read_tap_reg(OMAP_TAP_DIE_ID_44XX_1);
125*4882a593Smuzhiyun 		odi->id_2 = read_tap_reg(OMAP_TAP_DIE_ID_44XX_2);
126*4882a593Smuzhiyun 		odi->id_3 = read_tap_reg(OMAP_TAP_DIE_ID_44XX_3);
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun 		return;
129*4882a593Smuzhiyun 	}
130*4882a593Smuzhiyun 	odi->id_0 = read_tap_reg(OMAP_TAP_DIE_ID_0);
131*4882a593Smuzhiyun 	odi->id_1 = read_tap_reg(OMAP_TAP_DIE_ID_1);
132*4882a593Smuzhiyun 	odi->id_2 = read_tap_reg(OMAP_TAP_DIE_ID_2);
133*4882a593Smuzhiyun 	odi->id_3 = read_tap_reg(OMAP_TAP_DIE_ID_3);
134*4882a593Smuzhiyun }
135*4882a593Smuzhiyun 
omap_feed_randpool(void)136*4882a593Smuzhiyun static int __init omap_feed_randpool(void)
137*4882a593Smuzhiyun {
138*4882a593Smuzhiyun 	struct omap_die_id odi;
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun 	/* Throw the die ID into the entropy pool at boot */
141*4882a593Smuzhiyun 	omap_get_die_id(&odi);
142*4882a593Smuzhiyun 	add_device_randomness(&odi, sizeof(odi));
143*4882a593Smuzhiyun 	return 0;
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun omap_device_initcall(omap_feed_randpool);
146*4882a593Smuzhiyun 
omap2xxx_check_revision(void)147*4882a593Smuzhiyun void __init omap2xxx_check_revision(void)
148*4882a593Smuzhiyun {
149*4882a593Smuzhiyun 	int i, j;
150*4882a593Smuzhiyun 	u32 idcode, prod_id;
151*4882a593Smuzhiyun 	u16 hawkeye;
152*4882a593Smuzhiyun 	u8  dev_type, rev;
153*4882a593Smuzhiyun 	struct omap_die_id odi;
154*4882a593Smuzhiyun 
155*4882a593Smuzhiyun 	idcode = read_tap_reg(OMAP_TAP_IDCODE);
156*4882a593Smuzhiyun 	prod_id = read_tap_reg(tap_prod_id);
157*4882a593Smuzhiyun 	hawkeye = (idcode >> 12) & 0xffff;
158*4882a593Smuzhiyun 	rev = (idcode >> 28) & 0x0f;
159*4882a593Smuzhiyun 	dev_type = (prod_id >> 16) & 0x0f;
160*4882a593Smuzhiyun 	omap_get_die_id(&odi);
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun 	pr_debug("OMAP_TAP_IDCODE 0x%08x REV %i HAWKEYE 0x%04x MANF %03x\n",
163*4882a593Smuzhiyun 		 idcode, rev, hawkeye, (idcode >> 1) & 0x7ff);
164*4882a593Smuzhiyun 	pr_debug("OMAP_TAP_DIE_ID_0: 0x%08x\n", odi.id_0);
165*4882a593Smuzhiyun 	pr_debug("OMAP_TAP_DIE_ID_1: 0x%08x DEV_REV: %i\n",
166*4882a593Smuzhiyun 		 odi.id_1, (odi.id_1 >> 28) & 0xf);
167*4882a593Smuzhiyun 	pr_debug("OMAP_TAP_DIE_ID_2: 0x%08x\n", odi.id_2);
168*4882a593Smuzhiyun 	pr_debug("OMAP_TAP_DIE_ID_3: 0x%08x\n", odi.id_3);
169*4882a593Smuzhiyun 	pr_debug("OMAP_TAP_PROD_ID_0: 0x%08x DEV_TYPE: %i\n",
170*4882a593Smuzhiyun 		 prod_id, dev_type);
171*4882a593Smuzhiyun 
172*4882a593Smuzhiyun 	/* Check hawkeye ids */
173*4882a593Smuzhiyun 	for (i = 0; i < ARRAY_SIZE(omap_ids); i++) {
174*4882a593Smuzhiyun 		if (hawkeye == omap_ids[i].hawkeye)
175*4882a593Smuzhiyun 			break;
176*4882a593Smuzhiyun 	}
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun 	if (i == ARRAY_SIZE(omap_ids)) {
179*4882a593Smuzhiyun 		printk(KERN_ERR "Unknown OMAP CPU id\n");
180*4882a593Smuzhiyun 		return;
181*4882a593Smuzhiyun 	}
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun 	for (j = i; j < ARRAY_SIZE(omap_ids); j++) {
184*4882a593Smuzhiyun 		if (dev_type == omap_ids[j].dev)
185*4882a593Smuzhiyun 			break;
186*4882a593Smuzhiyun 	}
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun 	if (j == ARRAY_SIZE(omap_ids)) {
189*4882a593Smuzhiyun 		pr_err("Unknown OMAP device type. Handling it as OMAP%04x\n",
190*4882a593Smuzhiyun 		       omap_ids[i].type >> 16);
191*4882a593Smuzhiyun 		j = i;
192*4882a593Smuzhiyun 	}
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun 	sprintf(soc_name, "OMAP%04x", omap_rev() >> 16);
195*4882a593Smuzhiyun 	sprintf(soc_rev, "ES%x", (omap_rev() >> 12) & 0xf);
196*4882a593Smuzhiyun 
197*4882a593Smuzhiyun 	pr_info("%s", soc_name);
198*4882a593Smuzhiyun 	if ((omap_rev() >> 8) & 0x0f)
199*4882a593Smuzhiyun 		pr_cont("%s", soc_rev);
200*4882a593Smuzhiyun 	pr_cont("\n");
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun #define OMAP3_SHOW_FEATURE(feat)		\
204*4882a593Smuzhiyun 	if (omap3_has_ ##feat())		\
205*4882a593Smuzhiyun 		n += scnprintf(buf + n, sizeof(buf) - n, #feat " ");
206*4882a593Smuzhiyun 
omap3_cpuinfo(void)207*4882a593Smuzhiyun static void __init omap3_cpuinfo(void)
208*4882a593Smuzhiyun {
209*4882a593Smuzhiyun 	const char *cpu_name;
210*4882a593Smuzhiyun 	char buf[64];
211*4882a593Smuzhiyun 	int n = 0;
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	memset(buf, 0, sizeof(buf));
214*4882a593Smuzhiyun 
215*4882a593Smuzhiyun 	/*
216*4882a593Smuzhiyun 	 * OMAP3430 and OMAP3530 are assumed to be same.
217*4882a593Smuzhiyun 	 *
218*4882a593Smuzhiyun 	 * OMAP3525, OMAP3515 and OMAP3503 can be detected only based
219*4882a593Smuzhiyun 	 * on available features. Upon detection, update the CPU id
220*4882a593Smuzhiyun 	 * and CPU class bits.
221*4882a593Smuzhiyun 	 */
222*4882a593Smuzhiyun 	if (soc_is_omap3630()) {
223*4882a593Smuzhiyun 		if (omap3_has_iva() && omap3_has_sgx()) {
224*4882a593Smuzhiyun 			cpu_name = (omap3_has_isp()) ? "OMAP3630/DM3730" : "OMAP3621";
225*4882a593Smuzhiyun 		} else if (omap3_has_iva()) {
226*4882a593Smuzhiyun 			cpu_name = "DM3725";
227*4882a593Smuzhiyun 		} else if (omap3_has_sgx()) {
228*4882a593Smuzhiyun 			cpu_name = "OMAP3615/AM3715";
229*4882a593Smuzhiyun 		} else {
230*4882a593Smuzhiyun 			cpu_name = (omap3_has_isp()) ? "AM3703" : "OMAP3611";
231*4882a593Smuzhiyun 		}
232*4882a593Smuzhiyun 	} else if (soc_is_am35xx()) {
233*4882a593Smuzhiyun 		cpu_name = (omap3_has_sgx()) ? "AM3517" : "AM3505";
234*4882a593Smuzhiyun 	} else if (soc_is_ti816x()) {
235*4882a593Smuzhiyun 		cpu_name = "TI816X";
236*4882a593Smuzhiyun 	} else if (soc_is_am335x()) {
237*4882a593Smuzhiyun 		cpu_name =  "AM335X";
238*4882a593Smuzhiyun 	} else if (soc_is_am437x()) {
239*4882a593Smuzhiyun 		cpu_name =  "AM437x";
240*4882a593Smuzhiyun 	} else if (soc_is_ti814x()) {
241*4882a593Smuzhiyun 		cpu_name = "TI814X";
242*4882a593Smuzhiyun 	} else if (omap3_has_iva() && omap3_has_sgx()) {
243*4882a593Smuzhiyun 		/* OMAP3430, OMAP3525, OMAP3515, OMAP3503 devices */
244*4882a593Smuzhiyun 		cpu_name = "OMAP3430/3530";
245*4882a593Smuzhiyun 	} else if (omap3_has_iva()) {
246*4882a593Smuzhiyun 		cpu_name = "OMAP3525";
247*4882a593Smuzhiyun 	} else if (omap3_has_sgx()) {
248*4882a593Smuzhiyun 		cpu_name = "OMAP3515";
249*4882a593Smuzhiyun 	} else {
250*4882a593Smuzhiyun 		cpu_name = "OMAP3503";
251*4882a593Smuzhiyun 	}
252*4882a593Smuzhiyun 
253*4882a593Smuzhiyun 	scnprintf(soc_name, sizeof(soc_name), "%s", cpu_name);
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun 	/* Print verbose information */
256*4882a593Smuzhiyun 	n += scnprintf(buf, sizeof(buf) - n, "%s %s (", soc_name, soc_rev);
257*4882a593Smuzhiyun 
258*4882a593Smuzhiyun 	OMAP3_SHOW_FEATURE(l2cache);
259*4882a593Smuzhiyun 	OMAP3_SHOW_FEATURE(iva);
260*4882a593Smuzhiyun 	OMAP3_SHOW_FEATURE(sgx);
261*4882a593Smuzhiyun 	OMAP3_SHOW_FEATURE(neon);
262*4882a593Smuzhiyun 	OMAP3_SHOW_FEATURE(isp);
263*4882a593Smuzhiyun 	OMAP3_SHOW_FEATURE(192mhz_clk);
264*4882a593Smuzhiyun 	if (*(buf + n - 1) == ' ')
265*4882a593Smuzhiyun 		n--;
266*4882a593Smuzhiyun 	n += scnprintf(buf + n, sizeof(buf) - n, ")\n");
267*4882a593Smuzhiyun 	pr_info("%s", buf);
268*4882a593Smuzhiyun }
269*4882a593Smuzhiyun 
270*4882a593Smuzhiyun #define OMAP3_CHECK_FEATURE(status,feat)				\
271*4882a593Smuzhiyun 	if (((status & OMAP3_ ##feat## _MASK) 				\
272*4882a593Smuzhiyun 		>> OMAP3_ ##feat## _SHIFT) != FEAT_ ##feat## _NONE) { 	\
273*4882a593Smuzhiyun 		omap_features |= OMAP3_HAS_ ##feat;			\
274*4882a593Smuzhiyun 	}
275*4882a593Smuzhiyun 
omap3xxx_check_features(void)276*4882a593Smuzhiyun void __init omap3xxx_check_features(void)
277*4882a593Smuzhiyun {
278*4882a593Smuzhiyun 	u32 status;
279*4882a593Smuzhiyun 
280*4882a593Smuzhiyun 	omap_features = 0;
281*4882a593Smuzhiyun 
282*4882a593Smuzhiyun 	status = omap_ctrl_readl(OMAP3_CONTROL_OMAP_STATUS);
283*4882a593Smuzhiyun 
284*4882a593Smuzhiyun 	OMAP3_CHECK_FEATURE(status, L2CACHE);
285*4882a593Smuzhiyun 	OMAP3_CHECK_FEATURE(status, IVA);
286*4882a593Smuzhiyun 	OMAP3_CHECK_FEATURE(status, SGX);
287*4882a593Smuzhiyun 	OMAP3_CHECK_FEATURE(status, NEON);
288*4882a593Smuzhiyun 	OMAP3_CHECK_FEATURE(status, ISP);
289*4882a593Smuzhiyun 	if (soc_is_omap3630())
290*4882a593Smuzhiyun 		omap_features |= OMAP3_HAS_192MHZ_CLK;
291*4882a593Smuzhiyun 	if (soc_is_omap3430() || soc_is_omap3630())
292*4882a593Smuzhiyun 		omap_features |= OMAP3_HAS_IO_WAKEUP;
293*4882a593Smuzhiyun 	if (soc_is_omap3630() || omap_rev() == OMAP3430_REV_ES3_1 ||
294*4882a593Smuzhiyun 	    omap_rev() == OMAP3430_REV_ES3_1_2)
295*4882a593Smuzhiyun 		omap_features |= OMAP3_HAS_IO_CHAIN_CTRL;
296*4882a593Smuzhiyun 
297*4882a593Smuzhiyun 	omap_features |= OMAP3_HAS_SDRC;
298*4882a593Smuzhiyun 
299*4882a593Smuzhiyun 	/*
300*4882a593Smuzhiyun 	 * am35x fixups:
301*4882a593Smuzhiyun 	 * - The am35x Chip ID register has bits 12, 7:5, and 3:2 marked as
302*4882a593Smuzhiyun 	 *   reserved and therefore return 0 when read.  Unfortunately,
303*4882a593Smuzhiyun 	 *   OMAP3_CHECK_FEATURE() will interpret some of those zeroes to
304*4882a593Smuzhiyun 	 *   mean that a feature is present even though it isn't so clear
305*4882a593Smuzhiyun 	 *   the incorrectly set feature bits.
306*4882a593Smuzhiyun 	 */
307*4882a593Smuzhiyun 	if (soc_is_am35xx())
308*4882a593Smuzhiyun 		omap_features &= ~(OMAP3_HAS_IVA | OMAP3_HAS_ISP);
309*4882a593Smuzhiyun 
310*4882a593Smuzhiyun 	/*
311*4882a593Smuzhiyun 	 * TODO: Get additional info (where applicable)
312*4882a593Smuzhiyun 	 *       e.g. Size of L2 cache.
313*4882a593Smuzhiyun 	 */
314*4882a593Smuzhiyun 
315*4882a593Smuzhiyun 	omap3_cpuinfo();
316*4882a593Smuzhiyun }
317*4882a593Smuzhiyun 
omap4xxx_check_features(void)318*4882a593Smuzhiyun void __init omap4xxx_check_features(void)
319*4882a593Smuzhiyun {
320*4882a593Smuzhiyun 	u32 si_type;
321*4882a593Smuzhiyun 
322*4882a593Smuzhiyun 	si_type =
323*4882a593Smuzhiyun 	(read_tap_reg(OMAP4_CTRL_MODULE_CORE_STD_FUSE_PROD_ID_1) >> 16) & 0x03;
324*4882a593Smuzhiyun 
325*4882a593Smuzhiyun 	if (si_type == OMAP4_SILICON_TYPE_PERFORMANCE)
326*4882a593Smuzhiyun 		omap_features = OMAP4_HAS_PERF_SILICON;
327*4882a593Smuzhiyun }
328*4882a593Smuzhiyun 
ti81xx_check_features(void)329*4882a593Smuzhiyun void __init ti81xx_check_features(void)
330*4882a593Smuzhiyun {
331*4882a593Smuzhiyun 	omap_features = OMAP3_HAS_NEON;
332*4882a593Smuzhiyun 	omap3_cpuinfo();
333*4882a593Smuzhiyun }
334*4882a593Smuzhiyun 
am33xx_check_features(void)335*4882a593Smuzhiyun void __init am33xx_check_features(void)
336*4882a593Smuzhiyun {
337*4882a593Smuzhiyun 	u32 status;
338*4882a593Smuzhiyun 
339*4882a593Smuzhiyun 	omap_features = OMAP3_HAS_NEON;
340*4882a593Smuzhiyun 
341*4882a593Smuzhiyun 	status = omap_ctrl_readl(AM33XX_DEV_FEATURE);
342*4882a593Smuzhiyun 	if (status & AM33XX_SGX_MASK)
343*4882a593Smuzhiyun 		omap_features |= OMAP3_HAS_SGX;
344*4882a593Smuzhiyun 
345*4882a593Smuzhiyun 	omap3_cpuinfo();
346*4882a593Smuzhiyun }
347*4882a593Smuzhiyun 
omap3xxx_check_revision(void)348*4882a593Smuzhiyun void __init omap3xxx_check_revision(void)
349*4882a593Smuzhiyun {
350*4882a593Smuzhiyun 	const char *cpu_rev;
351*4882a593Smuzhiyun 	u32 cpuid, idcode;
352*4882a593Smuzhiyun 	u16 hawkeye;
353*4882a593Smuzhiyun 	u8 rev;
354*4882a593Smuzhiyun 
355*4882a593Smuzhiyun 	/*
356*4882a593Smuzhiyun 	 * We cannot access revision registers on ES1.0.
357*4882a593Smuzhiyun 	 * If the processor type is Cortex-A8 and the revision is 0x0
358*4882a593Smuzhiyun 	 * it means its Cortex r0p0 which is 3430 ES1.0.
359*4882a593Smuzhiyun 	 */
360*4882a593Smuzhiyun 	cpuid = read_cpuid_id();
361*4882a593Smuzhiyun 	if ((((cpuid >> 4) & 0xfff) == 0xc08) && ((cpuid & 0xf) == 0x0)) {
362*4882a593Smuzhiyun 		omap_revision = OMAP3430_REV_ES1_0;
363*4882a593Smuzhiyun 		cpu_rev = "1.0";
364*4882a593Smuzhiyun 		return;
365*4882a593Smuzhiyun 	}
366*4882a593Smuzhiyun 
367*4882a593Smuzhiyun 	/*
368*4882a593Smuzhiyun 	 * Detection for 34xx ES2.0 and above can be done with just
369*4882a593Smuzhiyun 	 * hawkeye and rev. See TRM 1.5.2 Device Identification.
370*4882a593Smuzhiyun 	 * Note that rev does not map directly to our defined processor
371*4882a593Smuzhiyun 	 * revision numbers as ES1.0 uses value 0.
372*4882a593Smuzhiyun 	 */
373*4882a593Smuzhiyun 	idcode = read_tap_reg(OMAP_TAP_IDCODE);
374*4882a593Smuzhiyun 	hawkeye = (idcode >> 12) & 0xffff;
375*4882a593Smuzhiyun 	rev = (idcode >> 28) & 0xff;
376*4882a593Smuzhiyun 
377*4882a593Smuzhiyun 	switch (hawkeye) {
378*4882a593Smuzhiyun 	case 0xb7ae:
379*4882a593Smuzhiyun 		/* Handle 34xx/35xx devices */
380*4882a593Smuzhiyun 		switch (rev) {
381*4882a593Smuzhiyun 		case 0: /* Take care of early samples */
382*4882a593Smuzhiyun 		case 1:
383*4882a593Smuzhiyun 			omap_revision = OMAP3430_REV_ES2_0;
384*4882a593Smuzhiyun 			cpu_rev = "2.0";
385*4882a593Smuzhiyun 			break;
386*4882a593Smuzhiyun 		case 2:
387*4882a593Smuzhiyun 			omap_revision = OMAP3430_REV_ES2_1;
388*4882a593Smuzhiyun 			cpu_rev = "2.1";
389*4882a593Smuzhiyun 			break;
390*4882a593Smuzhiyun 		case 3:
391*4882a593Smuzhiyun 			omap_revision = OMAP3430_REV_ES3_0;
392*4882a593Smuzhiyun 			cpu_rev = "3.0";
393*4882a593Smuzhiyun 			break;
394*4882a593Smuzhiyun 		case 4:
395*4882a593Smuzhiyun 			omap_revision = OMAP3430_REV_ES3_1;
396*4882a593Smuzhiyun 			cpu_rev = "3.1";
397*4882a593Smuzhiyun 			break;
398*4882a593Smuzhiyun 		case 7:
399*4882a593Smuzhiyun 		default:
400*4882a593Smuzhiyun 			/* Use the latest known revision as default */
401*4882a593Smuzhiyun 			omap_revision = OMAP3430_REV_ES3_1_2;
402*4882a593Smuzhiyun 			cpu_rev = "3.1.2";
403*4882a593Smuzhiyun 		}
404*4882a593Smuzhiyun 		break;
405*4882a593Smuzhiyun 	case 0xb868:
406*4882a593Smuzhiyun 		/*
407*4882a593Smuzhiyun 		 * Handle OMAP/AM 3505/3517 devices
408*4882a593Smuzhiyun 		 *
409*4882a593Smuzhiyun 		 * Set the device to be OMAP3517 here. Actual device
410*4882a593Smuzhiyun 		 * is identified later based on the features.
411*4882a593Smuzhiyun 		 */
412*4882a593Smuzhiyun 		switch (rev) {
413*4882a593Smuzhiyun 		case 0:
414*4882a593Smuzhiyun 			omap_revision = AM35XX_REV_ES1_0;
415*4882a593Smuzhiyun 			cpu_rev = "1.0";
416*4882a593Smuzhiyun 			break;
417*4882a593Smuzhiyun 		case 1:
418*4882a593Smuzhiyun 		default:
419*4882a593Smuzhiyun 			omap_revision = AM35XX_REV_ES1_1;
420*4882a593Smuzhiyun 			cpu_rev = "1.1";
421*4882a593Smuzhiyun 		}
422*4882a593Smuzhiyun 		break;
423*4882a593Smuzhiyun 	case 0xb891:
424*4882a593Smuzhiyun 		/* Handle 36xx devices */
425*4882a593Smuzhiyun 
426*4882a593Smuzhiyun 		switch(rev) {
427*4882a593Smuzhiyun 		case 0: /* Take care of early samples */
428*4882a593Smuzhiyun 			omap_revision = OMAP3630_REV_ES1_0;
429*4882a593Smuzhiyun 			cpu_rev = "1.0";
430*4882a593Smuzhiyun 			break;
431*4882a593Smuzhiyun 		case 1:
432*4882a593Smuzhiyun 			omap_revision = OMAP3630_REV_ES1_1;
433*4882a593Smuzhiyun 			cpu_rev = "1.1";
434*4882a593Smuzhiyun 			break;
435*4882a593Smuzhiyun 		case 2:
436*4882a593Smuzhiyun 		default:
437*4882a593Smuzhiyun 			omap_revision = OMAP3630_REV_ES1_2;
438*4882a593Smuzhiyun 			cpu_rev = "1.2";
439*4882a593Smuzhiyun 		}
440*4882a593Smuzhiyun 		break;
441*4882a593Smuzhiyun 	case 0xb81e:
442*4882a593Smuzhiyun 		switch (rev) {
443*4882a593Smuzhiyun 		case 0:
444*4882a593Smuzhiyun 			omap_revision = TI8168_REV_ES1_0;
445*4882a593Smuzhiyun 			cpu_rev = "1.0";
446*4882a593Smuzhiyun 			break;
447*4882a593Smuzhiyun 		case 1:
448*4882a593Smuzhiyun 			omap_revision = TI8168_REV_ES1_1;
449*4882a593Smuzhiyun 			cpu_rev = "1.1";
450*4882a593Smuzhiyun 			break;
451*4882a593Smuzhiyun 		case 2:
452*4882a593Smuzhiyun 			omap_revision = TI8168_REV_ES2_0;
453*4882a593Smuzhiyun 			cpu_rev = "2.0";
454*4882a593Smuzhiyun 			break;
455*4882a593Smuzhiyun 		case 3:
456*4882a593Smuzhiyun 		default:
457*4882a593Smuzhiyun 			omap_revision = TI8168_REV_ES2_1;
458*4882a593Smuzhiyun 			cpu_rev = "2.1";
459*4882a593Smuzhiyun 		}
460*4882a593Smuzhiyun 		break;
461*4882a593Smuzhiyun 	case 0xb944:
462*4882a593Smuzhiyun 		switch (rev) {
463*4882a593Smuzhiyun 		case 0:
464*4882a593Smuzhiyun 			omap_revision = AM335X_REV_ES1_0;
465*4882a593Smuzhiyun 			cpu_rev = "1.0";
466*4882a593Smuzhiyun 			break;
467*4882a593Smuzhiyun 		case 1:
468*4882a593Smuzhiyun 			omap_revision = AM335X_REV_ES2_0;
469*4882a593Smuzhiyun 			cpu_rev = "2.0";
470*4882a593Smuzhiyun 			break;
471*4882a593Smuzhiyun 		case 2:
472*4882a593Smuzhiyun 		default:
473*4882a593Smuzhiyun 			omap_revision = AM335X_REV_ES2_1;
474*4882a593Smuzhiyun 			cpu_rev = "2.1";
475*4882a593Smuzhiyun 			break;
476*4882a593Smuzhiyun 		}
477*4882a593Smuzhiyun 		break;
478*4882a593Smuzhiyun 	case 0xb98c:
479*4882a593Smuzhiyun 		switch (rev) {
480*4882a593Smuzhiyun 		case 0:
481*4882a593Smuzhiyun 			omap_revision = AM437X_REV_ES1_0;
482*4882a593Smuzhiyun 			cpu_rev = "1.0";
483*4882a593Smuzhiyun 			break;
484*4882a593Smuzhiyun 		case 1:
485*4882a593Smuzhiyun 			omap_revision = AM437X_REV_ES1_1;
486*4882a593Smuzhiyun 			cpu_rev = "1.1";
487*4882a593Smuzhiyun 			break;
488*4882a593Smuzhiyun 		case 2:
489*4882a593Smuzhiyun 		default:
490*4882a593Smuzhiyun 			omap_revision = AM437X_REV_ES1_2;
491*4882a593Smuzhiyun 			cpu_rev = "1.2";
492*4882a593Smuzhiyun 			break;
493*4882a593Smuzhiyun 		}
494*4882a593Smuzhiyun 		break;
495*4882a593Smuzhiyun 	case 0xb8f2:
496*4882a593Smuzhiyun 	case 0xb968:
497*4882a593Smuzhiyun 		switch (rev) {
498*4882a593Smuzhiyun 		case 0:
499*4882a593Smuzhiyun 		case 1:
500*4882a593Smuzhiyun 			omap_revision = TI8148_REV_ES1_0;
501*4882a593Smuzhiyun 			cpu_rev = "1.0";
502*4882a593Smuzhiyun 			break;
503*4882a593Smuzhiyun 		case 2:
504*4882a593Smuzhiyun 			omap_revision = TI8148_REV_ES2_0;
505*4882a593Smuzhiyun 			cpu_rev = "2.0";
506*4882a593Smuzhiyun 			break;
507*4882a593Smuzhiyun 		case 3:
508*4882a593Smuzhiyun 		default:
509*4882a593Smuzhiyun 			omap_revision = TI8148_REV_ES2_1;
510*4882a593Smuzhiyun 			cpu_rev = "2.1";
511*4882a593Smuzhiyun 			break;
512*4882a593Smuzhiyun 		}
513*4882a593Smuzhiyun 		break;
514*4882a593Smuzhiyun 	default:
515*4882a593Smuzhiyun 		/* Unknown default to latest silicon rev as default */
516*4882a593Smuzhiyun 		omap_revision = OMAP3630_REV_ES1_2;
517*4882a593Smuzhiyun 		cpu_rev = "1.2";
518*4882a593Smuzhiyun 		pr_warn("Warning: unknown chip type: hawkeye %04x, assuming OMAP3630ES1.2\n",
519*4882a593Smuzhiyun 			hawkeye);
520*4882a593Smuzhiyun 	}
521*4882a593Smuzhiyun 	sprintf(soc_rev, "ES%s", cpu_rev);
522*4882a593Smuzhiyun }
523*4882a593Smuzhiyun 
omap4xxx_check_revision(void)524*4882a593Smuzhiyun void __init omap4xxx_check_revision(void)
525*4882a593Smuzhiyun {
526*4882a593Smuzhiyun 	u32 idcode;
527*4882a593Smuzhiyun 	u16 hawkeye;
528*4882a593Smuzhiyun 	u8 rev;
529*4882a593Smuzhiyun 
530*4882a593Smuzhiyun 	/*
531*4882a593Smuzhiyun 	 * The IC rev detection is done with hawkeye and rev.
532*4882a593Smuzhiyun 	 * Note that rev does not map directly to defined processor
533*4882a593Smuzhiyun 	 * revision numbers as ES1.0 uses value 0.
534*4882a593Smuzhiyun 	 */
535*4882a593Smuzhiyun 	idcode = read_tap_reg(OMAP_TAP_IDCODE);
536*4882a593Smuzhiyun 	hawkeye = (idcode >> 12) & 0xffff;
537*4882a593Smuzhiyun 	rev = (idcode >> 28) & 0xf;
538*4882a593Smuzhiyun 
539*4882a593Smuzhiyun 	/*
540*4882a593Smuzhiyun 	 * Few initial 4430 ES2.0 samples IDCODE is same as ES1.0
541*4882a593Smuzhiyun 	 * Use ARM register to detect the correct ES version
542*4882a593Smuzhiyun 	 */
543*4882a593Smuzhiyun 	if (!rev && (hawkeye != 0xb94e) && (hawkeye != 0xb975)) {
544*4882a593Smuzhiyun 		idcode = read_cpuid_id();
545*4882a593Smuzhiyun 		rev = (idcode & 0xf) - 1;
546*4882a593Smuzhiyun 	}
547*4882a593Smuzhiyun 
548*4882a593Smuzhiyun 	switch (hawkeye) {
549*4882a593Smuzhiyun 	case 0xb852:
550*4882a593Smuzhiyun 		switch (rev) {
551*4882a593Smuzhiyun 		case 0:
552*4882a593Smuzhiyun 			omap_revision = OMAP4430_REV_ES1_0;
553*4882a593Smuzhiyun 			break;
554*4882a593Smuzhiyun 		case 1:
555*4882a593Smuzhiyun 		default:
556*4882a593Smuzhiyun 			omap_revision = OMAP4430_REV_ES2_0;
557*4882a593Smuzhiyun 		}
558*4882a593Smuzhiyun 		break;
559*4882a593Smuzhiyun 	case 0xb95c:
560*4882a593Smuzhiyun 		switch (rev) {
561*4882a593Smuzhiyun 		case 3:
562*4882a593Smuzhiyun 			omap_revision = OMAP4430_REV_ES2_1;
563*4882a593Smuzhiyun 			break;
564*4882a593Smuzhiyun 		case 4:
565*4882a593Smuzhiyun 			omap_revision = OMAP4430_REV_ES2_2;
566*4882a593Smuzhiyun 			break;
567*4882a593Smuzhiyun 		case 6:
568*4882a593Smuzhiyun 		default:
569*4882a593Smuzhiyun 			omap_revision = OMAP4430_REV_ES2_3;
570*4882a593Smuzhiyun 		}
571*4882a593Smuzhiyun 		break;
572*4882a593Smuzhiyun 	case 0xb94e:
573*4882a593Smuzhiyun 		switch (rev) {
574*4882a593Smuzhiyun 		case 0:
575*4882a593Smuzhiyun 			omap_revision = OMAP4460_REV_ES1_0;
576*4882a593Smuzhiyun 			break;
577*4882a593Smuzhiyun 		case 2:
578*4882a593Smuzhiyun 		default:
579*4882a593Smuzhiyun 			omap_revision = OMAP4460_REV_ES1_1;
580*4882a593Smuzhiyun 			break;
581*4882a593Smuzhiyun 		}
582*4882a593Smuzhiyun 		break;
583*4882a593Smuzhiyun 	case 0xb975:
584*4882a593Smuzhiyun 		switch (rev) {
585*4882a593Smuzhiyun 		case 0:
586*4882a593Smuzhiyun 		default:
587*4882a593Smuzhiyun 			omap_revision = OMAP4470_REV_ES1_0;
588*4882a593Smuzhiyun 			break;
589*4882a593Smuzhiyun 		}
590*4882a593Smuzhiyun 		break;
591*4882a593Smuzhiyun 	default:
592*4882a593Smuzhiyun 		/* Unknown default to latest silicon rev as default */
593*4882a593Smuzhiyun 		omap_revision = OMAP4430_REV_ES2_3;
594*4882a593Smuzhiyun 	}
595*4882a593Smuzhiyun 
596*4882a593Smuzhiyun 	sprintf(soc_name, "OMAP%04x", omap_rev() >> 16);
597*4882a593Smuzhiyun 	sprintf(soc_rev, "ES%d.%d", (omap_rev() >> 12) & 0xf,
598*4882a593Smuzhiyun 						(omap_rev() >> 8) & 0xf);
599*4882a593Smuzhiyun 	pr_info("%s %s\n", soc_name, soc_rev);
600*4882a593Smuzhiyun }
601*4882a593Smuzhiyun 
omap5xxx_check_revision(void)602*4882a593Smuzhiyun void __init omap5xxx_check_revision(void)
603*4882a593Smuzhiyun {
604*4882a593Smuzhiyun 	u32 idcode;
605*4882a593Smuzhiyun 	u16 hawkeye;
606*4882a593Smuzhiyun 	u8 rev;
607*4882a593Smuzhiyun 
608*4882a593Smuzhiyun 	idcode = read_tap_reg(OMAP_TAP_IDCODE);
609*4882a593Smuzhiyun 	hawkeye = (idcode >> 12) & 0xffff;
610*4882a593Smuzhiyun 	rev = (idcode >> 28) & 0xff;
611*4882a593Smuzhiyun 	switch (hawkeye) {
612*4882a593Smuzhiyun 	case 0xb942:
613*4882a593Smuzhiyun 		switch (rev) {
614*4882a593Smuzhiyun 		case 0:
615*4882a593Smuzhiyun 			/* No support for ES1.0 Test chip */
616*4882a593Smuzhiyun 			BUG();
617*4882a593Smuzhiyun 		case 1:
618*4882a593Smuzhiyun 		default:
619*4882a593Smuzhiyun 			omap_revision = OMAP5430_REV_ES2_0;
620*4882a593Smuzhiyun 		}
621*4882a593Smuzhiyun 		break;
622*4882a593Smuzhiyun 
623*4882a593Smuzhiyun 	case 0xb998:
624*4882a593Smuzhiyun 		switch (rev) {
625*4882a593Smuzhiyun 		case 0:
626*4882a593Smuzhiyun 			/* No support for ES1.0 Test chip */
627*4882a593Smuzhiyun 			BUG();
628*4882a593Smuzhiyun 		case 1:
629*4882a593Smuzhiyun 		default:
630*4882a593Smuzhiyun 			omap_revision = OMAP5432_REV_ES2_0;
631*4882a593Smuzhiyun 		}
632*4882a593Smuzhiyun 		break;
633*4882a593Smuzhiyun 
634*4882a593Smuzhiyun 	default:
635*4882a593Smuzhiyun 		/* Unknown default to latest silicon rev as default*/
636*4882a593Smuzhiyun 		omap_revision = OMAP5430_REV_ES2_0;
637*4882a593Smuzhiyun 	}
638*4882a593Smuzhiyun 
639*4882a593Smuzhiyun 	sprintf(soc_name, "OMAP%04x", omap_rev() >> 16);
640*4882a593Smuzhiyun 	sprintf(soc_rev, "ES%d.0", (omap_rev() >> 12) & 0xf);
641*4882a593Smuzhiyun 
642*4882a593Smuzhiyun 	pr_info("%s %s\n", soc_name, soc_rev);
643*4882a593Smuzhiyun }
644*4882a593Smuzhiyun 
dra7xxx_check_revision(void)645*4882a593Smuzhiyun void __init dra7xxx_check_revision(void)
646*4882a593Smuzhiyun {
647*4882a593Smuzhiyun 	u32 idcode;
648*4882a593Smuzhiyun 	u16 hawkeye;
649*4882a593Smuzhiyun 	u8 rev, package;
650*4882a593Smuzhiyun 	struct omap_die_id odi;
651*4882a593Smuzhiyun 
652*4882a593Smuzhiyun 	omap_get_die_id(&odi);
653*4882a593Smuzhiyun 	package = (odi.id_2 >> 16) & 0x3;
654*4882a593Smuzhiyun 	idcode = read_tap_reg(OMAP_TAP_IDCODE);
655*4882a593Smuzhiyun 	hawkeye = (idcode >> 12) & 0xffff;
656*4882a593Smuzhiyun 	rev = (idcode >> 28) & 0xff;
657*4882a593Smuzhiyun 	switch (hawkeye) {
658*4882a593Smuzhiyun 	case 0xbb50:
659*4882a593Smuzhiyun 		switch (rev) {
660*4882a593Smuzhiyun 		case 0:
661*4882a593Smuzhiyun 		default:
662*4882a593Smuzhiyun 			switch (package) {
663*4882a593Smuzhiyun 			case 0x2:
664*4882a593Smuzhiyun 				omap_revision = DRA762_ABZ_REV_ES1_0;
665*4882a593Smuzhiyun 				break;
666*4882a593Smuzhiyun 			case 0x3:
667*4882a593Smuzhiyun 				omap_revision = DRA762_ACD_REV_ES1_0;
668*4882a593Smuzhiyun 				break;
669*4882a593Smuzhiyun 			default:
670*4882a593Smuzhiyun 				omap_revision = DRA762_REV_ES1_0;
671*4882a593Smuzhiyun 				break;
672*4882a593Smuzhiyun 			}
673*4882a593Smuzhiyun 			break;
674*4882a593Smuzhiyun 		}
675*4882a593Smuzhiyun 		break;
676*4882a593Smuzhiyun 
677*4882a593Smuzhiyun 	case 0xb990:
678*4882a593Smuzhiyun 		switch (rev) {
679*4882a593Smuzhiyun 		case 0:
680*4882a593Smuzhiyun 			omap_revision = DRA752_REV_ES1_0;
681*4882a593Smuzhiyun 			break;
682*4882a593Smuzhiyun 		case 1:
683*4882a593Smuzhiyun 			omap_revision = DRA752_REV_ES1_1;
684*4882a593Smuzhiyun 			break;
685*4882a593Smuzhiyun 		case 2:
686*4882a593Smuzhiyun 		default:
687*4882a593Smuzhiyun 			omap_revision = DRA752_REV_ES2_0;
688*4882a593Smuzhiyun 			break;
689*4882a593Smuzhiyun 		}
690*4882a593Smuzhiyun 		break;
691*4882a593Smuzhiyun 
692*4882a593Smuzhiyun 	case 0xb9bc:
693*4882a593Smuzhiyun 		switch (rev) {
694*4882a593Smuzhiyun 		case 0:
695*4882a593Smuzhiyun 			omap_revision = DRA722_REV_ES1_0;
696*4882a593Smuzhiyun 			break;
697*4882a593Smuzhiyun 		case 1:
698*4882a593Smuzhiyun 			omap_revision = DRA722_REV_ES2_0;
699*4882a593Smuzhiyun 			break;
700*4882a593Smuzhiyun 		case 2:
701*4882a593Smuzhiyun 		default:
702*4882a593Smuzhiyun 			omap_revision = DRA722_REV_ES2_1;
703*4882a593Smuzhiyun 			break;
704*4882a593Smuzhiyun 		}
705*4882a593Smuzhiyun 		break;
706*4882a593Smuzhiyun 
707*4882a593Smuzhiyun 	default:
708*4882a593Smuzhiyun 		/* Unknown default to latest silicon rev as default*/
709*4882a593Smuzhiyun 		pr_warn("%s: unknown idcode=0x%08x (hawkeye=0x%08x,rev=0x%x)\n",
710*4882a593Smuzhiyun 			__func__, idcode, hawkeye, rev);
711*4882a593Smuzhiyun 		omap_revision = DRA752_REV_ES2_0;
712*4882a593Smuzhiyun 	}
713*4882a593Smuzhiyun 
714*4882a593Smuzhiyun 	sprintf(soc_name, "DRA%03x", omap_rev() >> 16);
715*4882a593Smuzhiyun 	sprintf(soc_rev, "ES%d.%d", (omap_rev() >> 12) & 0xf,
716*4882a593Smuzhiyun 		(omap_rev() >> 8) & 0xf);
717*4882a593Smuzhiyun 
718*4882a593Smuzhiyun 	pr_info("%s %s\n", soc_name, soc_rev);
719*4882a593Smuzhiyun }
720*4882a593Smuzhiyun 
721*4882a593Smuzhiyun /*
722*4882a593Smuzhiyun  * Set up things for map_io and processor detection later on. Gets called
723*4882a593Smuzhiyun  * pretty much first thing from board init. For multi-omap, this gets
724*4882a593Smuzhiyun  * cpu_is_omapxxxx() working accurately enough for map_io. Then we'll try to
725*4882a593Smuzhiyun  * detect the exact revision later on in omap2_detect_revision() once map_io
726*4882a593Smuzhiyun  * is done.
727*4882a593Smuzhiyun  */
omap2_set_globals_tap(u32 class,void __iomem * tap)728*4882a593Smuzhiyun void __init omap2_set_globals_tap(u32 class, void __iomem *tap)
729*4882a593Smuzhiyun {
730*4882a593Smuzhiyun 	omap_revision = class;
731*4882a593Smuzhiyun 	tap_base = tap;
732*4882a593Smuzhiyun 
733*4882a593Smuzhiyun 	/* XXX What is this intended to do? */
734*4882a593Smuzhiyun 	if (soc_is_omap34xx())
735*4882a593Smuzhiyun 		tap_prod_id = 0x0210;
736*4882a593Smuzhiyun 	else
737*4882a593Smuzhiyun 		tap_prod_id = 0x0208;
738*4882a593Smuzhiyun }
739*4882a593Smuzhiyun 
740*4882a593Smuzhiyun #ifdef CONFIG_SOC_BUS
741*4882a593Smuzhiyun 
742*4882a593Smuzhiyun static const char * const omap_types[] = {
743*4882a593Smuzhiyun 	[OMAP2_DEVICE_TYPE_TEST]	= "TST",
744*4882a593Smuzhiyun 	[OMAP2_DEVICE_TYPE_EMU]		= "EMU",
745*4882a593Smuzhiyun 	[OMAP2_DEVICE_TYPE_SEC]		= "HS",
746*4882a593Smuzhiyun 	[OMAP2_DEVICE_TYPE_GP]		= "GP",
747*4882a593Smuzhiyun 	[OMAP2_DEVICE_TYPE_BAD]		= "BAD",
748*4882a593Smuzhiyun };
749*4882a593Smuzhiyun 
omap_get_family(void)750*4882a593Smuzhiyun static const char * __init omap_get_family(void)
751*4882a593Smuzhiyun {
752*4882a593Smuzhiyun 	if (soc_is_omap24xx())
753*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "OMAP2");
754*4882a593Smuzhiyun 	else if (soc_is_omap34xx())
755*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "OMAP3");
756*4882a593Smuzhiyun 	else if (soc_is_omap44xx())
757*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "OMAP4");
758*4882a593Smuzhiyun 	else if (soc_is_omap54xx())
759*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "OMAP5");
760*4882a593Smuzhiyun 	else if (soc_is_am33xx() || soc_is_am335x())
761*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "AM33xx");
762*4882a593Smuzhiyun 	else if (soc_is_am43xx())
763*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "AM43xx");
764*4882a593Smuzhiyun 	else if (soc_is_dra7xx())
765*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "DRA7");
766*4882a593Smuzhiyun 	else
767*4882a593Smuzhiyun 		return kasprintf(GFP_KERNEL, "Unknown");
768*4882a593Smuzhiyun }
769*4882a593Smuzhiyun 
770*4882a593Smuzhiyun static ssize_t
type_show(struct device * dev,struct device_attribute * attr,char * buf)771*4882a593Smuzhiyun type_show(struct device *dev, struct device_attribute *attr, char *buf)
772*4882a593Smuzhiyun {
773*4882a593Smuzhiyun 	return sprintf(buf, "%s\n", omap_types[omap_type()]);
774*4882a593Smuzhiyun }
775*4882a593Smuzhiyun 
776*4882a593Smuzhiyun static DEVICE_ATTR_RO(type);
777*4882a593Smuzhiyun 
778*4882a593Smuzhiyun static struct attribute *omap_soc_attrs[] = {
779*4882a593Smuzhiyun 	&dev_attr_type.attr,
780*4882a593Smuzhiyun 	NULL
781*4882a593Smuzhiyun };
782*4882a593Smuzhiyun 
783*4882a593Smuzhiyun ATTRIBUTE_GROUPS(omap_soc);
784*4882a593Smuzhiyun 
omap_soc_device_init(void)785*4882a593Smuzhiyun void __init omap_soc_device_init(void)
786*4882a593Smuzhiyun {
787*4882a593Smuzhiyun 	struct soc_device *soc_dev;
788*4882a593Smuzhiyun 	struct soc_device_attribute *soc_dev_attr;
789*4882a593Smuzhiyun 
790*4882a593Smuzhiyun 	soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL);
791*4882a593Smuzhiyun 	if (!soc_dev_attr)
792*4882a593Smuzhiyun 		return;
793*4882a593Smuzhiyun 
794*4882a593Smuzhiyun 	soc_dev_attr->machine  = soc_name;
795*4882a593Smuzhiyun 	soc_dev_attr->family   = omap_get_family();
796*4882a593Smuzhiyun 	soc_dev_attr->revision = soc_rev;
797*4882a593Smuzhiyun 	soc_dev_attr->custom_attr_group = omap_soc_groups[0];
798*4882a593Smuzhiyun 
799*4882a593Smuzhiyun 	soc_dev = soc_device_register(soc_dev_attr);
800*4882a593Smuzhiyun 	if (IS_ERR(soc_dev)) {
801*4882a593Smuzhiyun 		kfree(soc_dev_attr);
802*4882a593Smuzhiyun 		return;
803*4882a593Smuzhiyun 	}
804*4882a593Smuzhiyun }
805*4882a593Smuzhiyun #endif /* CONFIG_SOC_BUS */
806