zynqmp_common.c 6.69 KB
Newer Older
1
2
3
/*
 * Copyright (c) 2013-2016, ARM Limited and Contributors. All rights reserved.
 *
dp-arm's avatar
dp-arm committed
4
 * SPDX-License-Identifier: BSD-3-Clause
5
6
7
 */

#include <debug.h>
8
#include <generic_delay_timer.h>
9
10
#include <mmio.h>
#include <platform.h>
11
12
#include <stdbool.h>
#include <string.h>
13
14
#include <xlat_tables.h>
#include "../zynqmp_private.h"
15
#include "pm_api_sys.h"
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30

/*
 * Table of regions to map using the MMU.
 * This doesn't include TZRAM as the 'mem_layout' argument passed to
 * configure_mmu_elx() will give the available subset of that,
 */
const mmap_region_t plat_arm_mmap[] = {
	{ DEVICE0_BASE, DEVICE0_BASE, DEVICE0_SIZE, MT_DEVICE | MT_RW | MT_SECURE },
	{ DEVICE1_BASE, DEVICE1_BASE, DEVICE1_SIZE, MT_DEVICE | MT_RW | MT_SECURE },
	{ CRF_APB_BASE, CRF_APB_BASE, CRF_APB_SIZE, MT_DEVICE | MT_RW | MT_SECURE },
	{0}
};

static unsigned int zynqmp_get_silicon_ver(void)
{
31
	static unsigned int ver;
32

33
34
35
36
37
38
	if (!ver) {
		ver = mmio_read_32(ZYNQMP_CSU_BASEADDR +
				   ZYNQMP_CSU_VERSION_OFFSET);
		ver &= ZYNQMP_SILICON_VER_MASK;
		ver >>= ZYNQMP_SILICON_VER_SHIFT;
	}
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53

	return ver;
}

unsigned int zynqmp_get_uart_clk(void)
{
	unsigned int ver = zynqmp_get_silicon_ver();

	switch (ver) {
	case ZYNQMP_CSU_VERSION_VELOCE:
		return 48000;
	case ZYNQMP_CSU_VERSION_EP108:
		return 25000000;
	case ZYNQMP_CSU_VERSION_QEMU:
		return 133000000;
54
55
56
	default:
		/* Do nothing in default case */
		break;
57
58
59
60
61
62
63
64
	}

	return 100000000;
}

#if LOG_LEVEL >= LOG_LEVEL_NOTICE
static const struct {
	unsigned int id;
65
	unsigned int ver;
66
	char *name;
67
	bool evexists;
68
69
70
71
72
} zynqmp_devices[] = {
	{
		.id = 0x10,
		.name = "3EG",
	},
73
74
75
76
77
	{
		.id = 0x10,
		.ver = 0x2c,
		.name = "3CG",
	},
78
79
80
81
	{
		.id = 0x11,
		.name = "2EG",
	},
82
83
84
85
86
	{
		.id = 0x11,
		.ver = 0x2c,
		.name = "2CG",
	},
87
88
89
	{
		.id = 0x20,
		.name = "5EV",
90
		.evexists = true,
91
	},
92
93
94
95
	{
		.id = 0x20,
		.ver = 0x100,
		.name = "5EG",
96
		.evexists = true,
97
98
99
100
101
102
	},
	{
		.id = 0x20,
		.ver = 0x12c,
		.name = "5CG",
	},
103
104
105
	{
		.id = 0x21,
		.name = "4EV",
106
		.evexists = true,
107
	},
108
109
110
111
	{
		.id = 0x21,
		.ver = 0x100,
		.name = "4EG",
112
		.evexists = true,
113
114
115
116
117
118
	},
	{
		.id = 0x21,
		.ver = 0x12c,
		.name = "4CG",
	},
119
120
121
	{
		.id = 0x30,
		.name = "7EV",
122
		.evexists = true,
123
	},
124
125
126
127
	{
		.id = 0x30,
		.ver = 0x100,
		.name = "7EG",
128
		.evexists = true,
129
130
131
132
133
134
	},
	{
		.id = 0x30,
		.ver = 0x12c,
		.name = "7CG",
	},
135
136
137
138
	{
		.id = 0x38,
		.name = "9EG",
	},
139
140
141
142
143
	{
		.id = 0x38,
		.ver = 0x2c,
		.name = "9CG",
	},
144
145
146
147
	{
		.id = 0x39,
		.name = "6EG",
	},
148
149
150
151
152
	{
		.id = 0x39,
		.ver = 0x2c,
		.name = "6CG",
	},
153
154
155
156
	{
		.id = 0x40,
		.name = "11EG",
	},
157
158
159
160
161
	{ /* For testing purpose only */
		.id = 0x50,
		.ver = 0x2c,
		.name = "15CG",
	},
162
163
164
165
166
167
168
169
170
171
172
173
	{
		.id = 0x50,
		.name = "15EG",
	},
	{
		.id = 0x58,
		.name = "19EG",
	},
	{
		.id = 0x59,
		.name = "17EG",
	},
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
	{
		.id = 0x60,
		.name = "28DR",
	},
	{
		.id = 0x61,
		.name = "21DR",
	},
	{
		.id = 0x62,
		.name = "29DR",
	},
	{
		.id = 0x63,
		.name = "23DR",
	},
	{
		.id = 0x64,
		.name = "27DR",
	},
	{
		.id = 0x65,
		.name = "25DR",
	},
198
199
};

