Add misc support for Arria 10. Signed-off-by: Tien Fong Chee <tien.fong.chee@intel.com> Signed-off-by: Ley Foon Tan <ley.foon.tan@intel.com>master
parent
caf36e1edb
commit
35b9800ff2
@ -0,0 +1,259 @@ |
||||
/*
|
||||
* Copyright (C) 2016-2017 Intel Corporation |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0 |
||||
*/ |
||||
|
||||
#include <altera.h> |
||||
#include <common.h> |
||||
#include <errno.h> |
||||
#include <fdtdec.h> |
||||
#include <miiphy.h> |
||||
#include <netdev.h> |
||||
#include <ns16550.h> |
||||
#include <watchdog.h> |
||||
#include <asm/arch/misc.h> |
||||
#include <asm/arch/pinmux.h> |
||||
#include <asm/arch/reset_manager.h> |
||||
#include <asm/arch/sdram_arria10.h> |
||||
#include <asm/arch/system_manager.h> |
||||
#include <asm/arch/nic301.h> |
||||
#include <asm/io.h> |
||||
#include <asm/pl310.h> |
||||
|
||||
#define PINMUX_UART0_TX_SHARED_IO_OFFSET_Q1_3 0x08 |
||||
#define PINMUX_UART0_TX_SHARED_IO_OFFSET_Q2_11 0x58 |
||||
#define PINMUX_UART0_TX_SHARED_IO_OFFSET_Q3_3 0x68 |
||||
#define PINMUX_UART1_TX_SHARED_IO_OFFSET_Q1_7 0x18 |
||||
#define PINMUX_UART1_TX_SHARED_IO_OFFSET_Q3_7 0x78 |
||||
#define PINMUX_UART1_TX_SHARED_IO_OFFSET_Q4_3 0x98 |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
#if defined(CONFIG_SPL_BUILD) |
||||
static struct pl310_regs *const pl310 = |
||||
(struct pl310_regs *)CONFIG_SYS_PL310_BASE; |
||||
static const struct socfpga_noc_fw_ocram *noc_fw_ocram_base = |
||||
(void *)SOCFPGA_SDR_FIREWALL_OCRAM_ADDRESS; |
||||
#endif |
||||
|
||||
static struct socfpga_system_manager *sysmgr_regs = |
||||
(struct socfpga_system_manager *)SOCFPGA_SYSMGR_ADDRESS; |
||||
|
||||
/*
|
||||
* DesignWare Ethernet initialization |
||||
*/ |
||||
#ifdef CONFIG_ETH_DESIGNWARE |
||||
void dwmac_deassert_reset(const unsigned int of_reset_id, |
||||
const u32 phymode) |
||||
{ |
||||
u32 reset; |
||||
|
||||
if (of_reset_id == EMAC0_RESET) { |
||||
reset = SOCFPGA_RESET(EMAC0); |
||||
} else if (of_reset_id == EMAC1_RESET) { |
||||
reset = SOCFPGA_RESET(EMAC1); |
||||
} else if (of_reset_id == EMAC2_RESET) { |
||||
reset = SOCFPGA_RESET(EMAC2); |
||||
} else { |
||||
printf("GMAC: Invalid reset ID (%i)!\n", of_reset_id); |
||||
return; |
||||
} |
||||
|
||||
clrsetbits_le32(&sysmgr_regs->emac[of_reset_id - EMAC0_RESET], |
||||
SYSMGR_EMACGRP_CTRL_PHYSEL_MASK, |
||||
phymode); |
||||
|
||||
/* Release the EMAC controller from reset */ |
||||
socfpga_per_reset(reset, 0); |
||||
} |
||||
#endif |
||||
|
||||
#if defined(CONFIG_SPL_BUILD) |
||||
/*
|
||||
+ * This function initializes security policies to be consistent across |
||||
+ * all logic units in the Arria 10. |
||||
+ * |
||||
+ * The idea is to set all security policies to be normal, nonsecure |
||||
+ * for all units. |
||||
+ */ |
||||
static void initialize_security_policies(void) |
||||
{ |
||||
/* Put OCRAM in non-secure */ |
||||
writel(0x003f0000, &noc_fw_ocram_base->region0); |
||||
writel(0x1, &noc_fw_ocram_base->enable); |
||||
} |
||||
|
||||
int arch_early_init_r(void) |
||||
{ |
||||
initialize_security_policies(); |
||||
|
||||
/* Configure the L2 controller to make SDRAM start at 0 */ |
||||
writel(0x1, &pl310->pl310_addr_filter_start); |
||||
|
||||
/* assert reset to all except L4WD0 and L4TIMER0 */ |
||||
socfpga_per_reset_all(); |
||||
|
||||
/* configuring the clock based on handoff */ |
||||
/* TODO: Add call to cm_basic_init() */ |
||||
|
||||
/* Add device descriptor to FPGA device table */ |
||||
socfpga_fpga_add(); |
||||
return 0; |
||||
} |
||||
#else |
||||
int arch_early_init_r(void) |
||||
{ |
||||
return 0; |
||||
} |
||||
#endif |
||||
|
||||
/*
|
||||
* This function looking the 1st encounter UART peripheral, |
||||
* and then return its offset of the dedicated/shared IO pin |
||||
* mux. offset value (zero and above). |
||||
*/ |
||||
static int find_peripheral_uart(const void *blob, |
||||
int child, const char *node_name) |
||||
{ |
||||
int len; |
||||
fdt_addr_t base_addr = 0; |
||||
fdt_size_t size; |
||||
const u32 *cell; |
||||
u32 value, offset = 0; |
||||
|
||||
base_addr = fdtdec_get_addr_size(blob, child, "reg", &size); |
||||
if (base_addr != FDT_ADDR_T_NONE) { |
||||
cell = fdt_getprop(blob, child, "pinctrl-single,pins", |
||||
&len); |
||||
if (cell != NULL) { |
||||
for (; len > 0; len -= (2 * sizeof(u32))) { |
||||
offset = fdt32_to_cpu(*cell++); |
||||
value = fdt32_to_cpu(*cell++); |
||||
/* Found UART peripheral. */ |
||||
if (value == PINMUX_UART) |
||||
return offset; |
||||
} |
||||
} |
||||
} |
||||
return -EINVAL; |
||||
} |
||||
|
||||
/*
|
||||
* This function looks up the 1st encounter UART peripheral, |
||||
* and then return its offset of the dedicated/shared IO pin |
||||
* mux. UART peripheral is found if the offset is not in negative |
||||
* value. |
||||
*/ |
||||
static int is_peripheral_uart_true(const void *blob, |
||||
int node, const char *child_name) |
||||
{ |
||||
int child, len; |
||||
const char *node_name; |
||||
|
||||
child = fdt_first_subnode(blob, node); |
||||
|
||||
if (child < 0) |
||||
return -EINVAL; |
||||
|
||||
node_name = fdt_get_name(blob, child, &len); |
||||
|
||||
while (node_name) { |
||||
if (!strcmp(child_name, node_name)) |
||||
return find_peripheral_uart(blob, child, node_name); |
||||
|
||||
child = fdt_next_subnode(blob, child); |
||||
if (child < 0) |
||||
break; |
||||
|
||||
node_name = fdt_get_name(blob, child, &len); |
||||
} |
||||
|
||||
return -1; |
||||
} |
||||
|
||||
/*
|
||||
* This function looking the 1st encounter UART dedicated IO peripheral, |
||||
* and then return based address of the 1st encounter UART dedicated |
||||
* IO peripheral. |
||||
*/ |
||||
unsigned int dedicated_uart_com_port(const void *blob) |
||||
{ |
||||
int node; |
||||
|
||||
node = fdtdec_next_compatible(blob, 0, |
||||
COMPAT_ALTERA_SOCFPGA_PINCTRL_SINGLE); |
||||
if (node < 0) |
||||
return 0; |
||||
|
||||
if (is_peripheral_uart_true(blob, node, "dedicated") >= 0) |
||||
return SOCFPGA_UART1_ADDRESS; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
/*
|
||||
* This function looking the 1st encounter UART shared IO peripheral, and then |
||||
* return based address of the 1st encounter UART shared IO peripheral. |
||||
*/ |
||||
unsigned int shared_uart_com_port(const void *blob) |
||||
{ |
||||
int node, ret; |
||||
|
||||
node = fdtdec_next_compatible(blob, 0, |
||||
COMPAT_ALTERA_SOCFPGA_PINCTRL_SINGLE); |
||||
if (node < 0) |
||||
return 0; |
||||
|
||||
ret = is_peripheral_uart_true(blob, node, "shared"); |
||||
|
||||
if (ret == PINMUX_UART0_TX_SHARED_IO_OFFSET_Q1_3 || |
||||
ret == PINMUX_UART0_TX_SHARED_IO_OFFSET_Q2_11 || |
||||
ret == PINMUX_UART0_TX_SHARED_IO_OFFSET_Q3_3) |
||||
return SOCFPGA_UART0_ADDRESS; |
||||
else if (ret == PINMUX_UART1_TX_SHARED_IO_OFFSET_Q1_7 || |
||||
ret == PINMUX_UART1_TX_SHARED_IO_OFFSET_Q3_7 || |
||||
ret == PINMUX_UART1_TX_SHARED_IO_OFFSET_Q4_3) |
||||
return SOCFPGA_UART1_ADDRESS; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
/*
|
||||
* This function looking the 1st encounter UART peripheral, and then return |
||||
* base address of the 1st encounter UART peripheral. |
||||
*/ |
||||
unsigned int uart_com_port(const void *blob) |
||||
{ |
||||
unsigned int ret; |
||||
|
||||
ret = dedicated_uart_com_port(blob); |
||||
|
||||
if (ret) |
||||
return ret; |
||||
|
||||
return shared_uart_com_port(blob); |
||||
} |
||||
|
||||
/*
|
||||
* Print CPU information |
||||
*/ |
||||
#if defined(CONFIG_DISPLAY_CPUINFO) |
||||
int print_cpuinfo(void) |
||||
{ |
||||
const u32 bsel = |
||||
SYSMGR_GET_BOOTINFO_BSEL(readl(&sysmgr_regs->bootinfo)); |
||||
|
||||
puts("CPU: Altera SoCFPGA Arria 10\n"); |
||||
|
||||
printf("BOOT: %s\n", bsel_str[bsel].name); |
||||
return 0; |
||||
} |
||||
#endif |
||||
|
||||
#ifdef CONFIG_ARCH_MISC_INIT |
||||
int arch_misc_init(void) |
||||
{ |
||||
return 0; |
||||
} |
||||
#endif |
Loading…
Reference in new issue