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