blob: 91aa5cc89c387d90800c1146a77c6e1f196061ae [file] [log] [blame]
Mike Frysinger94bae5c2008-03-30 15:46:13 -04001/*
2 * U-boot - cpu.c CPU specific functions
3 *
4 * Copyright (c) 2005-2008 Analog Devices Inc.
5 *
6 * (C) Copyright 2000-2004
7 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
8 *
9 * Licensed under the GPL-2 or later.
10 */
11
12#include <common.h>
13#include <command.h>
Sonic Zhang3fbfdd12014-07-17 19:00:29 +080014#include <serial.h>
15#include <version.h>
16#include <i2c.h>
17
Mike Frysinger94bae5c2008-03-30 15:46:13 -040018#include <asm/blackfin.h>
19#include <asm/cplb.h>
Sonic Zhang3fbfdd12014-07-17 19:00:29 +080020#include <asm/clock.h>
Mike Frysinger94bae5c2008-03-30 15:46:13 -040021#include <asm/mach-common/bits/core.h>
Mike Frysingerc396f3b2008-10-11 21:44:00 -040022#include <asm/mach-common/bits/ebiu.h>
Mike Frysinger94bae5c2008-03-30 15:46:13 -040023#include <asm/mach-common/bits/trace.h>
24
25#include "cpu.h"
Mike Frysinger31c7c502011-05-30 13:47:38 -040026#include "initcode.h"
Martin Dorwigcb2c2862015-01-26 15:22:54 -070027#include "exports.h"
Mike Frysinger94bae5c2008-03-30 15:46:13 -040028
Mike Frysingerc396f3b2008-10-11 21:44:00 -040029ulong bfin_poweron_retx;
Sonic Zhang3fbfdd12014-07-17 19:00:29 +080030DECLARE_GLOBAL_DATA_PTR;
Mike Frysingerc396f3b2008-10-11 21:44:00 -040031
Sonic Zhang8a9561c2013-02-05 18:57:49 +080032#if defined(CONFIG_CORE1_RUN) && defined(COREB_L1_CODE_START)
33void bfin_core1_start(void)
34{
35#ifdef BF561_FAMILY
36 /* Enable core 1 */
37 bfin_write_SYSCR(bfin_read_SYSCR() & ~0x0020);
38#else
39 /* Enable core 1 */
40 bfin_write32(RCU0_SVECT1, COREB_L1_CODE_START);
41 bfin_write32(RCU0_CRCTL, 0);
42
43 bfin_write32(RCU0_CRCTL, 0x2);
44
45 /* Check if core 1 starts */
46 while (!(bfin_read32(RCU0_CRSTAT) & 0x2))
47 continue;
48
49 bfin_write32(RCU0_CRCTL, 0);
50
51 /* flag to notify cces core 1 application */
52 bfin_write32(SDU0_MSG_SET, (1 << 19));
53#endif
Sonic Zhang3fbfdd12014-07-17 19:00:29 +080054}
55#endif
56
57__attribute__((always_inline))
58static inline void serial_early_puts(const char *s)
59{
60#ifdef CONFIG_DEBUG_EARLY_SERIAL
61 serial_puts("Early: ");
62 serial_puts(s);
63#endif
64}
65
66static int global_board_data_init(void)
67{
68#ifndef CONFIG_SYS_GBL_DATA_ADDR
69# define CONFIG_SYS_GBL_DATA_ADDR 0
70#endif
71#ifndef CONFIG_SYS_BD_INFO_ADDR
72# define CONFIG_SYS_BD_INFO_ADDR 0
73#endif
74
75 bd_t *bd;
76
77 if (CONFIG_SYS_GBL_DATA_ADDR) {
78 gd = (gd_t *)(CONFIG_SYS_GBL_DATA_ADDR);
79 memset((void *)gd, 0, GENERATED_GBL_DATA_SIZE);
80 } else {
81 static gd_t _bfin_gd;
82 gd = &_bfin_gd;
83 }
84 if (CONFIG_SYS_BD_INFO_ADDR) {
85 bd = (bd_t *)(CONFIG_SYS_BD_INFO_ADDR);
86 memset(bd, 0, GENERATED_BD_INFO_SIZE);
87 } else {
88 static bd_t _bfin_bd;
89 bd = &_bfin_bd;
90 }
91
92 gd->bd = bd;
93
94 bd->bi_r_version = version_string;
95 bd->bi_cpu = __stringify(CONFIG_BFIN_CPU);
96 bd->bi_board_name = CONFIG_SYS_BOARD;
97 bd->bi_vco = get_vco();
98 bd->bi_cclk = get_cclk();
99 bd->bi_sclk = get_sclk();
100 bd->bi_memstart = CONFIG_SYS_SDRAM_BASE;
101 bd->bi_memsize = CONFIG_SYS_MAX_RAM_SIZE;
102
103 gd->ram_size = CONFIG_SYS_MAX_RAM_SIZE;
104
105 return 0;
106}
107
108static void display_global_data(void)
109{
110 bd_t *bd;
111
112#ifndef CONFIG_DEBUG_EARLY_SERIAL
113 return;
114#endif
115
116 bd = gd->bd;
117 printf(" gd: %p\n", gd);
118 printf(" |-flags: %lx\n", gd->flags);
119 printf(" |-board_type: %lx\n", gd->arch.board_type);
120 printf(" |-baudrate: %u\n", gd->baudrate);
121 printf(" |-have_console: %lx\n", gd->have_console);
122 printf(" |-ram_size: %lx\n", gd->ram_size);
123 printf(" |-env_addr: %lx\n", gd->env_addr);
124 printf(" |-env_valid: %lx\n", gd->env_valid);
Martin Dorwigcb2c2862015-01-26 15:22:54 -0700125 printf(" |-jt(%p): %p\n", gd->jt, gd->jt->get_version);
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800126 printf(" \\-bd: %p\n", gd->bd);
127 printf(" |-bi_boot_params: %lx\n", bd->bi_boot_params);
128 printf(" |-bi_memstart: %lx\n", bd->bi_memstart);
129 printf(" |-bi_memsize: %lx\n", bd->bi_memsize);
130 printf(" |-bi_flashstart: %lx\n", bd->bi_flashstart);
131 printf(" |-bi_flashsize: %lx\n", bd->bi_flashsize);
132 printf(" \\-bi_flashoffset: %lx\n", bd->bi_flashoffset);
133}
134
135#define CPLB_PAGE_SIZE (4 * 1024 * 1024)
136#define CPLB_PAGE_MASK (~(CPLB_PAGE_SIZE - 1))
137#if defined(__ADSPBF60x__)
138#define CPLB_EX_PAGE_SIZE (16 * 1024 * 1024)
139#define CPLB_EX_PAGE_MASK (~(CPLB_EX_PAGE_SIZE - 1))
140#else
141#define CPLB_EX_PAGE_SIZE CPLB_PAGE_SIZE
142#define CPLB_EX_PAGE_MASK CPLB_PAGE_MASK
143#endif
144void init_cplbtables(void)
145{
146 uint32_t *ICPLB_ADDR, *ICPLB_DATA;
147 uint32_t *DCPLB_ADDR, *DCPLB_DATA;
148 uint32_t extern_memory;
149 size_t i;
150
151 void icplb_add(uint32_t addr, uint32_t data)
152 {
153 bfin_write32(ICPLB_ADDR + i, addr);
154 bfin_write32(ICPLB_DATA + i, data);
155 }
156 void dcplb_add(uint32_t addr, uint32_t data)
157 {
158 bfin_write32(DCPLB_ADDR + i, addr);
159 bfin_write32(DCPLB_DATA + i, data);
160 }
161
162 /* populate a few common entries ... we'll let
163 * the memory map and cplb exception handler do
164 * the rest of the work.
165 */
166 i = 0;
167 ICPLB_ADDR = (uint32_t *)ICPLB_ADDR0;
168 ICPLB_DATA = (uint32_t *)ICPLB_DATA0;
169 DCPLB_ADDR = (uint32_t *)DCPLB_ADDR0;
170 DCPLB_DATA = (uint32_t *)DCPLB_DATA0;
171
172 icplb_add(0xFFA00000, L1_IMEMORY);
173 dcplb_add(0xFF800000, L1_DMEMORY);
174 ++i;
175#if defined(__ADSPBF60x__)
176 icplb_add(0x0, 0x0);
177 dcplb_add(CONFIG_SYS_FLASH_BASE, PAGE_SIZE_16MB | CPLB_DIRTY |
178 CPLB_SUPV_WR | CPLB_USER_WR | CPLB_USER_RD | CPLB_VALID);
179 ++i;
180#endif
181
182 if (CONFIG_MEM_SIZE) {
183 uint32_t mbase = CONFIG_SYS_MONITOR_BASE;
184 uint32_t mend = mbase + CONFIG_SYS_MONITOR_LEN - 1;
185 mbase &= CPLB_PAGE_MASK;
186 mend &= CPLB_PAGE_MASK;
187
188 icplb_add(mbase, SDRAM_IKERNEL);
189 dcplb_add(mbase, SDRAM_DKERNEL);
190 ++i;
191
192 /*
193 * If the monitor crosses a 4 meg boundary, we'll need
194 * to lock two entries for it. We assume it doesn't
195 * cross two 4 meg boundaries ...
196 */
197 if (mbase != mend) {
198 icplb_add(mend, SDRAM_IKERNEL);
199 dcplb_add(mend, SDRAM_DKERNEL);
200 ++i;
201 }
202 }
203
204#ifndef __ADSPBF60x__
205 icplb_add(0x20000000, SDRAM_INON_CHBL);
206 dcplb_add(0x20000000, SDRAM_EBIU);
207 ++i;
208#endif
209
210 /* Add entries for the rest of external RAM up to the bootrom */
211 extern_memory = 0;
212
213#ifdef CONFIG_DEBUG_NULL_PTR
214 icplb_add(extern_memory,
215 (SDRAM_IKERNEL & ~PAGE_SIZE_MASK) | PAGE_SIZE_1KB);
216 dcplb_add(extern_memory,
217 (SDRAM_DKERNEL & ~PAGE_SIZE_MASK) | PAGE_SIZE_1KB);
218 ++i;
219 icplb_add(extern_memory, SDRAM_IKERNEL);
220 dcplb_add(extern_memory, SDRAM_DKERNEL);
221 extern_memory += CPLB_PAGE_SIZE;
222 ++i;
223#endif
224
225 while (i < 16 && extern_memory <
226 (CONFIG_SYS_MONITOR_BASE & CPLB_EX_PAGE_MASK)) {
227 icplb_add(extern_memory, SDRAM_IGENERIC);
228 dcplb_add(extern_memory, SDRAM_DGENERIC);
229 extern_memory += CPLB_EX_PAGE_SIZE;
230 ++i;
231 }
232 while (i < 16) {
233 icplb_add(0, 0);
234 dcplb_add(0, 0);
235 ++i;
236 }
Sonic Zhang8a9561c2013-02-05 18:57:49 +0800237}
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800238
239int print_cpuinfo(void)
240{
241 char buf[32];
242
243 printf("CPU: ADSP %s (Detected Rev: 0.%d) (%s boot)\n",
244 gd->bd->bi_cpu,
245 bfin_revid(),
246 get_bfin_boot_mode(CONFIG_BFIN_BOOT_MODE));
247
248 printf("Clock: VCO: %s MHz, ", strmhz(buf, get_vco()));
249 printf("Core: %s MHz, ", strmhz(buf, get_cclk()));
250#if defined(__ADSPBF60x__)
251 printf("System0: %s MHz, ", strmhz(buf, get_sclk0()));
252 printf("System1: %s MHz, ", strmhz(buf, get_sclk1()));
253 printf("Dclk: %s MHz\n", strmhz(buf, get_dclk()));
254#else
255 printf("System: %s MHz\n", strmhz(buf, get_sclk()));
Sonic Zhang8a9561c2013-02-05 18:57:49 +0800256#endif
257
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800258 return 0;
259}
260
261int exception_init(void)
262{
263 bfin_write_EVT3(trap);
264 return 0;
265}
266
267int irq_init(void)
268{
269#ifdef SIC_IMASK0
270 bfin_write_SIC_IMASK0(0);
271 bfin_write_SIC_IMASK1(0);
272# ifdef SIC_IMASK2
273 bfin_write_SIC_IMASK2(0);
274# endif
275#elif defined(SICA_IMASK0)
276 bfin_write_SICA_IMASK0(0);
277 bfin_write_SICA_IMASK1(0);
278#elif defined(SIC_IMASK)
279 bfin_write_SIC_IMASK(0);
280#endif
281 /* Set up a dummy NMI handler if needed. */
282 if (CONFIG_BFIN_BOOT_MODE == BFIN_BOOT_BYPASS || ANOMALY_05000219)
283 bfin_write_EVT2(evt_nmi); /* NMI */
284 bfin_write_EVT5(evt_default); /* hardware error */
285 bfin_write_EVT6(evt_default); /* core timer */
286 bfin_write_EVT7(evt_default);
287 bfin_write_EVT8(evt_default);
288 bfin_write_EVT9(evt_default);
289 bfin_write_EVT10(evt_default);
290 bfin_write_EVT11(evt_default);
291 bfin_write_EVT12(evt_default);
292 bfin_write_EVT13(evt_default);
293 bfin_write_EVT14(evt_default);
294 bfin_write_EVT15(evt_default);
295 bfin_write_ILAT(0);
296 CSYNC();
297 /* enable hardware error irq */
298 irq_flags = 0x3f;
299 local_irq_enable();
300 return 0;
301}
302
Sonic Zhang8a9561c2013-02-05 18:57:49 +0800303__attribute__ ((__noreturn__))
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400304void cpu_init_f(ulong bootflag, ulong loaded_from_ldr)
305{
Mike Frysingera48e0ed2009-04-24 23:39:41 -0400306#ifndef CONFIG_BFIN_BOOTROM_USES_EVT1
307 /* Build a NOP slide over the LDR jump block. Whee! */
308 char nops[0xC];
309 serial_early_puts("NOP Slide\n");
310 memset(nops, 0x00, sizeof(nops));
Mike Frysinger685ec2c2009-11-03 06:11:31 -0500311 memcpy((void *)L1_INST_SRAM, nops, sizeof(nops));
Mike Frysingera48e0ed2009-04-24 23:39:41 -0400312#endif
313
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400314 if (!loaded_from_ldr) {
315 /* Relocate sections into L1 if the LDR didn't do it -- don't
316 * check length because the linker script does the size
317 * checking at build time.
318 */
319 serial_early_puts("L1 Relocate\n");
Mike Frysinger685ec2c2009-11-03 06:11:31 -0500320 extern char _stext_l1[], _text_l1_lma[], _text_l1_len[];
321 memcpy(&_stext_l1, &_text_l1_lma, (unsigned long)_text_l1_len);
322 extern char _sdata_l1[], _data_l1_lma[], _data_l1_len[];
323 memcpy(&_sdata_l1, &_data_l1_lma, (unsigned long)_data_l1_len);
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400324 }
Mike Frysinger31c7c502011-05-30 13:47:38 -0400325
326 /*
327 * Make sure our async settings are committed. Some bootroms
328 * (like the BF537) will reset some registers on us after it
329 * has finished loading the LDR. Or if we're booting over
330 * JTAG, the initcode never got a chance to run. Or if we
331 * aren't booting from parallel flash, the initcode skipped
332 * this step completely.
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400333 */
Mike Frysinger31c7c502011-05-30 13:47:38 -0400334 program_async_controller(NULL);
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400335
Mike Frysingerc396f3b2008-10-11 21:44:00 -0400336 /* Save RETX so we can pass it while booting Linux */
337 bfin_poweron_retx = bootflag;
338
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400339#ifdef CONFIG_DEBUG_DUMP
340 /* Turn on hardware trace buffer */
341 bfin_write_TBUFCTL(TBUFPWR | TBUFEN);
342#endif
343
344#ifndef CONFIG_PANIC_HANG
345 /* Reset upon a double exception rather than just hanging.
346 * Do not do bfin_read on SWRST as that will reset status bits.
347 */
Sonic Zhanga99f03e2012-08-16 11:56:14 +0800348# ifdef SWRST
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400349 bfin_write_SWRST(DOUBLE_FAULT);
Sonic Zhanga99f03e2012-08-16 11:56:14 +0800350# endif
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400351#endif
352
Sonic Zhang8a9561c2013-02-05 18:57:49 +0800353#if defined(CONFIG_CORE1_RUN) && defined(COREB_L1_CODE_START)
354 bfin_core1_start();
355#endif
356
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800357 serial_early_puts("Init global data\n");
358 global_board_data_init();
359
360 board_init_f(0);
Masahiro Yamada83ee17e2013-10-21 10:14:05 +0900361
362 /* should not be reached */
363 while (1);
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400364}
365
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800366int arch_cpu_init(void)
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400367{
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800368 serial_early_puts("Init CPLB tables\n");
369 init_cplbtables();
370
371 serial_early_puts("Exceptions setup\n");
372 exception_init();
373
374#ifndef CONFIG_ICACHE_OFF
375 serial_early_puts("Turn on ICACHE\n");
376 icache_enable();
377#endif
378#ifndef CONFIG_DCACHE_OFF
379 serial_early_puts("Turn on DCACHE\n");
380 dcache_enable();
381#endif
382
383#ifdef DEBUG
384 if (GENERATED_GBL_DATA_SIZE < sizeof(*gd))
385 hang();
386#endif
387
388 /* Initialize */
389 serial_early_puts("IRQ init\n");
390 irq_init();
391
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400392 return 0;
393}
394
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800395int arch_misc_init(void)
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400396{
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800397#if defined(CONFIG_SYS_I2C)
398 i2c_reloc_fixup();
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400399#endif
Sonic Zhang3fbfdd12014-07-17 19:00:29 +0800400
401 display_global_data();
402
403 if (CONFIG_MEM_SIZE && bfin_os_log_check()) {
404 puts("\nLog buffer from operating system:\n");
405 bfin_os_log_dump();
406 puts("\n");
407 }
408
409 return 0;
410}
411
412int interrupt_init(void)
413{
Mike Frysinger94bae5c2008-03-30 15:46:13 -0400414 return 0;
415}