200
201
202
203
204
#define ZYNQMP_PL_STATUS_BIT	9
#define ZYNQMP_PL_STATUS_MASK	BIT(ZYNQMP_PL_STATUS_BIT)
#define ZYNQMP_CSU_VERSION_MASK	~(ZYNQMP_PL_STATUS_MASK)

static char *zynqmp_get_silicon_idcode_name(void)
205
{
206
207
208
	uint32_t id, ver, chipid[2];
	size_t i, j, len;
	const char *name = "EG/EV";
209

210
211
212
213
214
215
216
217
218
219
220
#ifdef IMAGE_BL32
	/*
	 * For BL32, get the chip id info directly by reading corresponding
	 * registers instead of making pm call. This has limitation
	 * that these registers should be configured to have access
	 * from APU which is default case.
	 */
	chipid[0] = mmio_read_32(ZYNQMP_CSU_BASEADDR + ZYNQMP_CSU_IDCODE_OFFSET);
	chipid[1] = mmio_read_32(EFUSE_BASEADDR + EFUSE_IPDISABLE_OFFSET);
#else
	if (pm_get_chipid(chipid) != PM_RET_SUCCESS)
221
		return "UNKN";
222
#endif
223

224
225
	id = chipid[0] & (ZYNQMP_CSU_IDCODE_DEVICE_CODE_MASK |
			  ZYNQMP_CSU_IDCODE_SVD_MASK);
226
	id >>= ZYNQMP_CSU_IDCODE_SVD_SHIFT;
227
	ver = chipid[1] >> ZYNQMP_EFUSE_IPDISABLE_SHIFT;
228

229
230
231
232
233
	for (i = 0; i < ARRAY_SIZE(zynqmp_devices); i++) {
		if (zynqmp_devices[i].id == id &&
		    zynqmp_devices[i].ver == (ver & ZYNQMP_CSU_VERSION_MASK))
			break;
	}
234

235
236
	if (i >= ARRAY_SIZE(zynqmp_devices))
		return "UNKN";
237

238
239
	if (!zynqmp_devices[i].evexists)
		return zynqmp_devices[i].name;
240

241
242
	if (ver & ZYNQMP_PL_STATUS_MASK)
		return zynqmp_devices[i].name;
243

244
245
246
247
	len = strlen(zynqmp_devices[i].name) - 2;
	for (j = 0; j < strlen(name); j++) {
		zynqmp_devices[i].name[len] = name[j];
		len++;
248
	}
249
250
251
	zynqmp_devices[i].name[len] = '\0';

	return zynqmp_devices[i].name;
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
}

