soc_info.c 13 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/*
 * Copyright (C) 2012  Henrik Nordstrom <henrik@henriknordstrom.net>
 * Copyright (C) 2015  Siarhei Siamashka <siarhei.siamashka@gmail.com>
 * Copyright (C) 2016  Bernhard Nortmann <bernhard.nortmann@web.de>
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

/**********************************************************************
 * SoC information and retrieval of soc_sram_info
 **********************************************************************/
#include "soc_info.h"

#include <stdio.h>
26
#include <string.h>
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96

/*
 * The FEL code from BROM in A10/A13/A20 sets up two stacks for itself. One
 * at 0x2000 (and growing down) for the IRQ handler. And another one at 0x7000
 * (and also growing down) for the regular code. In order to use the whole
 * 32 KiB in the A1/A2 sections of SRAM, we need to temporarily move these
 * stacks elsewhere. And the addresses 0x7D00-0x7FFF contain something
 * important too (overwriting them kills FEL). On A10/A13/A20 we can use
 * the SRAM sections A3/A4 (0x8000-0xBFFF) for this purpose.
 */
sram_swap_buffers a10_a13_a20_sram_swap_buffers[] = {
	/* 0x1C00-0x1FFF (IRQ stack) */
	{ .buf1 = 0x1C00, .buf2 = 0xA400, .size = 0x0400 },
	/* 0x5C00-0x6FFF (Stack) */
	{ .buf1 = 0x5C00, .buf2 = 0xA800, .size = 0x1400 },
	/* 0x7C00-0x7FFF (Something important) */
	{ .buf1 = 0x7C00, .buf2 = 0xBC00, .size = 0x0400 },
	{ .size = 0 }  /* End of the table */
};

/*
 * A31 is very similar to A10/A13/A20, except that it has no SRAM at 0x8000.
 * So we use the SRAM section B at 0x20000-0x2FFFF instead. In the FEL mode,
 * the MMU translation table is allocated by the BROM at 0x20000. But we can
 * also safely use it as the backup storage because the MMU is temporarily
 * disabled during the time of the SPL execution.
 */
sram_swap_buffers a31_sram_swap_buffers[] = {
	{ .buf1 = 0x1800, .buf2 = 0x20000, .size = 0x800 },
	{ .buf1 = 0x5C00, .buf2 = 0x20800, .size = 0x8000 - 0x5C00 },
	{ .size = 0 }  /* End of the table */
};

/*
 * A64 has 32KiB of SRAM A at 0x10000 and a large SRAM C at 0x18000. SRAM A
 * and SRAM C reside in the address space back-to-back without any gaps, thus
 * representing a singe large contiguous area. Everything is the same as on
 * A10/A13/A20, but just shifted by 0x10000.
 */
sram_swap_buffers a64_sram_swap_buffers[] = {
	/* 0x11C00-0x11FFF (IRQ stack) */
	{ .buf1 = 0x11C00, .buf2 = 0x1A400, .size = 0x0400 },
	/* 0x15C00-0x16FFF (Stack) */
	{ .buf1 = 0x15C00, .buf2 = 0x1A800, .size = 0x1400 },
	/* 0x17C00-0x17FFF (Something important) */
	{ .buf1 = 0x17C00, .buf2 = 0x1BC00, .size = 0x0400 },
	{ .size = 0 }  /* End of the table */
};

/*
 * Use the SRAM section at 0x44000 as the backup storage. This is the memory,
 * which is normally shared with the OpenRISC core (should we do an extra check
 * to ensure that this core is powered off and can't interfere?).
 */
sram_swap_buffers ar100_abusing_sram_swap_buffers[] = {
	{ .buf1 = 0x1800, .buf2 = 0x44000, .size = 0x800 },
	{ .buf1 = 0x5C00, .buf2 = 0x44800, .size = 0x8000 - 0x5C00 },
	{ .size = 0 }  /* End of the table */
};

