// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (C) 2024 PHYTEC Messtechnik GmbH * Author: Wadim Egorov */ #include #include #include #include #include #include #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