static unsigned int zynqmp_get_rtl_ver(void)
{
	uint32_t ver;

	ver = mmio_read_32(ZYNQMP_CSU_BASEADDR + ZYNQMP_CSU_VERSION_OFFSET);
	ver &= ZYNQMP_RTL_VER_MASK;
	ver >>= ZYNQMP_RTL_VER_SHIFT;

	return ver;
}

static char *zynqmp_print_silicon_idcode(void)
{
	uint32_t id, maskid, tmp;

	id = mmio_read_32(ZYNQMP_CSU_BASEADDR + ZYNQMP_CSU_IDCODE_OFFSET);

	tmp = id;
	tmp &= ZYNQMP_CSU_IDCODE_XILINX_ID_MASK |
273
	       ZYNQMP_CSU_IDCODE_FAMILY_MASK;
274
	maskid = ZYNQMP_CSU_IDCODE_XILINX_ID << ZYNQMP_CSU_IDCODE_XILINX_ID_SHIFT |
275
		 ZYNQMP_CSU_IDCODE_FAMILY << ZYNQMP_CSU_IDCODE_FAMILY_SHIFT;
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
	if (tmp != maskid) {
		ERROR("Incorrect XILINX IDCODE 0x%x, maskid 0x%x\n", id, maskid);
		return "UNKN";
	}
	VERBOSE("Xilinx IDCODE 0x%x\n", id);
	return zynqmp_get_silicon_idcode_name();
}

static unsigned int zynqmp_get_ps_ver(void)
{
	uint32_t ver = mmio_read_32(ZYNQMP_CSU_BASEADDR + ZYNQMP_CSU_VERSION_OFFSET);

	ver &= ZYNQMP_PS_VER_MASK;
	ver >>= ZYNQMP_PS_VER_SHIFT;

	return ver + 1;
}

static void zynqmp_print_platform_name(void)
{
	unsigned int ver = zynqmp_get_silicon_ver();
	unsigned int rtl = zynqmp_get_rtl_ver();
	char *label = "Unknown";

	switch (ver) {
	case ZYNQMP_CSU_VERSION_VELOCE:
		label = "VELOCE";
		break;
	case ZYNQMP_CSU_VERSION_EP108:
		label = "EP108";
		break;
	case ZYNQMP_CSU_VERSION_QEMU:
		label = "QEMU";
		break;
	case ZYNQMP_CSU_VERSION_SILICON:
		label = "silicon";
		break;
313
314
315
	default:
		/* Do nothing in default case */
		break;
316
317
	}

318
	NOTICE("ATF running on XCZU%s/%s v%d/RTL%d.%d at 0x%x\n",
319
	       zynqmp_print_silicon_idcode(), label, zynqmp_get_ps_ver(),
320
	       (rtl & 0xf0) >> 4, rtl & 0xf, BL31_BASE);
321
322
323
324
325
}
#else
static inline void zynqmp_print_platform_name(void) { }
#endif

326
327
unsigned int zynqmp_get_bootmode(void)
{
328
	uint32_t r;
329
330
331
	unsigned int ret;

	ret = pm_mmio_read(CRL_APB_BOOT_MODE_USER, &r);
332

333
	if (ret != PM_RET_SUCCESS)
334
		r = mmio_read_32(CRL_APB_BOOT_MODE_USER);
335
336
337
338

	return r & CRL_APB_BOOT_MODE_MASK;
}

339
340
341
void zynqmp_config_setup(void)
{
	zynqmp_print_platform_name();
342
	generic_delay_timer_init();
343
344
}

345
unsigned int plat_get_syscnt_freq2(void)
346
{
347
	unsigned int ver = zynqmp_get_silicon_ver();
348

349
350
351
352
353
354
355
	switch (ver) {
	case ZYNQMP_CSU_VERSION_VELOCE:
		return 10000;
	case ZYNQMP_CSU_VERSION_EP108:
		return 4000000;
	case ZYNQMP_CSU_VERSION_QEMU:
		return 50000000;
356
357
358
	default:
		/* Do nothing in default case */
		break;
359
	}
360

361
	return mmio_read_32(IOU_SCNTRS_BASEFREQ);
362
}