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