mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-15 06:34:03 +00:00
83d290c56f
When U-Boot started using SPDX tags we were among the early adopters and there weren't a lot of other examples to borrow from. So we picked the area of the file that usually had a full license text and replaced it with an appropriate SPDX-License-Identifier: entry. Since then, the Linux Kernel has adopted SPDX tags and they place it as the very first line in a file (except where shebangs are used, then it's second line) and with slightly different comment styles than us. In part due to community overlap, in part due to better tag visibility and in part for other minor reasons, switch over to that style. This commit changes all instances where we have a single declared license in the tag as both the before and after are identical in tag contents. There's also a few places where I found we did not have a tag and have introduced one. Signed-off-by: Tom Rini <trini@konsulko.com>
286 lines
6.5 KiB
C
286 lines
6.5 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Board initialization for EP93xx
|
|
*
|
|
* Copyright (C) 2013
|
|
* Sergey Kostanbaev <sergey.kostanbaev <at> fairwaves.ru>
|
|
*
|
|
* Copyright (C) 2009
|
|
* Matthias Kaehlcke <matthias <at> kaehlcke.net>
|
|
*
|
|
* (C) Copyright 2002 2003
|
|
* Network Audio Technologies, Inc. <www.netaudiotech.com>
|
|
* Adam Bezanson <bezanson <at> netaudiotech.com>
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <common.h>
|
|
#include <netdev.h>
|
|
#include <asm/io.h>
|
|
#include <asm/mach-types.h>
|
|
#include <asm/arch/ep93xx.h>
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
/*
|
|
* usb_div: 4, nbyp2: 1, pll2_en: 1
|
|
* pll2_x1: 368640000.000000, pll2_x2ip: 15360000.000000,
|
|
* pll2_x2: 384000000.000000, pll2_out: 192000000.000000
|
|
*/
|
|
#define CLKSET2_VAL (23 << SYSCON_CLKSET_PLL_X2IPD_SHIFT | \
|
|
24 << SYSCON_CLKSET_PLL_X2FBD2_SHIFT | \
|
|
24 << SYSCON_CLKSET_PLL_X1FBD1_SHIFT | \
|
|
1 << SYSCON_CLKSET_PLL_PS_SHIFT | \
|
|
SYSCON_CLKSET2_PLL2_EN | \
|
|
SYSCON_CLKSET2_NBYP2 | \
|
|
3 << SYSCON_CLKSET2_USB_DIV_SHIFT)
|
|
|
|
#define SMC_BCR6_VALUE (2 << SMC_BCR_IDCY_SHIFT | 5 << SMC_BCR_WST1_SHIFT | \
|
|
SMC_BCR_BLE | 2 << SMC_BCR_WST2_SHIFT | \
|
|
1 << SMC_BCR_MW_SHIFT)
|
|
|
|
/* delay execution before timers are initialized */
|
|
static inline void early_udelay(uint32_t usecs)
|
|
{
|
|
/* loop takes 4 cycles at 5.0ns (fastest case, running at 200MHz) */
|
|
register uint32_t loops = (usecs * 1000) / 20;
|
|
|
|
__asm__ volatile ("1:\n"
|
|
"subs %0, %1, #1\n"
|
|
"bne 1b" : "=r" (loops) : "0" (loops));
|
|
}
|
|
|
|
#ifndef CONFIG_EP93XX_NO_FLASH_CFG
|
|
static void flash_cfg(void)
|
|
{
|
|
struct smc_regs *smc = (struct smc_regs *)SMC_BASE;
|
|
|
|
writel(SMC_BCR6_VALUE, &smc->bcr6);
|
|
}
|
|
#else
|
|
#define flash_cfg()
|
|
#endif
|
|
|
|
int board_init(void)
|
|
{
|
|
/*
|
|
* Setup PLL2, PPL1 has been set during lowlevel init
|
|
*/
|
|
struct syscon_regs *syscon = (struct syscon_regs *)SYSCON_BASE;
|
|
writel(CLKSET2_VAL, &syscon->clkset2);
|
|
|
|
/*
|
|
* the user's guide recommends to wait at least 1 ms for PLL2 to
|
|
* stabilize
|
|
*/
|
|
early_udelay(1000);
|
|
|
|
/* Go to Async mode */
|
|
__asm__ volatile ("mrc p15, 0, r0, c1, c0, 0");
|
|
__asm__ volatile ("orr r0, r0, #0xc0000000");
|
|
__asm__ volatile ("mcr p15, 0, r0, c1, c0, 0");
|
|
|
|
icache_enable();
|
|
|
|
#ifdef USE_920T_MMU
|
|
dcache_enable();
|
|
#endif
|
|
|
|
/* Machine number, as defined in linux/arch/arm/tools/mach-types */
|
|
gd->bd->bi_arch_number = CONFIG_MACH_TYPE;
|
|
|
|
/* adress of boot parameters */
|
|
gd->bd->bi_boot_params = LINUX_BOOT_PARAM_ADDR;
|
|
|
|
/* We have a console */
|
|
gd->have_console = 1;
|
|
|
|
enable_interrupts();
|
|
|
|
flash_cfg();
|
|
|
|
green_led_on();
|
|
red_led_off();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int board_early_init_f(void)
|
|
{
|
|
/*
|
|
* set UARTBAUD bit to drive UARTs with 14.7456MHz instead of
|
|
* 14.7456/2 MHz
|
|
*/
|
|
struct syscon_regs *syscon = (struct syscon_regs *)SYSCON_BASE;
|
|
writel(SYSCON_PWRCNT_UART_BAUD, &syscon->pwrcnt);
|
|
return 0;
|
|
}
|
|
|
|
int board_eth_init(bd_t *bd)
|
|
{
|
|
return ep93xx_eth_initialize(0, MAC_BASE);
|
|
}
|
|
|
|
static void dram_fill_bank_addr(unsigned dram_addr_mask, unsigned dram_bank_cnt,
|
|
unsigned dram_bank_base[CONFIG_NR_DRAM_BANKS])
|
|
{
|
|
if (dram_bank_cnt == 1) {
|
|
dram_bank_base[0] = PHYS_SDRAM_1;
|
|
} else {
|
|
/* Table lookup for holes in address space. Maximum memory
|
|
* for the single SDCS may be up to 256Mb. We start scanning
|
|
* banks from 1Mb, so it could be up to 128 banks theoretically.
|
|
* We need at maximum 7 bits for the loockup, 8 slots is
|
|
* enough for the worst case.
|
|
*/
|
|
unsigned tbl[8];
|
|
unsigned i = dram_bank_cnt / 2;
|
|
unsigned j = 0x00100000; /* 1 Mb */
|
|
unsigned *ptbl = tbl;
|
|
do {
|
|
while (!(dram_addr_mask & j)) {
|
|
j <<= 1;
|
|
}
|
|
*ptbl++ = j;
|
|
j <<= 1;
|
|
i >>= 1;
|
|
} while (i != 0);
|
|
|
|
for (i = dram_bank_cnt, j = 0;
|
|
(i != 0) && (j < CONFIG_NR_DRAM_BANKS); --i, ++j) {
|
|
unsigned addr = PHYS_SDRAM_1;
|
|
unsigned k;
|
|
unsigned bit;
|
|
|
|
for (k = 0, bit = 1; k < 8; k++, bit <<= 1) {
|
|
if (bit & j)
|
|
addr |= tbl[k];
|
|
}
|
|
|
|
dram_bank_base[j] = addr;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* called in board_init_f (before relocation) */
|
|
static unsigned dram_init_banksize_int(int print)
|
|
{
|
|
/*
|
|
* Collect information of banks that has been filled during lowlevel
|
|
* initialization
|
|
*/
|
|
unsigned i;
|
|
unsigned dram_bank_base[CONFIG_NR_DRAM_BANKS];
|
|
unsigned dram_total = 0;
|
|
unsigned dram_bank_size = *(unsigned *)
|
|
(PHYS_SDRAM_1 | UBOOT_MEMORYCNF_BANK_SIZE);
|
|
unsigned dram_addr_mask = *(unsigned *)
|
|
(PHYS_SDRAM_1 | UBOOT_MEMORYCNF_BANK_MASK);
|
|
unsigned dram_bank_cnt = *(unsigned *)
|
|
(PHYS_SDRAM_1 | UBOOT_MEMORYCNF_BANK_COUNT);
|
|
|
|
dram_fill_bank_addr(dram_addr_mask, dram_bank_cnt, dram_bank_base);
|
|
|
|
for (i = 0; i < dram_bank_cnt; i++) {
|
|
gd->bd->bi_dram[i].start = dram_bank_base[i];
|
|
gd->bd->bi_dram[i].size = dram_bank_size;
|
|
dram_total += dram_bank_size;
|
|
}
|
|
for (; i < CONFIG_NR_DRAM_BANKS; i++) {
|
|
gd->bd->bi_dram[i].start = 0;
|
|
gd->bd->bi_dram[i].size = 0;
|
|
}
|
|
|
|
if (print) {
|
|
printf("DRAM mask: %08x\n", dram_addr_mask);
|
|
printf("DRAM total %u banks:\n", dram_bank_cnt);
|
|
printf("bank base-address size\n");
|
|
|
|
if (dram_bank_cnt > CONFIG_NR_DRAM_BANKS) {
|
|
printf("WARNING! UBoot was configured for %u banks,\n"
|
|
"but %u has been found. "
|
|
"Supressing extra memory banks\n",
|
|
CONFIG_NR_DRAM_BANKS, dram_bank_cnt);
|
|
dram_bank_cnt = CONFIG_NR_DRAM_BANKS;
|
|
}
|
|
|
|
for (i = 0; i < dram_bank_cnt; i++) {
|
|
printf(" %u %08x %08x\n",
|
|
i, dram_bank_base[i], dram_bank_size);
|
|
}
|
|
printf(" ------------------------------------------\n"
|
|
"Total %9d\n\n",
|
|
dram_total);
|
|
}
|
|
|
|
return dram_total;
|
|
}
|
|
|
|
int dram_init_banksize(void)
|
|
{
|
|
dram_init_banksize_int(0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* called in board_init_f (before relocation) */
|
|
int dram_init(void)
|
|
{
|
|
struct syscon_regs *syscon = (struct syscon_regs *)SYSCON_BASE;
|
|
unsigned sec_id = readl(SECURITY_EXTENSIONID);
|
|
unsigned chip_id = readl(&syscon->chipid);
|
|
|
|
printf("CPU: Cirrus Logic ");
|
|
switch (sec_id & 0x000001FE) {
|
|
case 0x00000008:
|
|
printf("EP9301");
|
|
break;
|
|
case 0x00000004:
|
|
printf("EP9307");
|
|
break;
|
|
case 0x00000002:
|
|
printf("EP931x");
|
|
break;
|
|
case 0x00000000:
|
|
printf("EP9315");
|
|
break;
|
|
default:
|
|
printf("<unknown>");
|
|
break;
|
|
}
|
|
|
|
printf(" - Rev. ");
|
|
switch (chip_id & 0xF0000000) {
|
|
case 0x00000000:
|
|
printf("A");
|
|
break;
|
|
case 0x10000000:
|
|
printf("B");
|
|
break;
|
|
case 0x20000000:
|
|
printf("C");
|
|
break;
|
|
case 0x30000000:
|
|
printf("D0");
|
|
break;
|
|
case 0x40000000:
|
|
printf("D1");
|
|
break;
|
|
case 0x50000000:
|
|
printf("E0");
|
|
break;
|
|
case 0x60000000:
|
|
printf("E1");
|
|
break;
|
|
case 0x70000000:
|
|
printf("E2");
|
|
break;
|
|
default:
|
|
printf("?");
|
|
break;
|
|
}
|
|
printf(" (SecExtID=%.8x/ChipID=%.8x)\n", sec_id, chip_id);
|
|
|
|
gd->ram_size = dram_init_banksize_int(1);
|
|
return 0;
|
|
}
|