| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * Copyright (C) 2024 PHYTEC Messtechnik GmbH |
| * Author: Wadim Egorov <w.egorov@phytec.de> |
| */ |
| |
| #include <env_internal.h> |
| #include <fdt_support.h> |
| #include <dm/ofnode.h> |
| #include <spl.h> |
| #include <malloc.h> |
| #include <asm/arch/hardware.h> |
| |
| #include "../am6_som_detection.h" |
| |
| #if IS_ENABLED(CONFIG_ENV_IS_IN_FAT) || IS_ENABLED(CONFIG_ENV_IS_IN_MMC) |
| int mmc_get_env_dev(void) |
| { |
| u32 boot_device = get_boot_device(); |
| |
| switch (boot_device) { |
| case BOOT_DEVICE_MMC1: |
| return 0; |
| case BOOT_DEVICE_MMC2: |
| return 1; |
| }; |
| |
| return CONFIG_SYS_MMC_ENV_DEV; |
| } |
| #endif |
| |
| enum env_location env_get_location(enum env_operation op, int prio) |
| { |
| u32 boot_device = get_boot_device(); |
| |
| if (prio) |
| return ENVL_UNKNOWN; |
| |
| switch (boot_device) { |
| case BOOT_DEVICE_MMC1: |
| case BOOT_DEVICE_MMC2: |
| if (CONFIG_IS_ENABLED(ENV_IS_IN_FAT)) |
| return ENVL_FAT; |
| if (CONFIG_IS_ENABLED(ENV_IS_IN_MMC)) |
| return ENVL_MMC; |
| case BOOT_DEVICE_SPI: |
| if (CONFIG_IS_ENABLED(ENV_IS_IN_SPI_FLASH)) |
| return ENVL_SPI_FLASH; |
| default: |
| return ENVL_NOWHERE; |
| }; |
| } |
| |
| #if IS_ENABLED(CONFIG_BOARD_LATE_INIT) |
| int board_late_init(void) |
| { |
| u32 boot_device = get_boot_device(); |
| |
| switch (boot_device) { |
| case BOOT_DEVICE_MMC1: |
| env_set_ulong("mmcdev", 0); |
| env_set("boot", "mmc"); |
| break; |
| case BOOT_DEVICE_MMC2: |
| env_set_ulong("mmcdev", 1); |
| env_set("boot", "mmc"); |
| break; |
| case BOOT_DEVICE_SPI: |
| env_set("boot", "spi"); |
| break; |
| case BOOT_DEVICE_ETHERNET: |
| env_set("boot", "net"); |
| break; |
| }; |
| |
| if (IS_ENABLED(CONFIG_PHYTEC_SOM_DETECTION_BLOCKS)) { |
| struct phytec_api3_element *block_element; |
| struct phytec_eeprom_data data; |
| int ret; |
| |
| ret = phytec_eeprom_data_setup(&data, 0, EEPROM_ADDR); |
| if (ret || !data.valid) |
| return 0; |
| |
| PHYTEC_API3_FOREACH_BLOCK(block_element, &data) { |
| switch (block_element->block_type) { |
| case PHYTEC_API3_BLOCK_MAC: |
| phytec_blocks_add_mac_to_env(block_element); |
| break; |
| default: |
| debug("%s: Unknown block type %i\n", __func__, |
| block_element->block_type); |
| } |
| } |
| } |
| |
| return 0; |
| } |
| #endif |
| |
| #if IS_ENABLED(CONFIG_OF_LIBFDT) && IS_ENABLED(CONFIG_OF_BOARD_SETUP) |
| static int fdt_apply_overlay_from_fit(const char *overlay_path, void *fdt) |
| { |
| u64 loadaddr; |
| ofnode node; |
| int ret; |
| |
| node = ofnode_path(overlay_path); |
| if (!ofnode_valid(node)) |
| return -FDT_ERR_NOTFOUND; |
| |
| ret = ofnode_read_u64(node, "load", &loadaddr); |
| if (ret) |
| return ret; |
| |
| return fdt_overlay_apply_verbose(fdt, (void *)loadaddr); |
| } |
| |
| static void fdt_apply_som_overlays(void *blob) |
| { |
| void *fdt_copy; |
| u32 fdt_size; |
| struct phytec_eeprom_data data; |
| int err; |
| |
| fdt_size = fdt_totalsize(blob); |
| fdt_copy = malloc(fdt_size); |
| if (!fdt_copy) |
| goto fixup_error; |
| |
| memcpy(fdt_copy, blob, fdt_size); |
| |
| err = phytec_eeprom_data_setup(&data, 0, EEPROM_ADDR); |
| if (err) |
| goto fixup_error; |
| |
| if (phytec_get_am6_rtc(&data) == 0) { |
| err = fdt_apply_overlay_from_fit("/fit-images/som-no-rtc", fdt_copy); |
| if (err) |
| goto fixup_error; |
| } |
| |
| if (phytec_get_am6_spi(&data) == PHYTEC_EEPROM_VALUE_X) { |
| err = fdt_apply_overlay_from_fit("/fit-images/som-no-spi", fdt_copy); |
| if (err) |
| goto fixup_error; |
| } |
| |
| if (phytec_get_am6_eth(&data) == 0) { |
| err = fdt_apply_overlay_from_fit("/fit-images/som-no-eth", fdt_copy); |
| if (err) |
| goto fixup_error; |
| } |
| |
| if (phytec_am6_is_qspi(&data)) { |
| err = fdt_apply_overlay_from_fit("/fit-images/som-qspi-nor", fdt_copy); |
| if (err) |
| goto fixup_error; |
| } |
| |
| memcpy(blob, fdt_copy, fdt_size); |
| |
| cleanup: |
| free(fdt_copy); |
| return; |
| |
| fixup_error: |
| pr_err("Failed to apply SoM overlays\n"); |
| goto cleanup; |
| } |
| |
| int ft_board_setup(void *blob, struct bd_info *bd) |
| { |
| fdt_apply_som_overlays(blob); |
| fdt_copy_fixed_partitions(blob); |
| |
| return 0; |
| } |
| #endif |