2019-04-10 08:43:34 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
|
|
|
* Copyright 2019 NXP
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
2020-05-10 17:40:02 +00:00
|
|
|
#include <init.h>
|
2019-04-10 08:43:34 +00:00
|
|
|
#include <malloc.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fsl_ddr.h>
|
2020-05-10 17:39:56 +00:00
|
|
|
#include <net.h>
|
2020-10-31 03:38:53 +00:00
|
|
|
#include <asm/global_data.h>
|
2019-04-10 08:43:34 +00:00
|
|
|
#include <asm/io.h>
|
|
|
|
#include <hwconfig.h>
|
|
|
|
#include <fdt_support.h>
|
|
|
|
#include <linux/libfdt.h>
|
2019-08-02 15:44:25 +00:00
|
|
|
#include <env_internal.h>
|
2019-04-10 08:43:34 +00:00
|
|
|
#include <asm/arch-fsl-layerscape/soc.h>
|
2019-07-30 14:29:59 +00:00
|
|
|
#include <asm/arch-fsl-layerscape/fsl_icid.h>
|
2019-04-10 08:43:34 +00:00
|
|
|
#include <i2c.h>
|
|
|
|
#include <asm/arch/soc.h>
|
|
|
|
#ifdef CONFIG_FSL_LS_PPA
|
|
|
|
#include <asm/arch/ppa.h>
|
|
|
|
#endif
|
|
|
|
#include <fsl_immap.h>
|
|
|
|
#include <netdev.h>
|
|
|
|
|
|
|
|
#include <fdtdec.h>
|
|
|
|
#include <miiphy.h>
|
|
|
|
#include "../common/qixis.h"
|
2019-12-10 14:55:39 +00:00
|
|
|
#include "../drivers/net/fsl_enetc.h"
|
2019-04-10 08:43:34 +00:00
|
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
2019-04-10 08:43:35 +00:00
|
|
|
int config_board_mux(void)
|
|
|
|
{
|
2020-03-19 08:48:23 +00:00
|
|
|
#ifndef CONFIG_LPUART
|
2019-04-10 08:43:35 +00:00
|
|
|
#if defined(CONFIG_TARGET_LS1028AQDS) && defined(CONFIG_FSL_QIXIS)
|
|
|
|
u8 reg;
|
|
|
|
|
|
|
|
reg = QIXIS_READ(brdcfg[13]);
|
|
|
|
/* Field| Function
|
|
|
|
* 7-6 | Controls I2C3 routing (net CFG_MUX_I2C3):
|
|
|
|
* I2C3 | 10= Routes {SCL, SDA} to CAN1 transceiver as {TX, RX}.
|
|
|
|
* 5-4 | Controls I2C4 routing (net CFG_MUX_I2C4):
|
|
|
|
* I2C4 |11= Routes {SCL, SDA} to CAN2 transceiver as {TX, RX}.
|
|
|
|
*/
|
|
|
|
reg &= ~(0xf0);
|
|
|
|
reg |= 0xb0;
|
|
|
|
QIXIS_WRITE(brdcfg[13], reg);
|
|
|
|
|
|
|
|
reg = QIXIS_READ(brdcfg[15]);
|
|
|
|
/* Field| Function
|
|
|
|
* 7 | Controls the CAN1 transceiver (net CFG_CAN1_STBY):
|
|
|
|
* CAN1 | 0= CAN #1 transceiver enabled
|
|
|
|
* 6 | Controls the CAN2 transceiver (net CFG_CAN2_STBY):
|
|
|
|
* CAN2 | 0= CAN #2 transceiver enabled
|
|
|
|
*/
|
|
|
|
reg &= ~(0xc0);
|
|
|
|
QIXIS_WRITE(brdcfg[15], reg);
|
|
|
|
#endif
|
2020-03-19 08:48:23 +00:00
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-03-19 08:48:23 +00:00
|
|
|
#ifdef CONFIG_LPUART
|
|
|
|
u32 get_lpuart_clk(void)
|
|
|
|
{
|
|
|
|
return gd->bus_clk / CONFIG_SYS_FSL_LPUART_CLK_DIV;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
int board_init(void)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_ENV_IS_NOWHERE
|
|
|
|
gd->env_addr = (ulong)&default_environment[0];
|
|
|
|
#endif
|
|
|
|
|
2019-09-30 10:16:57 +00:00
|
|
|
#ifdef CONFIG_FSL_CAAM
|
|
|
|
sec_init();
|
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
#ifdef CONFIG_FSL_LS_PPA
|
|
|
|
ppa_init();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef CONFIG_SYS_EARLY_PCI_INIT
|
|
|
|
pci_init();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(CONFIG_TARGET_LS1028ARDB)
|
|
|
|
u8 val = I2C_MUX_CH_DEFAULT;
|
|
|
|
|
2021-02-09 11:52:45 +00:00
|
|
|
#if !CONFIG_IS_ENABLED(DM_I2C)
|
2019-04-10 08:43:34 +00:00
|
|
|
i2c_write(I2C_MUX_PCA_ADDR_PRI, 0x0b, 1, &val, 1);
|
2019-07-10 13:16:49 +00:00
|
|
|
#else
|
|
|
|
struct udevice *dev;
|
|
|
|
|
|
|
|
if (!i2c_get_chip_for_busnum(0, I2C_MUX_PCA_ADDR_PRI, 1, &dev))
|
|
|
|
dm_i2c_write(dev, 0x0b, &val, 1);
|
|
|
|
#endif
|
2019-11-18 05:26:09 +00:00
|
|
|
#endif
|
2019-07-10 13:16:49 +00:00
|
|
|
|
2019-11-18 05:26:09 +00:00
|
|
|
#if defined(CONFIG_TARGET_LS1028ARDB)
|
|
|
|
u8 reg;
|
|
|
|
|
|
|
|
reg = QIXIS_READ(brdcfg[4]);
|
|
|
|
/*
|
|
|
|
* Field | Function
|
|
|
|
* 3 | DisplayPort Power Enable (net DP_PWR_EN):
|
|
|
|
* DPPWR | 0= DP_PWR is enabled.
|
|
|
|
*/
|
|
|
|
reg &= ~(DP_PWD_EN_DEFAULT_MASK);
|
|
|
|
QIXIS_WRITE(brdcfg[4], reg);
|
2019-04-10 08:43:34 +00:00
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-06-26 06:13:33 +00:00
|
|
|
int board_eth_init(struct bd_info *bis)
|
2019-04-10 08:43:34 +00:00
|
|
|
{
|
|
|
|
return pci_eth_init(bis);
|
|
|
|
}
|
|
|
|
|
2020-01-10 23:05:39 +00:00
|
|
|
#ifdef CONFIG_MISC_INIT_R
|
|
|
|
int misc_init_r(void)
|
2019-04-10 08:43:35 +00:00
|
|
|
{
|
|
|
|
config_board_mux();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
int board_early_init_f(void)
|
|
|
|
{
|
2020-03-19 08:48:23 +00:00
|
|
|
#ifdef CONFIG_LPUART
|
|
|
|
u8 uart;
|
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
#ifdef CONFIG_SYS_I2C_EARLY_INIT
|
|
|
|
i2c_early_init_f();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
fsl_lsch3_early_init_f();
|
2020-03-19 08:48:23 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_LPUART
|
|
|
|
/*
|
|
|
|
* Field| Function
|
|
|
|
* --------------------------------------------------------------
|
|
|
|
* 7-6 | Controls I2C3 routing (net CFG_MUX_I2C3):
|
|
|
|
* I2C3 | 11= Routes {SCL, SDA} to LPUART1 header as {SOUT, SIN}.
|
|
|
|
* --------------------------------------------------------------
|
|
|
|
* 5-4 | Controls I2C4 routing (net CFG_MUX_I2C4):
|
|
|
|
* I2C4 |11= Routes {SCL, SDA} to LPUART1 header as {CTS_B, RTS_B}.
|
|
|
|
*/
|
|
|
|
/* use lpuart0 as system console */
|
|
|
|
uart = QIXIS_READ(brdcfg[13]);
|
|
|
|
uart &= ~CFG_LPUART_MUX_MASK;
|
|
|
|
uart |= CFG_LPUART_EN;
|
|
|
|
QIXIS_WRITE(brdcfg[13], uart);
|
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void detail_board_ddr_info(void)
|
|
|
|
{
|
|
|
|
puts("\nDDR ");
|
|
|
|
print_size(gd->bd->bi_dram[0].size + gd->bd->bi_dram[1].size, "");
|
|
|
|
print_ddr_info(0);
|
|
|
|
}
|
|
|
|
|
2020-04-14 09:24:48 +00:00
|
|
|
int esdhc_status_fixup(void *blob, const char *compat)
|
|
|
|
{
|
|
|
|
void __iomem *dcfg_ccsr = (void __iomem *)DCFG_BASE;
|
|
|
|
char esdhc1_path[] = "/soc/mmc@2140000";
|
|
|
|
char esdhc2_path[] = "/soc/mmc@2150000";
|
|
|
|
char dspi1_path[] = "/soc/spi@2100000";
|
|
|
|
char dspi2_path[] = "/soc/spi@2110000";
|
|
|
|
u32 mux_sdhc1, mux_sdhc2;
|
|
|
|
u32 io = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The PMUX IO-expander for mux select is used to control
|
|
|
|
* the muxing of various onboard interfaces.
|
|
|
|
*/
|
|
|
|
|
|
|
|
io = in_le32(dcfg_ccsr + DCFG_RCWSR12);
|
|
|
|
mux_sdhc1 = (io >> DCFG_RCWSR12_SDHC_SHIFT) & DCFG_RCWSR12_SDHC_MASK;
|
|
|
|
|
|
|
|
/* Disable esdhc1/dspi1 if not selected. */
|
|
|
|
if (mux_sdhc1 != 0)
|
|
|
|
do_fixup_by_path(blob, esdhc1_path, "status", "disabled",
|
|
|
|
sizeof("disabled"), 1);
|
|
|
|
if (mux_sdhc1 != 2)
|
|
|
|
do_fixup_by_path(blob, dspi1_path, "status", "disabled",
|
|
|
|
sizeof("disabled"), 1);
|
|
|
|
|
|
|
|
io = in_le32(dcfg_ccsr + DCFG_RCWSR13);
|
|
|
|
mux_sdhc2 = (io >> DCFG_RCWSR13_SDHC_SHIFT) & DCFG_RCWSR13_SDHC_MASK;
|
|
|
|
|
|
|
|
/* Disable esdhc2/dspi2 if not selected. */
|
|
|
|
if (mux_sdhc2 != 0)
|
|
|
|
do_fixup_by_path(blob, esdhc2_path, "status", "disabled",
|
|
|
|
sizeof("disabled"), 1);
|
|
|
|
if (mux_sdhc2 != 2)
|
|
|
|
do_fixup_by_path(blob, dspi2_path, "status", "disabled",
|
|
|
|
sizeof("disabled"), 1);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
#ifdef CONFIG_OF_BOARD_SETUP
|
2020-06-26 06:13:33 +00:00
|
|
|
int ft_board_setup(void *blob, struct bd_info *bd)
|
2019-04-10 08:43:34 +00:00
|
|
|
{
|
|
|
|
u64 base[CONFIG_NR_DRAM_BANKS];
|
|
|
|
u64 size[CONFIG_NR_DRAM_BANKS];
|
|
|
|
|
|
|
|
ft_cpu_setup(blob, bd);
|
|
|
|
|
|
|
|
/* fixup DT for the two GPP DDR banks */
|
|
|
|
base[0] = gd->bd->bi_dram[0].start;
|
|
|
|
size[0] = gd->bd->bi_dram[0].size;
|
|
|
|
base[1] = gd->bd->bi_dram[1].start;
|
|
|
|
size[1] = gd->bd->bi_dram[1].size;
|
|
|
|
|
|
|
|
#ifdef CONFIG_RESV_RAM
|
|
|
|
/* reduce size if reserved memory is within this bank */
|
|
|
|
if (gd->arch.resv_ram >= base[0] &&
|
|
|
|
gd->arch.resv_ram < base[0] + size[0])
|
|
|
|
size[0] = gd->arch.resv_ram - base[0];
|
|
|
|
else if (gd->arch.resv_ram >= base[1] &&
|
|
|
|
gd->arch.resv_ram < base[1] + size[1])
|
|
|
|
size[1] = gd->arch.resv_ram - base[1];
|
|
|
|
#endif
|
|
|
|
|
|
|
|
fdt_fixup_memory_banks(blob, base, size, 2);
|
|
|
|
|
2019-07-30 14:29:59 +00:00
|
|
|
fdt_fixup_icid(blob);
|
|
|
|
|
2019-12-10 14:55:39 +00:00
|
|
|
#ifdef CONFIG_FSL_ENETC
|
|
|
|
fdt_fixup_enetc_mac(blob);
|
|
|
|
#endif
|
|
|
|
|
2019-04-10 08:43:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_FSL_QIXIS
|
|
|
|
int checkboard(void)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_TFABOOT
|
|
|
|
enum boot_src src = get_boot_src();
|
|
|
|
#endif
|
|
|
|
u8 sw;
|
|
|
|
|
|
|
|
int clock;
|
|
|
|
char *board;
|
|
|
|
char buf[64] = {0};
|
|
|
|
static const char *freq[6] = {"100.00", "125.00", "156.25",
|
|
|
|
"161.13", "322.26", "100.00 SS"};
|
|
|
|
|
|
|
|
cpu_name(buf);
|
|
|
|
/* find the board details */
|
|
|
|
sw = QIXIS_READ(id);
|
|
|
|
|
|
|
|
switch (sw) {
|
|
|
|
case 0x46:
|
|
|
|
board = "QDS";
|
|
|
|
break;
|
|
|
|
case 0x47:
|
|
|
|
board = "RDB";
|
|
|
|
break;
|
|
|
|
case 0x49:
|
|
|
|
board = "HSSI";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
board = "unknown";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
sw = QIXIS_READ(arch);
|
|
|
|
printf("Board: %s-%s, Version: %c, boot from ",
|
|
|
|
buf, board, (sw & 0xf) + 'A' - 1);
|
|
|
|
|
|
|
|
sw = QIXIS_READ(brdcfg[0]);
|
|
|
|
sw = (sw & QIXIS_LBMAP_MASK) >> QIXIS_LBMAP_SHIFT;
|
|
|
|
|
|
|
|
#ifdef CONFIG_TFABOOT
|
|
|
|
if (src == BOOT_SOURCE_SD_MMC) {
|
|
|
|
puts("SD\n");
|
|
|
|
} else if (src == BOOT_SOURCE_SD_MMC2) {
|
|
|
|
puts("eMMC\n");
|
|
|
|
} else {
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_SD_BOOT
|
|
|
|
puts("SD\n");
|
|
|
|
#elif defined(CONFIG_EMMC_BOOT)
|
|
|
|
puts("eMMC\n");
|
|
|
|
#else
|
|
|
|
switch (sw) {
|
|
|
|
case 0:
|
|
|
|
case 4:
|
|
|
|
printf("NOR\n");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
printf("NAND\n");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printf("invalid setting of SW%u\n", QIXIS_LBMAP_SWITCH);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_TFABOOT
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
printf("FPGA: v%d (%s)\n", QIXIS_READ(scver), board);
|
|
|
|
puts("SERDES1 Reference : ");
|
|
|
|
|
|
|
|
sw = QIXIS_READ(brdcfg[2]);
|
|
|
|
#ifdef CONFIG_TARGET_LS1028ARDB
|
|
|
|
clock = (sw >> 6) & 3;
|
|
|
|
#else
|
|
|
|
clock = (sw >> 4) & 0xf;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
printf("Clock1 = %sMHz ", freq[clock]);
|
|
|
|
#ifdef CONFIG_TARGET_LS1028ARDB
|
|
|
|
clock = (sw >> 4) & 3;
|
|
|
|
#else
|
|
|
|
clock = sw & 0xf;
|
|
|
|
#endif
|
|
|
|
printf("Clock2 = %sMHz\n", freq[clock]);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|