| /* |
| * U-Boot - cpu.c CPU specific functions |
| * |
| * Copyright (c) 2005-2008 Analog Devices Inc. |
| * |
| * (C) Copyright 2000-2004 |
| * Wolfgang Denk, DENX Software Engineering, wd@denx.de. |
| * |
| * Licensed under the GPL-2 or later. |
| */ |
| |
| #include <common.h> |
| #include <command.h> |
| #include <serial.h> |
| #include <version.h> |
| #include <i2c.h> |
| |
| #include <asm/blackfin.h> |
| #include <asm/cplb.h> |
| #include <asm/clock.h> |
| #include <asm/mach-common/bits/core.h> |
| #include <asm/mach-common/bits/ebiu.h> |
| #include <asm/mach-common/bits/trace.h> |
| |
| #include "cpu.h" |
| #include "initcode.h" |
| #include "exports.h" |
| |
| ulong bfin_poweron_retx; |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| #if defined(CONFIG_CORE1_RUN) && defined(COREB_L1_CODE_START) |
| void bfin_core1_start(void) |
| { |
| #ifdef BF561_FAMILY |
| /* Enable core 1 */ |
| bfin_write_SYSCR(bfin_read_SYSCR() & ~0x0020); |
| #else |
| /* Enable core 1 */ |
| bfin_write32(RCU0_SVECT1, COREB_L1_CODE_START); |
| bfin_write32(RCU0_CRCTL, 0); |
| |
| bfin_write32(RCU0_CRCTL, 0x2); |
| |
| /* Check if core 1 starts */ |
| while (!(bfin_read32(RCU0_CRSTAT) & 0x2)) |
| continue; |
| |
| bfin_write32(RCU0_CRCTL, 0); |
| |
| /* flag to notify cces core 1 application */ |
| bfin_write32(SDU0_MSG_SET, (1 << 19)); |
| #endif |
| } |
| #endif |
| |
| __attribute__((always_inline)) |
| static inline void serial_early_puts(const char *s) |
| { |
| #ifdef CONFIG_DEBUG_EARLY_SERIAL |
| serial_puts("Early: "); |
| serial_puts(s); |
| #endif |
| } |
| |
| static int global_board_data_init(void) |
| { |
| #ifndef CONFIG_SYS_GBL_DATA_ADDR |
| # define CONFIG_SYS_GBL_DATA_ADDR 0 |
| #endif |
| #ifndef CONFIG_SYS_BD_INFO_ADDR |
| # define CONFIG_SYS_BD_INFO_ADDR 0 |
| #endif |
| |
| bd_t *bd; |
| |
| if (CONFIG_SYS_GBL_DATA_ADDR) { |
| gd = (gd_t *)(CONFIG_SYS_GBL_DATA_ADDR); |
| memset((void *)gd, 0, GENERATED_GBL_DATA_SIZE); |
| } else { |
| static gd_t _bfin_gd; |
| gd = &_bfin_gd; |
| } |
| if (CONFIG_SYS_BD_INFO_ADDR) { |
| bd = (bd_t *)(CONFIG_SYS_BD_INFO_ADDR); |
| memset(bd, 0, GENERATED_BD_INFO_SIZE); |
| } else { |
| static bd_t _bfin_bd; |
| bd = &_bfin_bd; |
| } |
| |
| gd->bd = bd; |
| |
| bd->bi_r_version = version_string; |
| bd->bi_cpu = __stringify(CONFIG_BFIN_CPU); |
| bd->bi_board_name = CONFIG_SYS_BOARD; |
| bd->bi_vco = get_vco(); |
| bd->bi_cclk = get_cclk(); |
| bd->bi_sclk = get_sclk(); |
| bd->bi_memstart = CONFIG_SYS_SDRAM_BASE; |
| bd->bi_memsize = CONFIG_SYS_MAX_RAM_SIZE; |
| |
| gd->ram_size = CONFIG_SYS_MAX_RAM_SIZE; |
| |
| return 0; |
| } |
| |
| static void display_global_data(void) |
| { |
| bd_t *bd; |
| |
| #ifndef CONFIG_DEBUG_EARLY_SERIAL |
| return; |
| #endif |
| |
| bd = gd->bd; |
| printf(" gd: %p\n", gd); |
| printf(" |-flags: %lx\n", gd->flags); |
| printf(" |-board_type: %lx\n", gd->arch.board_type); |
| printf(" |-baudrate: %u\n", gd->baudrate); |
| printf(" |-have_console: %lx\n", gd->have_console); |
| printf(" |-ram_size: %lx\n", gd->ram_size); |
| printf(" |-env_addr: %lx\n", gd->env_addr); |
| printf(" |-env_valid: %lx\n", gd->env_valid); |
| printf(" |-jt(%p): %p\n", gd->jt, gd->jt->get_version); |
| printf(" \\-bd: %p\n", gd->bd); |
| printf(" |-bi_boot_params: %lx\n", bd->bi_boot_params); |
| printf(" |-bi_memstart: %lx\n", bd->bi_memstart); |
| printf(" |-bi_memsize: %lx\n", bd->bi_memsize); |
| printf(" |-bi_flashstart: %lx\n", bd->bi_flashstart); |
| printf(" |-bi_flashsize: %lx\n", bd->bi_flashsize); |
| printf(" \\-bi_flashoffset: %lx\n", bd->bi_flashoffset); |
| } |
| |
| #define CPLB_PAGE_SIZE (4 * 1024 * 1024) |
| #define CPLB_PAGE_MASK (~(CPLB_PAGE_SIZE - 1)) |
| #if defined(__ADSPBF60x__) |
| #define CPLB_EX_PAGE_SIZE (16 * 1024 * 1024) |
| #define CPLB_EX_PAGE_MASK (~(CPLB_EX_PAGE_SIZE - 1)) |
| #else |
| #define CPLB_EX_PAGE_SIZE CPLB_PAGE_SIZE |
| #define CPLB_EX_PAGE_MASK CPLB_PAGE_MASK |
| #endif |
| void init_cplbtables(void) |
| { |
| uint32_t *ICPLB_ADDR, *ICPLB_DATA; |
| uint32_t *DCPLB_ADDR, *DCPLB_DATA; |
| uint32_t extern_memory; |
| size_t i; |
| |
| void icplb_add(uint32_t addr, uint32_t data) |
| { |
| bfin_write32(ICPLB_ADDR + i, addr); |
| bfin_write32(ICPLB_DATA + i, data); |
| } |
| void dcplb_add(uint32_t addr, uint32_t data) |
| { |
| bfin_write32(DCPLB_ADDR + i, addr); |
| bfin_write32(DCPLB_DATA + i, data); |
| } |
| |
| /* populate a few common entries ... we'll let |
| * the memory map and cplb exception handler do |
| * the rest of the work. |
| */ |
| i = 0; |
| ICPLB_ADDR = (uint32_t *)ICPLB_ADDR0; |
| ICPLB_DATA = (uint32_t *)ICPLB_DATA0; |
| DCPLB_ADDR = (uint32_t *)DCPLB_ADDR0; |
| DCPLB_DATA = (uint32_t *)DCPLB_DATA0; |
| |
| icplb_add(0xFFA00000, L1_IMEMORY); |
| dcplb_add(0xFF800000, L1_DMEMORY); |
| ++i; |
| #if defined(__ADSPBF60x__) |
| icplb_add(0x0, 0x0); |
| dcplb_add(CONFIG_SYS_FLASH_BASE, PAGE_SIZE_16MB | CPLB_DIRTY | |
| CPLB_SUPV_WR | CPLB_USER_WR | CPLB_USER_RD | CPLB_VALID); |
| ++i; |
| #endif |
| |
| if (CONFIG_MEM_SIZE) { |
| uint32_t mbase = CONFIG_SYS_MONITOR_BASE; |
| uint32_t mend = mbase + CONFIG_SYS_MONITOR_LEN - 1; |
| mbase &= CPLB_PAGE_MASK; |
| mend &= CPLB_PAGE_MASK; |
| |
| icplb_add(mbase, SDRAM_IKERNEL); |
| dcplb_add(mbase, SDRAM_DKERNEL); |
| ++i; |
| |
| /* |
| * If the monitor crosses a 4 meg boundary, we'll need |
| * to lock two entries for it. We assume it doesn't |
| * cross two 4 meg boundaries ... |
| */ |
| if (mbase != mend) { |
| icplb_add(mend, SDRAM_IKERNEL); |
| dcplb_add(mend, SDRAM_DKERNEL); |
| ++i; |
| } |
| } |
| |
| #ifndef __ADSPBF60x__ |
| icplb_add(0x20000000, SDRAM_INON_CHBL); |
| dcplb_add(0x20000000, SDRAM_EBIU); |
| ++i; |
| #endif |
| |
| /* Add entries for the rest of external RAM up to the bootrom */ |
| extern_memory = 0; |
| |
| #ifdef CONFIG_DEBUG_NULL_PTR |
| icplb_add(extern_memory, |
| (SDRAM_IKERNEL & ~PAGE_SIZE_MASK) | PAGE_SIZE_1KB); |
| dcplb_add(extern_memory, |
| (SDRAM_DKERNEL & ~PAGE_SIZE_MASK) | PAGE_SIZE_1KB); |
| ++i; |
| icplb_add(extern_memory, SDRAM_IKERNEL); |
| dcplb_add(extern_memory, SDRAM_DKERNEL); |
| extern_memory += CPLB_PAGE_SIZE; |
| ++i; |
| #endif |
| |
| while (i < 16 && extern_memory < |
| (CONFIG_SYS_MONITOR_BASE & CPLB_EX_PAGE_MASK)) { |
| icplb_add(extern_memory, SDRAM_IGENERIC); |
| dcplb_add(extern_memory, SDRAM_DGENERIC); |
| extern_memory += CPLB_EX_PAGE_SIZE; |
| ++i; |
| } |
| while (i < 16) { |
| icplb_add(0, 0); |
| dcplb_add(0, 0); |
| ++i; |
| } |
| } |
| |
| int print_cpuinfo(void) |
| { |
| char buf[32]; |
| |
| printf("CPU: ADSP %s (Detected Rev: 0.%d) (%s boot)\n", |
| gd->bd->bi_cpu, |
| bfin_revid(), |
| get_bfin_boot_mode(CONFIG_BFIN_BOOT_MODE)); |
| |
| printf("Clock: VCO: %s MHz, ", strmhz(buf, get_vco())); |
| printf("Core: %s MHz, ", strmhz(buf, get_cclk())); |
| #if defined(__ADSPBF60x__) |
| printf("System0: %s MHz, ", strmhz(buf, get_sclk0())); |
| printf("System1: %s MHz, ", strmhz(buf, get_sclk1())); |
| printf("Dclk: %s MHz\n", strmhz(buf, get_dclk())); |
| #else |
| printf("System: %s MHz\n", strmhz(buf, get_sclk())); |
| #endif |
| |
| return 0; |
| } |
| |
| int exception_init(void) |
| { |
| bfin_write_EVT3(trap); |
| return 0; |
| } |
| |
| int irq_init(void) |
| { |
| #ifdef SIC_IMASK0 |
| bfin_write_SIC_IMASK0(0); |
| bfin_write_SIC_IMASK1(0); |
| # ifdef SIC_IMASK2 |
| bfin_write_SIC_IMASK2(0); |
| # endif |
| #elif defined(SICA_IMASK0) |
| bfin_write_SICA_IMASK0(0); |
| bfin_write_SICA_IMASK1(0); |
| #elif defined(SIC_IMASK) |
| bfin_write_SIC_IMASK(0); |
| #endif |
| /* Set up a dummy NMI handler if needed. */ |
| if (CONFIG_BFIN_BOOT_MODE == BFIN_BOOT_BYPASS || ANOMALY_05000219) |
| bfin_write_EVT2(evt_nmi); /* NMI */ |
| bfin_write_EVT5(evt_default); /* hardware error */ |
| bfin_write_EVT6(evt_default); /* core timer */ |
| bfin_write_EVT7(evt_default); |
| bfin_write_EVT8(evt_default); |
| bfin_write_EVT9(evt_default); |
| bfin_write_EVT10(evt_default); |
| bfin_write_EVT11(evt_default); |
| bfin_write_EVT12(evt_default); |
| bfin_write_EVT13(evt_default); |
| bfin_write_EVT14(evt_default); |
| bfin_write_EVT15(evt_default); |
| bfin_write_ILAT(0); |
| CSYNC(); |
| /* enable hardware error irq */ |
| irq_flags = 0x3f; |
| local_irq_enable(); |
| return 0; |
| } |
| |
| __attribute__ ((__noreturn__)) |
| void cpu_init_f(ulong bootflag, ulong loaded_from_ldr) |
| { |
| #ifndef CONFIG_BFIN_BOOTROM_USES_EVT1 |
| /* Build a NOP slide over the LDR jump block. Whee! */ |
| char nops[0xC]; |
| serial_early_puts("NOP Slide\n"); |
| memset(nops, 0x00, sizeof(nops)); |
| memcpy((void *)L1_INST_SRAM, nops, sizeof(nops)); |
| #endif |
| |
| if (!loaded_from_ldr) { |
| /* Relocate sections into L1 if the LDR didn't do it -- don't |
| * check length because the linker script does the size |
| * checking at build time. |
| */ |
| serial_early_puts("L1 Relocate\n"); |
| extern char _stext_l1[], _text_l1_lma[], _text_l1_len[]; |
| memcpy(&_stext_l1, &_text_l1_lma, (unsigned long)_text_l1_len); |
| extern char _sdata_l1[], _data_l1_lma[], _data_l1_len[]; |
| memcpy(&_sdata_l1, &_data_l1_lma, (unsigned long)_data_l1_len); |
| } |
| |
| /* |
| * Make sure our async settings are committed. Some bootroms |
| * (like the BF537) will reset some registers on us after it |
| * has finished loading the LDR. Or if we're booting over |
| * JTAG, the initcode never got a chance to run. Or if we |
| * aren't booting from parallel flash, the initcode skipped |
| * this step completely. |
| */ |
| program_async_controller(NULL); |
| |
| /* Save RETX so we can pass it while booting Linux */ |
| bfin_poweron_retx = bootflag; |
| |
| #ifdef CONFIG_DEBUG_DUMP |
| /* Turn on hardware trace buffer */ |
| bfin_write_TBUFCTL(TBUFPWR | TBUFEN); |
| #endif |
| |
| #ifndef CONFIG_PANIC_HANG |
| /* Reset upon a double exception rather than just hanging. |
| * Do not do bfin_read on SWRST as that will reset status bits. |
| */ |
| # ifdef SWRST |
| bfin_write_SWRST(DOUBLE_FAULT); |
| # endif |
| #endif |
| |
| #if defined(CONFIG_CORE1_RUN) && defined(COREB_L1_CODE_START) |
| bfin_core1_start(); |
| #endif |
| |
| serial_early_puts("Init global data\n"); |
| global_board_data_init(); |
| |
| board_init_f(0); |
| |
| /* should not be reached */ |
| while (1); |
| } |
| |
| int arch_cpu_init(void) |
| { |
| serial_early_puts("Init CPLB tables\n"); |
| init_cplbtables(); |
| |
| serial_early_puts("Exceptions setup\n"); |
| exception_init(); |
| |
| #ifndef CONFIG_ICACHE_OFF |
| serial_early_puts("Turn on ICACHE\n"); |
| icache_enable(); |
| #endif |
| #ifndef CONFIG_DCACHE_OFF |
| serial_early_puts("Turn on DCACHE\n"); |
| dcache_enable(); |
| #endif |
| |
| #ifdef DEBUG |
| if (GENERATED_GBL_DATA_SIZE < sizeof(*gd)) |
| hang(); |
| #endif |
| |
| /* Initialize */ |
| serial_early_puts("IRQ init\n"); |
| irq_init(); |
| |
| return 0; |
| } |
| |
| int arch_misc_init(void) |
| { |
| #if defined(CONFIG_SYS_I2C) |
| i2c_reloc_fixup(); |
| #endif |
| |
| display_global_data(); |
| |
| if (CONFIG_MEM_SIZE && bfin_os_log_check()) { |
| puts("\nLog buffer from operating system:\n"); |
| bfin_os_log_dump(); |
| puts("\n"); |
| } |
| |
| return 0; |
| } |
| |
| int interrupt_init(void) |
| { |
| return 0; |
| } |