|  | /* | 
|  | * U-boot - bootldr.c | 
|  | * | 
|  | * Copyright (c) 2005-2008 Analog Devices Inc. | 
|  | * | 
|  | * See file CREDITS for list of people who contributed to this | 
|  | * project. | 
|  | * | 
|  | * Licensed under the GPL-2 or later. | 
|  | */ | 
|  |  | 
|  | #include <config.h> | 
|  | #include <common.h> | 
|  | #include <command.h> | 
|  |  | 
|  | #include <asm/blackfin.h> | 
|  | #include <asm/mach-common/bits/bootrom.h> | 
|  |  | 
|  | /* Simple sanity check on the specified address to make sure it contains | 
|  | * an LDR image of some sort. | 
|  | */ | 
|  | static bool ldr_valid_signature(uint8_t *data) | 
|  | { | 
|  | #if defined(__ADSPBF561__) | 
|  |  | 
|  | /* BF56x has a 4 byte global header */ | 
|  | if (data[3] == 0xA0) | 
|  | return true; | 
|  |  | 
|  | #elif defined(__ADSPBF531__) || defined(__ADSPBF532__) || defined(__ADSPBF533__) || \ | 
|  | defined(__ADSPBF534__) || defined(__ADSPBF536__) || defined(__ADSPBF537__) || \ | 
|  | defined(__ADSPBF538__) || defined(__ADSPBF539__) | 
|  |  | 
|  | /* all the BF53x should start at this address mask */ | 
|  | uint32_t addr; | 
|  | memmove(&addr, data, sizeof(addr)); | 
|  | if ((addr & 0xFF0FFF0F) == 0xFF000000) | 
|  | return true; | 
|  | #else | 
|  |  | 
|  | /* everything newer has a magic byte */ | 
|  | uint32_t count; | 
|  | memmove(&count, data + 8, sizeof(count)); | 
|  | if (data[3] == 0xAD && count == 0) | 
|  | return true; | 
|  |  | 
|  | #endif | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* If the Blackfin is new enough, the Blackfin on-chip ROM supports loading | 
|  | * LDRs from random memory addresses.  So whenever possible, use that.  In | 
|  | * the older cases (BF53x/BF561), parse the LDR format ourselves. | 
|  | */ | 
|  | #define ZEROFILL  0x0001 | 
|  | #define RESVECT   0x0002 | 
|  | #define INIT      0x0008 | 
|  | #define IGNORE    0x0010 | 
|  | #define FINAL     0x8000 | 
|  | static void ldr_load(uint8_t *base_addr) | 
|  | { | 
|  | #if defined(__ADSPBF531__) || defined(__ADSPBF532__) || defined(__ADSPBF533__) || \ | 
|  | /*defined(__ADSPBF534__) || defined(__ADSPBF536__) || defined(__ADSPBF537__) ||*/\ | 
|  | defined(__ADSPBF538__) || defined(__ADSPBF539__) || defined(__ADSPBF561__) | 
|  |  | 
|  | uint32_t addr; | 
|  | uint32_t count; | 
|  | uint16_t flags; | 
|  |  | 
|  | /* the bf56x has a 4 byte global header ... but it is useless to | 
|  | * us when booting an LDR from a memory address, so skip it | 
|  | */ | 
|  | # ifdef __ADSPBF561__ | 
|  | base_addr += 4; | 
|  | # endif | 
|  |  | 
|  | memmove(&flags, base_addr + 8, sizeof(flags)); | 
|  | bfin_write_EVT1(flags & RESVECT ? 0xFFA00000 : 0xFFA08000); | 
|  |  | 
|  | do { | 
|  | /* block header may not be aligned */ | 
|  | memmove(&addr, base_addr, sizeof(addr)); | 
|  | memmove(&count, base_addr+4, sizeof(count)); | 
|  | memmove(&flags, base_addr+8, sizeof(flags)); | 
|  | base_addr += sizeof(addr) + sizeof(count) + sizeof(flags); | 
|  |  | 
|  | printf("loading to 0x%08x (0x%x bytes) flags: 0x%04x\n", | 
|  | addr, count, flags); | 
|  |  | 
|  | if (!(flags & IGNORE)) { | 
|  | if (flags & ZEROFILL) | 
|  | memset((void *)addr, 0x00, count); | 
|  | else | 
|  | memcpy((void *)addr, base_addr, count); | 
|  |  | 
|  | if (flags & INIT) { | 
|  | void (*init)(void) = (void *)addr; | 
|  | init(); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!(flags & ZEROFILL)) | 
|  | base_addr += count; | 
|  | } while (!(flags & FINAL)); | 
|  |  | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* For BF537, we use the _BOOTROM_BOOT_DXE_FLASH funky ROM function. | 
|  | * For all other BF53x/BF56x, we just call the entry point. | 
|  | * For everything else (newer), we use _BOOTROM_MEMBOOT ROM function. | 
|  | */ | 
|  | static void ldr_exec(void *addr) | 
|  | { | 
|  | #if defined(__ADSPBF534__) || defined(__ADSPBF536__) || defined(__ADSPBF537__) | 
|  |  | 
|  | /* restore EVT1 to reset value as this is what the bootrom uses as | 
|  | * the default entry point when booting the final block of LDRs | 
|  | */ | 
|  | bfin_write_EVT1(L1_INST_SRAM); | 
|  | __asm__("call (%0);" : : "a"(_BOOTROM_MEMBOOT), "q7"(addr) : "RETS", "memory"); | 
|  |  | 
|  | #elif defined(__ADSPBF531__) || defined(__ADSPBF532__) || defined(__ADSPBF533__) || \ | 
|  | defined(__ADSPBF538__) || defined(__ADSPBF539__) || defined(__ADSPBF561__) | 
|  |  | 
|  | void (*ldr_entry)(void) = (void *)bfin_read_EVT1(); | 
|  | ldr_entry(); | 
|  |  | 
|  | #else | 
|  |  | 
|  | int32_t (*BOOTROM_MEM)(void *, int32_t, int32_t, void *) = (void *)_BOOTROM_MEMBOOT; | 
|  | BOOTROM_MEM(addr, 0, 0, NULL); | 
|  |  | 
|  | #endif | 
|  | } | 
|  |  | 
|  | /* | 
|  | * the bootldr command loads an address, checks to see if there | 
|  | *   is a Boot stream that the on-chip BOOTROM can understand, | 
|  | *   and loads it via the BOOTROM Callback. It is possible | 
|  | *   to also add booting from SPI, or TWI, but this function does | 
|  | *   not currently support that. | 
|  | */ | 
|  | int do_bootldr(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[]) | 
|  | { | 
|  | void *addr; | 
|  |  | 
|  | /* Get the address */ | 
|  | if (argc < 2) | 
|  | addr = (void *)load_addr; | 
|  | else | 
|  | addr = (void *)simple_strtoul(argv[1], NULL, 16); | 
|  |  | 
|  | /* Check if it is a LDR file */ | 
|  | if (ldr_valid_signature(addr)) { | 
|  | printf("## Booting ldr image at 0x%p ...\n", addr); | 
|  | ldr_load(addr); | 
|  |  | 
|  | icache_disable(); | 
|  | dcache_disable(); | 
|  |  | 
|  | ldr_exec(addr); | 
|  | } else | 
|  | printf("## No ldr image at address 0x%p\n", addr); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | U_BOOT_CMD(bootldr, 2, 0, do_bootldr, | 
|  | "boot ldr image from memory", | 
|  | "[addr]\n" | 
|  | "" | 
|  | ); |