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