/*
 * A80 has 40KiB SRAM A1 at 0x10000 where the SPL has to be loaded to. The
 * secure SRAM B at 0x20000 is used as backup area for FEL stacks and data.
 */
sram_swap_buffers a80_sram_swap_buffers[] = {
	{ .buf1 = 0x11800, .buf2 = 0x20000, .size = 0x800 },
	{ .buf1 = 0x15400, .buf2 = 0x20800, .size = 0x18000 - 0x15400 },
	{ .size = 0 }  /* End of the table */
};

Icenowy Zheng's avatar
Icenowy Zheng committed
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
/*
 * H6 has 32KiB of SRAM A at 0x20000 and a large SRAM C at 0x28000. SRAM A
 * and SRAM C reside in the address space back-to-back without any gaps, thus
 * representing a singe large contiguous area. Everything is the same as on
 * A10/A13/A20, but just shifted by 0x20000.
 */
sram_swap_buffers h6_sram_swap_buffers[] = {
	/* 0x21C00-0x21FFF (IRQ stack) */
	{ .buf1 = 0x21C00, .buf2 = 0x2A400, .size = 0x0400 },
	/* 0x25C00-0x26FFF (Stack) */
	{ .buf1 = 0x25C00, .buf2 = 0x2A800, .size = 0x1400 },
	/* 0x27C00-0x27FFF (Something important) */
	{ .buf1 = 0x27C00, .buf2 = 0x2BC00, .size = 0x0400 },
	{ .size = 0 }  /* End of the table */
};

113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
/*
 * V831 has 96KiB SRAM A1 at 0x20000 where the SPL has to be loaded to.
 * SRAM C is continuous with SRAM A1, and both SRAMs are tried to be used
 * by BROM. Memory space is allocated both from the start of SRAM A1 and
 * the end of SRAM C.
 * The start of SRAM C is in between these areas, and can serve as backup
 * of IRQ stack, which is inside the first 32KiB of SRAM A1. Other areas
 * that are critical on older SoCs seem to be already in SRAM C, which
 * we do not need to preserve.
 */
sram_swap_buffers v831_sram_swap_buffers[] = {
	{ .buf1 = 0x21000, .buf2 = 0x38000, .size = 0x1000 },
	{ .size = 0 }  /* End of the table */
};

Jernej Skrabec's avatar
Jernej Skrabec committed
128
129
/* H616 situation is the same as V831 one, except it has 32 KiB of SRAM A1. */
sram_swap_buffers h616_sram_swap_buffers[] = {
130
	{ .buf1 = 0x21000, .buf2 = 0x52a00, .size = 0x1000 },
Jernej Skrabec's avatar
Jernej Skrabec committed
131
132
133
	{ .size = 0 }  /* End of the table */
};

134
135
136
137
138
139
140
141
142
143
const watchdog_info wd_a10_compat = {
	.reg_mode = 0x01C20C94,
	.reg_mode_value = 3,
};

const watchdog_info wd_h3_compat = {
	.reg_mode = 0x01C20Cb8,
	.reg_mode_value = 1,
};

Andre Przywara's avatar
Andre Przywara committed
144
145
146
147
148
149
150
151
152
153
const watchdog_info wd_a80 = {
	.reg_mode = 0x06000CB8,
	.reg_mode_value = 1,
};

const watchdog_info wd_h6_compat = {
	.reg_mode = 0x030090b8,
	.reg_mode_value = 1,
};

