| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * Copyright 2022 NXP |
| */ |
| |
| #include <command.h> |
| #include <cpu_func.h> |
| #include <hang.h> |
| #include <image.h> |
| #include <init.h> |
| #include <log.h> |
| #include <spl.h> |
| #include <asm/global_data.h> |
| #include <asm/io.h> |
| #include <asm/arch/imx93_pins.h> |
| #include <asm/arch/mu.h> |
| #include <asm/arch/clock.h> |
| #include <asm/arch/sys_proto.h> |
| #include <asm/mach-imx/boot_mode.h> |
| #include <asm/mach-imx/mxc_i2c.h> |
| #include <asm/arch-mx7ulp/gpio.h> |
| #include <asm/mach-imx/ele_api.h> |
| #include <asm/mach-imx/syscounter.h> |
| #include <asm/sections.h> |
| #include <dm/uclass.h> |
| #include <dm/device.h> |
| #include <dm/uclass-internal.h> |
| #include <dm/device-internal.h> |
| #include <linux/delay.h> |
| #include <asm/arch/clock.h> |
| #include <asm/arch/ccm_regs.h> |
| #include <asm/arch/ddr.h> |
| #include <power/pmic.h> |
| #include <power/pca9450.h> |
| #include <asm/arch/trdc.h> |
| |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| int spl_board_boot_device(enum boot_device boot_dev_spl) |
| { |
| return BOOT_DEVICE_BOOTROM; |
| } |
| |
| void spl_board_init(void) |
| { |
| int ret; |
| |
| ret = ele_start_rng(); |
| if (ret) |
| printf("Fail to start RNG: %d\n", ret); |
| |
| puts("Normal Boot\n"); |
| } |
| |
| extern struct dram_timing_info dram_timing_1866mts; |
| void spl_dram_init(void) |
| { |
| struct dram_timing_info *ptiming = &dram_timing; |
| |
| if (is_voltage_mode(VOLT_LOW_DRIVE)) |
| ptiming = &dram_timing_1866mts; |
| |
| printf("DDR: %uMTS\n", ptiming->fsp_msg[0].drate); |
| ddr_init(ptiming); |
| } |
| |
| #if CONFIG_IS_ENABLED(DM_PMIC_PCA9450) |
| int power_init_board(void) |
| { |
| struct udevice *dev; |
| int ret; |
| unsigned int val = 0, buck_val; |
| |
| ret = pmic_get("pmic@25", &dev); |
| if (ret == -ENODEV) { |
| puts("No pca9450@25\n"); |
| return 0; |
| } |
| if (ret != 0) |
| return ret; |
| |
| /* BUCKxOUT_DVS0/1 control BUCK123 output */ |
| pmic_reg_write(dev, PCA9450_BUCK123_DVS, 0x29); |
| |
| /* enable DVS control through PMIC_STBY_REQ */ |
| pmic_reg_write(dev, PCA9450_BUCK1CTRL, 0x59); |
| |
| ret = pmic_reg_read(dev, PCA9450_PWR_CTRL); |
| if (ret < 0) |
| return ret; |
| |
| val = ret; |
| |
| if (is_voltage_mode(VOLT_LOW_DRIVE)) { |
| buck_val = 0x0c; /* 0.8v for Low drive mode */ |
| printf("PMIC: Low Drive Voltage Mode\n"); |
| } else if (is_voltage_mode(VOLT_NOMINAL_DRIVE)) { |
| buck_val = 0x10; /* 0.85v for Nominal drive mode */ |
| printf("PMIC: Nominal Voltage Mode\n"); |
| } else { |
| buck_val = 0x14; /* 0.9v for Over drive mode */ |
| printf("PMIC: Over Drive Voltage Mode\n"); |
| } |
| |
| if (val & PCA9450_REG_PWRCTRL_TOFF_DEB) { |
| pmic_reg_write(dev, PCA9450_BUCK1OUT_DVS0, buck_val); |
| pmic_reg_write(dev, PCA9450_BUCK3OUT_DVS0, buck_val); |
| } else { |
| pmic_reg_write(dev, PCA9450_BUCK1OUT_DVS0, buck_val + 0x4); |
| pmic_reg_write(dev, PCA9450_BUCK3OUT_DVS0, buck_val + 0x4); |
| } |
| |
| if (IS_ENABLED(CONFIG_IMX93_EVK_LPDDR4X)) { |
| /* Set VDDQ to 1.1V from buck2 */ |
| pmic_reg_write(dev, PCA9450_BUCK2OUT_DVS0, 0x28); |
| } |
| |
| /* set standby voltage to 0.65v */ |
| if (val & PCA9450_REG_PWRCTRL_TOFF_DEB) |
| pmic_reg_write(dev, PCA9450_BUCK1OUT_DVS1, 0x0); |
| else |
| pmic_reg_write(dev, PCA9450_BUCK1OUT_DVS1, 0x4); |
| |
| /* I2C_LT_EN*/ |
| pmic_reg_write(dev, 0xa, 0x3); |
| return 0; |
| } |
| #endif |
| |
| void board_init_f(ulong dummy) |
| { |
| int ret; |
| |
| /* Clear the BSS. */ |
| memset(__bss_start, 0, __bss_end - __bss_start); |
| |
| timer_init(); |
| |
| arch_cpu_init(); |
| |
| board_early_init_f(); |
| |
| spl_early_init(); |
| |
| preloader_console_init(); |
| |
| ret = imx9_probe_mu(); |
| if (ret) { |
| printf("Fail to init Sentinel API\n"); |
| } else { |
| debug("SOC: 0x%x\n", gd->arch.soc_rev); |
| debug("LC: 0x%x\n", gd->arch.lifecycle); |
| } |
| |
| clock_init_late(); |
| |
| power_init_board(); |
| |
| if (!is_voltage_mode(VOLT_LOW_DRIVE)) |
| set_arm_clk(get_cpu_speed_grade_hz()); |
| |
| /* Init power of mix */ |
| soc_power_init(); |
| |
| /* Setup TRDC for DDR access */ |
| trdc_init(); |
| |
| /* DDR initialization */ |
| spl_dram_init(); |
| |
| /* Put M33 into CPUWAIT for following kick */ |
| ret = m33_prepare(); |
| if (!ret) |
| printf("M33 prepare ok\n"); |
| |
| board_init_r(NULL, 0); |
| } |