1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright (C) 2015-2016 Wills Wang <wills.wang@live.com>
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * SPDX-License-Identifier: GPL-2.0+
5*4882a593Smuzhiyun */
6*4882a593Smuzhiyun
7*4882a593Smuzhiyun #include <common.h>
8*4882a593Smuzhiyun #include <asm/io.h>
9*4882a593Smuzhiyun #include <asm/addrspace.h>
10*4882a593Smuzhiyun #include <asm/types.h>
11*4882a593Smuzhiyun #include <mach/ath79.h>
12*4882a593Smuzhiyun #include <mach/ar71xx_regs.h>
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun struct ath79_soc_desc {
15*4882a593Smuzhiyun const enum ath79_soc_type soc;
16*4882a593Smuzhiyun const char *chip;
17*4882a593Smuzhiyun const int major;
18*4882a593Smuzhiyun const int minor;
19*4882a593Smuzhiyun };
20*4882a593Smuzhiyun
21*4882a593Smuzhiyun static const struct ath79_soc_desc desc[] = {
22*4882a593Smuzhiyun {ATH79_SOC_AR7130, "7130",
23*4882a593Smuzhiyun REV_ID_MAJOR_AR71XX, AR71XX_REV_ID_MINOR_AR7130},
24*4882a593Smuzhiyun {ATH79_SOC_AR7141, "7141",
25*4882a593Smuzhiyun REV_ID_MAJOR_AR71XX, AR71XX_REV_ID_MINOR_AR7141},
26*4882a593Smuzhiyun {ATH79_SOC_AR7161, "7161",
27*4882a593Smuzhiyun REV_ID_MAJOR_AR71XX, AR71XX_REV_ID_MINOR_AR7161},
28*4882a593Smuzhiyun {ATH79_SOC_AR7240, "7240", REV_ID_MAJOR_AR7240, 0},
29*4882a593Smuzhiyun {ATH79_SOC_AR7241, "7241", REV_ID_MAJOR_AR7241, 0},
30*4882a593Smuzhiyun {ATH79_SOC_AR7242, "7242", REV_ID_MAJOR_AR7242, 0},
31*4882a593Smuzhiyun {ATH79_SOC_AR9130, "9130",
32*4882a593Smuzhiyun REV_ID_MAJOR_AR913X, AR913X_REV_ID_MINOR_AR9130},
33*4882a593Smuzhiyun {ATH79_SOC_AR9132, "9132",
34*4882a593Smuzhiyun REV_ID_MAJOR_AR913X, AR913X_REV_ID_MINOR_AR9132},
35*4882a593Smuzhiyun {ATH79_SOC_AR9330, "9330", REV_ID_MAJOR_AR9330, 0},
36*4882a593Smuzhiyun {ATH79_SOC_AR9331, "9331", REV_ID_MAJOR_AR9331, 0},
37*4882a593Smuzhiyun {ATH79_SOC_AR9341, "9341", REV_ID_MAJOR_AR9341, 0},
38*4882a593Smuzhiyun {ATH79_SOC_AR9342, "9342", REV_ID_MAJOR_AR9342, 0},
39*4882a593Smuzhiyun {ATH79_SOC_AR9344, "9344", REV_ID_MAJOR_AR9344, 0},
40*4882a593Smuzhiyun {ATH79_SOC_QCA9533, "9533", REV_ID_MAJOR_QCA9533, 0},
41*4882a593Smuzhiyun {ATH79_SOC_QCA9533, "9533",
42*4882a593Smuzhiyun REV_ID_MAJOR_QCA9533_V2, 0},
43*4882a593Smuzhiyun {ATH79_SOC_QCA9556, "9556", REV_ID_MAJOR_QCA9556, 0},
44*4882a593Smuzhiyun {ATH79_SOC_QCA9558, "9558", REV_ID_MAJOR_QCA9558, 0},
45*4882a593Smuzhiyun {ATH79_SOC_TP9343, "9343", REV_ID_MAJOR_TP9343, 0},
46*4882a593Smuzhiyun {ATH79_SOC_QCA9561, "9561", REV_ID_MAJOR_QCA9561, 0},
47*4882a593Smuzhiyun };
48*4882a593Smuzhiyun
mach_cpu_init(void)49*4882a593Smuzhiyun int mach_cpu_init(void)
50*4882a593Smuzhiyun {
51*4882a593Smuzhiyun void __iomem *base;
52*4882a593Smuzhiyun enum ath79_soc_type soc = ATH79_SOC_UNKNOWN;
53*4882a593Smuzhiyun u32 id, major, minor = 0;
54*4882a593Smuzhiyun u32 rev = 0, ver = 1;
55*4882a593Smuzhiyun int i;
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun base = map_physmem(AR71XX_RESET_BASE, AR71XX_RESET_SIZE,
58*4882a593Smuzhiyun MAP_NOCACHE);
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun id = readl(base + AR71XX_RESET_REG_REV_ID);
61*4882a593Smuzhiyun major = id & REV_ID_MAJOR_MASK;
62*4882a593Smuzhiyun switch (major) {
63*4882a593Smuzhiyun case REV_ID_MAJOR_AR71XX:
64*4882a593Smuzhiyun case REV_ID_MAJOR_AR913X:
65*4882a593Smuzhiyun minor = id & AR71XX_REV_ID_MINOR_MASK;
66*4882a593Smuzhiyun rev = id >> AR71XX_REV_ID_REVISION_SHIFT;
67*4882a593Smuzhiyun rev &= AR71XX_REV_ID_REVISION_MASK;
68*4882a593Smuzhiyun break;
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun case REV_ID_MAJOR_QCA9533_V2:
71*4882a593Smuzhiyun ver = 2;
72*4882a593Smuzhiyun /* drop through */
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun case REV_ID_MAJOR_AR9341:
75*4882a593Smuzhiyun case REV_ID_MAJOR_AR9342:
76*4882a593Smuzhiyun case REV_ID_MAJOR_AR9344:
77*4882a593Smuzhiyun case REV_ID_MAJOR_QCA9533:
78*4882a593Smuzhiyun case REV_ID_MAJOR_QCA9556:
79*4882a593Smuzhiyun case REV_ID_MAJOR_QCA9558:
80*4882a593Smuzhiyun case REV_ID_MAJOR_TP9343:
81*4882a593Smuzhiyun case REV_ID_MAJOR_QCA9561:
82*4882a593Smuzhiyun rev = id & AR71XX_REV_ID_REVISION2_MASK;
83*4882a593Smuzhiyun break;
84*4882a593Smuzhiyun default:
85*4882a593Smuzhiyun rev = id & AR71XX_REV_ID_REVISION_MASK;
86*4882a593Smuzhiyun break;
87*4882a593Smuzhiyun }
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(desc); i++) {
90*4882a593Smuzhiyun if ((desc[i].major == major) &&
91*4882a593Smuzhiyun (desc[i].minor == minor)) {
92*4882a593Smuzhiyun soc = desc[i].soc;
93*4882a593Smuzhiyun break;
94*4882a593Smuzhiyun }
95*4882a593Smuzhiyun }
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun gd->arch.id = id;
98*4882a593Smuzhiyun gd->arch.soc = soc;
99*4882a593Smuzhiyun gd->arch.rev = rev;
100*4882a593Smuzhiyun gd->arch.ver = ver;
101*4882a593Smuzhiyun return 0;
102*4882a593Smuzhiyun }
103*4882a593Smuzhiyun
print_cpuinfo(void)104*4882a593Smuzhiyun int print_cpuinfo(void)
105*4882a593Smuzhiyun {
106*4882a593Smuzhiyun enum ath79_soc_type soc = ATH79_SOC_UNKNOWN;
107*4882a593Smuzhiyun const char *chip = "????";
108*4882a593Smuzhiyun u32 id, rev, ver;
109*4882a593Smuzhiyun int i;
110*4882a593Smuzhiyun
111*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(desc); i++) {
112*4882a593Smuzhiyun if (desc[i].soc == gd->arch.soc) {
113*4882a593Smuzhiyun chip = desc[i].chip;
114*4882a593Smuzhiyun soc = desc[i].soc;
115*4882a593Smuzhiyun break;
116*4882a593Smuzhiyun }
117*4882a593Smuzhiyun }
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun id = gd->arch.id;
120*4882a593Smuzhiyun rev = gd->arch.rev;
121*4882a593Smuzhiyun ver = gd->arch.ver;
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun switch (soc) {
124*4882a593Smuzhiyun case ATH79_SOC_QCA9533:
125*4882a593Smuzhiyun case ATH79_SOC_QCA9556:
126*4882a593Smuzhiyun case ATH79_SOC_QCA9558:
127*4882a593Smuzhiyun case ATH79_SOC_QCA9561:
128*4882a593Smuzhiyun printf("Qualcomm Atheros QCA%s ver %u rev %u\n", chip,
129*4882a593Smuzhiyun ver, rev);
130*4882a593Smuzhiyun break;
131*4882a593Smuzhiyun case ATH79_SOC_TP9343:
132*4882a593Smuzhiyun printf("Qualcomm Atheros TP%s rev %u\n", chip, rev);
133*4882a593Smuzhiyun break;
134*4882a593Smuzhiyun case ATH79_SOC_UNKNOWN:
135*4882a593Smuzhiyun printf("ATH79: unknown SoC, id:0x%08x", id);
136*4882a593Smuzhiyun break;
137*4882a593Smuzhiyun default:
138*4882a593Smuzhiyun printf("Atheros AR%s rev %u\n", chip, rev);
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun return 0;
142*4882a593Smuzhiyun }
143