154
155
156
soc_info_t soc_info_table[] = {
	{
		.soc_id       = 0x1623, /* Allwinner A10 */
157
		.name         = "A10",
158
159
160
		.scratch_addr = 0x1000,
		.thunk_addr   = 0xA200, .thunk_size = 0x200,
		.swap_buffers = a10_a13_a20_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
161
		.sram_size    = 48 * 1024,
162
		.needs_l2en   = true,
163
		.sid_base     = 0x01C23800,
164
		.watchdog     = &wd_a10_compat,
165
	},{
166
		.soc_id       = 0x1625, /* Allwinner A10s, A13, R8 */
167
		.name         = "A13",
168
169
170
		.scratch_addr = 0x1000,
		.thunk_addr   = 0xA200, .thunk_size = 0x200,
		.swap_buffers = a10_a13_a20_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
171
		.sram_size    = 48 * 1024,
172
		.needs_l2en   = true,
173
		.sid_base     = 0x01C23800,
Andre Przywara's avatar
Andre Przywara committed
174
		.watchdog     = &wd_a10_compat,
175
176
	},{
		.soc_id       = 0x1651, /* Allwinner A20 */
177
		.name         = "A20",
178
179
180
		.scratch_addr = 0x1000,
		.thunk_addr   = 0xA200, .thunk_size = 0x200,
		.swap_buffers = a10_a13_a20_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
181
		.sram_size    = 48 * 1024,
182
		.sid_base     = 0x01C23800,
183
		.watchdog     = &wd_a10_compat,
184
185
	},{
		.soc_id       = 0x1650, /* Allwinner A23 */
186
		.name         = "A23",
187
188
189
		.scratch_addr = 0x1000,
		.thunk_addr   = 0x46E00, .thunk_size = 0x200,
		.swap_buffers = ar100_abusing_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
190
		.sram_size    = 64 * 1024,
191
		.sid_base     = 0x01C23800,
Andre Przywara's avatar
Andre Przywara committed
192
		.watchdog     = &wd_h3_compat,
193
194
	},{
		.soc_id       = 0x1633, /* Allwinner A31 */
195
		.name         = "A31",
196
197
198
		.scratch_addr = 0x1000,
		.thunk_addr   = 0x22E00, .thunk_size = 0x200,
		.swap_buffers = a31_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
199
		.sram_size    = 32 * 1024,
Andre Przywara's avatar
Andre Przywara committed
200
		.watchdog     = &wd_h3_compat,
201
202
	},{
		.soc_id       = 0x1667, /* Allwinner A33, R16 */
203
		.name         = "A33",
204
205
206
		.scratch_addr = 0x1000,
		.thunk_addr   = 0x46E00, .thunk_size = 0x200,
		.swap_buffers = ar100_abusing_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
207
		.sram_size    = 32 * 1024,
208
		.sid_base     = 0x01C23800,
Andre Przywara's avatar
Andre Przywara committed
209
		.watchdog     = &wd_h3_compat,
210
211
	},{
		.soc_id       = 0x1689, /* Allwinner A64 */
212
		.name         = "A64",
213
214
215
216
		.spl_addr     = 0x10000,
		.scratch_addr = 0x11000,
		.thunk_addr   = 0x1A200, .thunk_size = 0x200,
		.swap_buffers = a64_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
217
		.sram_size    = 192 * 1024,
218
219
		.sid_base     = 0x01C14000,
		.sid_offset   = 0x200,
220
		.rvbar_reg    = 0x017000A0,
221
222
		/* Check L.NOP in the OpenRISC reset vector */
		.needs_smc_workaround_if_zero_word_at_addr = 0x40004,
Andre Przywara's avatar
Andre Przywara committed
223
		.watchdog     = &wd_h3_compat,
224
225
	},{
		.soc_id       = 0x1639, /* Allwinner A80 */
226
		.name         = "A80",
227
228
229
230
		.spl_addr     = 0x10000,
		.scratch_addr = 0x11000,
		.thunk_addr   = 0x23400, .thunk_size = 0x200,
		.swap_buffers = a80_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
231
		.sram_size    = 40 * 1024,
232
233
		.sid_base     = 0X01C0E000,
		.sid_offset   = 0x200,
Andre Przywara's avatar
Andre Przywara committed
234
		.watchdog     = &wd_a80,
235
236
	},{
		.soc_id       = 0x1673, /* Allwinner A83T */
237
		.name         = "A83T",
238
		.scratch_addr = 0x1000,
Jack Mitchell's avatar
Jack Mitchell committed
239
		.mmu_tt_addr  = 0x44000,
240
241
		.thunk_addr   = 0x46E00, .thunk_size = 0x200,
		.swap_buffers = ar100_abusing_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
242
		.sram_size    = 32 * 1024,
243
244
		.sid_base     = 0x01C14000,
		.sid_offset   = 0x200,
Andre Przywara's avatar
Andre Przywara committed
245
		.watchdog     = &wd_h3_compat,
246
247
	},{
		.soc_id       = 0x1680, /* Allwinner H3, H2+ */
248
		.name         = "H3",
249
250
251
252
		.scratch_addr = 0x1000,
		.mmu_tt_addr  = 0x8000,
		.thunk_addr   = 0xA200, .thunk_size = 0x200,
		.swap_buffers = a10_a13_a20_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
253
		.sram_size    = 108 * 1024,
254
255
		.sid_base     = 0x01C14000,
		.sid_offset   = 0x200,
Icenowy Zheng's avatar
Icenowy Zheng committed
256
		.sid_fix      = true,
257
258
		/* Check L.NOP in the OpenRISC reset vector */
		.needs_smc_workaround_if_zero_word_at_addr = 0x40004,
259
		.watchdog     = &wd_h3_compat,
260
261
262
263
264
265
266
	},{
		.soc_id       = 0x1681, /* Allwinner V3s */
		.name         = "V3s",
		.scratch_addr = 0x1000,
		.mmu_tt_addr  = 0x8000,
		.thunk_addr   = 0xA200, .thunk_size = 0x200,
		.swap_buffers = a10_a13_a20_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
267
		.sram_size    = 60 * 1024,
268
		.sid_base     = 0x01C23800,
Andre Przywara's avatar
Andre Przywara committed
269
		.watchdog     = &wd_h3_compat,
270
271
	},{
		.soc_id       = 0x1718, /* Allwinner H5 */
272
		.name         = "H5",
273
274
275
276
		.spl_addr     = 0x10000,
		.scratch_addr = 0x11000,
		.thunk_addr   = 0x1A200, .thunk_size = 0x200,
		.swap_buffers = a64_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
277
		.sram_size    = 144 * 1024,
278
279
		.sid_base     = 0x01C14000,
		.sid_offset   = 0x200,
280
		.rvbar_reg    = 0x017000A0,
281
282
		/* Check L.NOP in the OpenRISC reset vector */
		.needs_smc_workaround_if_zero_word_at_addr = 0x40004,
283
		.watchdog     = &wd_h3_compat,
284
285
	},{
		.soc_id       = 0x1701, /* Allwinner R40 */
286
		.name         = "R40",
287
288
289
		.scratch_addr = 0x1000,
		.thunk_addr   = 0xA200, .thunk_size = 0x200,
		.swap_buffers = a10_a13_a20_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
290
		.sram_size    = 48 * 1024,
291
292
		.sid_base     = 0x01C1B000,
		.sid_offset   = 0x200,
Andre Przywara's avatar
Andre Przywara committed
293
		.watchdog     = &wd_a10_compat,
Icenowy Zheng's avatar
Icenowy Zheng committed
294
295
296
297
298
299
300
	},{
		.soc_id       = 0x1728, /* Allwinner H6 */
		.name         = "H6",
		.spl_addr     = 0x20000,
		.scratch_addr = 0x21000,
		.thunk_addr   = 0x2A200, .thunk_size = 0x200,
		.swap_buffers = h6_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
301
		.sram_size    = 152 * 1024,
Icenowy Zheng's avatar
Icenowy Zheng committed
302
303
304
305
306
		.sid_base     = 0x03006000,
		.sid_offset   = 0x200,
		.rvbar_reg    = 0x09010040,
		/* Check L.NOP in the OpenRISC reset vector */
		.needs_smc_workaround_if_zero_word_at_addr = 0x100004,
Andre Przywara's avatar
Andre Przywara committed
307
		.watchdog     = &wd_h6_compat,
308
309
310
311
312
313
314
	},{
		.soc_id       = 0x1817, /* Allwinner V831 */
		.name         = "V831",
		.spl_addr     = 0x20000,
		.scratch_addr = 0x21000,
		.thunk_addr   = 0x2A200, .thunk_size = 0x200,
		.swap_buffers = v831_sram_swap_buffers,
Andre Przywara's avatar
Andre Przywara committed
315
		.sram_size    = 228 * 1024,
316
317
		.sid_base     = 0x03006000,
		.sid_offset   = 0x200,
Andre Przywara's avatar
Andre Przywara committed
318
		.watchdog     = &wd_h6_compat,
Jernej Skrabec's avatar
Jernej Skrabec committed
319
320
321
322
323
	},{
		.soc_id       = 0x1823, /* Allwinner H616 */
		.name         = "H616",
		.spl_addr     = 0x20000,
		.scratch_addr = 0x21000,
324
		.thunk_addr   = 0x53a00, .thunk_size = 0x200,
Jernej Skrabec's avatar
Jernej Skrabec committed
325
		.swap_buffers = h616_sram_swap_buffers,
326
		.sram_size    = 207 * 1024,
Jernej Skrabec's avatar
Jernej Skrabec committed
327
328
329
		.sid_base     = 0x03006000,
		.sid_offset   = 0x200,
		.rvbar_reg    = 0x09010040,
Andre Przywara's avatar
Andre Przywara committed
330
		.watchdog     = &wd_h6_compat,
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
	},{
		.swap_buffers = NULL /* End of the table */
	}
};

/*
 * This generic record assumes BROM with similar properties to A10/A13/A20/A31,
 * but no extra SRAM sections beyond 0x8000. It also assumes that the IRQ
 * handler stack usage never exceeds 0x400 bytes.
 *
 * The users may or may not hope that the 0x7000-0x8000 area is also unused
 * by the BROM and re-purpose it for the SPL stack.
 *
 * The size limit for the ".text + .data" sections is ~21 KiB.
 */
sram_swap_buffers generic_sram_swap_buffers[] = {
	{ .buf1 = 0x1C00, .buf2 = 0x5800, .size = 0x400 },
	{ .size = 0 }  /* End of the table */
};

soc_info_t generic_soc_info = {
	.scratch_addr = 0x1000,
	.thunk_addr   = 0x5680, .thunk_size = 0x180,
	.swap_buffers = generic_sram_swap_buffers,
};

/* functions to retrieve SoC information */

soc_info_t *get_soc_info_from_id(uint32_t soc_id)
{
361
	soc_info_t *soc, *result = NULL;
362

363
364
365
	for (soc = soc_info_table; soc->swap_buffers; soc++)
		if (soc->soc_id == soc_id) {
			result = soc;
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
			break;
		}

	if (!result) {
		printf("Warning: no 'soc_sram_info' data for your SoC (id=%04X)\n",
		       soc_id);
		result = &generic_soc_info;
	}
	return result;
}

soc_info_t *get_soc_info_from_version(struct aw_fel_version *buf)
{
	return get_soc_info_from_id(buf->soc_id);
}
381
382
383

void get_soc_name_from_id(soc_name_t buffer, uint32_t soc_id)
{
384
385
386
387
	soc_info_t *soc;
	for (soc = soc_info_table; soc->swap_buffers; soc++)
		if (soc->soc_id == soc_id && soc->name != NULL) {
			strncpy(buffer, soc->name, sizeof(soc_name_t) - 1);
388
389
390
391
392
393
			return;
		}

	/* unknown SoC (or name string missing), use the hexadecimal ID */
	snprintf(buffer, sizeof(soc_name_t) - 1, "0x%04X", soc_id);
}