18b81a39eSGhennadi Procopciuc /*
2*88b8aa97SGhennadi Procopciuc * Copyright 2024-2025 NXP
38b81a39eSGhennadi Procopciuc *
48b81a39eSGhennadi Procopciuc * SPDX-License-Identifier: BSD-3-Clause
58b81a39eSGhennadi Procopciuc */
68b81a39eSGhennadi Procopciuc
78b81a39eSGhennadi Procopciuc #include <assert.h>
88b81a39eSGhennadi Procopciuc
9*88b8aa97SGhennadi Procopciuc #include <drivers/io/io_block.h>
108b81a39eSGhennadi Procopciuc #include <drivers/io/io_driver.h>
118b81a39eSGhennadi Procopciuc #include <drivers/io/io_fip.h>
128b81a39eSGhennadi Procopciuc #include <drivers/io/io_memmap.h>
13*88b8aa97SGhennadi Procopciuc #include <drivers/mmc.h>
14*88b8aa97SGhennadi Procopciuc #include <drivers/partition/partition.h>
15*88b8aa97SGhennadi Procopciuc #include <lib/utils.h>
16*88b8aa97SGhennadi Procopciuc #include <lib/xlat_tables/xlat_tables_v2.h>
178b81a39eSGhennadi Procopciuc #include <plat/common/platform.h>
188b81a39eSGhennadi Procopciuc #include <tools_share/firmware_image_package.h>
198b81a39eSGhennadi Procopciuc
208b81a39eSGhennadi Procopciuc #include <plat_io_storage.h>
218b81a39eSGhennadi Procopciuc
228b81a39eSGhennadi Procopciuc struct plat_io_policy {
238b81a39eSGhennadi Procopciuc uintptr_t *dev_handle;
248b81a39eSGhennadi Procopciuc uintptr_t image_spec;
258b81a39eSGhennadi Procopciuc int (*check)(const uintptr_t spec);
268b81a39eSGhennadi Procopciuc };
278b81a39eSGhennadi Procopciuc
288b81a39eSGhennadi Procopciuc static uintptr_t fip_dev_handle;
298b81a39eSGhennadi Procopciuc
30*88b8aa97SGhennadi Procopciuc static io_block_spec_t fip_mmc_spec;
318b81a39eSGhennadi Procopciuc
32*88b8aa97SGhennadi Procopciuc static uintptr_t mmc_dev_handle;
33*88b8aa97SGhennadi Procopciuc
open_mmc(const uintptr_t spec)34*88b8aa97SGhennadi Procopciuc static int open_mmc(const uintptr_t spec)
358b81a39eSGhennadi Procopciuc {
36*88b8aa97SGhennadi Procopciuc uintptr_t temp_handle;
378b81a39eSGhennadi Procopciuc int result;
388b81a39eSGhennadi Procopciuc
39*88b8aa97SGhennadi Procopciuc result = io_dev_init(mmc_dev_handle, (uintptr_t)0U);
408b81a39eSGhennadi Procopciuc if (result != 0) {
418b81a39eSGhennadi Procopciuc return result;
428b81a39eSGhennadi Procopciuc }
438b81a39eSGhennadi Procopciuc
44*88b8aa97SGhennadi Procopciuc result = io_open(mmc_dev_handle, spec, &temp_handle);
458b81a39eSGhennadi Procopciuc if (result == 0) {
468b81a39eSGhennadi Procopciuc (void)io_close(temp_handle);
478b81a39eSGhennadi Procopciuc }
488b81a39eSGhennadi Procopciuc
498b81a39eSGhennadi Procopciuc return result;
508b81a39eSGhennadi Procopciuc }
518b81a39eSGhennadi Procopciuc
open_fip(const uintptr_t spec)528b81a39eSGhennadi Procopciuc static int open_fip(const uintptr_t spec)
538b81a39eSGhennadi Procopciuc {
548b81a39eSGhennadi Procopciuc uintptr_t temp_handle = 0U;
558b81a39eSGhennadi Procopciuc int result;
568b81a39eSGhennadi Procopciuc
578b81a39eSGhennadi Procopciuc /* See if a Firmware Image Package is available */
588b81a39eSGhennadi Procopciuc result = io_dev_init(fip_dev_handle, (uintptr_t)FIP_IMAGE_ID);
598b81a39eSGhennadi Procopciuc if (result != 0) {
608b81a39eSGhennadi Procopciuc return result;
618b81a39eSGhennadi Procopciuc }
628b81a39eSGhennadi Procopciuc
638b81a39eSGhennadi Procopciuc result = io_open(fip_dev_handle, spec, &temp_handle);
648b81a39eSGhennadi Procopciuc if (result == 0) {
658b81a39eSGhennadi Procopciuc (void)io_close(temp_handle);
668b81a39eSGhennadi Procopciuc }
678b81a39eSGhennadi Procopciuc
688b81a39eSGhennadi Procopciuc return result;
698b81a39eSGhennadi Procopciuc }
708b81a39eSGhennadi Procopciuc
plat_s32g2_io_setup(void)718b81a39eSGhennadi Procopciuc void plat_s32g2_io_setup(void)
728b81a39eSGhennadi Procopciuc {
73*88b8aa97SGhennadi Procopciuc static const io_block_dev_spec_t mmc_dev_spec = {
74*88b8aa97SGhennadi Procopciuc /* It's used as temp buffer in block driver. */
75*88b8aa97SGhennadi Procopciuc .buffer = {
76*88b8aa97SGhennadi Procopciuc .offset = IO_BUFFER_BASE,
77*88b8aa97SGhennadi Procopciuc .length = IO_BUFFER_SIZE,
78*88b8aa97SGhennadi Procopciuc },
79*88b8aa97SGhennadi Procopciuc .ops = {
80*88b8aa97SGhennadi Procopciuc .read = mmc_read_blocks,
81*88b8aa97SGhennadi Procopciuc .write = mmc_write_blocks,
82*88b8aa97SGhennadi Procopciuc },
83*88b8aa97SGhennadi Procopciuc .block_size = MMC_BLOCK_SIZE,
84*88b8aa97SGhennadi Procopciuc };
858b81a39eSGhennadi Procopciuc static const io_dev_connector_t *fip_dev_con;
86*88b8aa97SGhennadi Procopciuc static const io_dev_connector_t *mmc_dev_con;
878b81a39eSGhennadi Procopciuc
88*88b8aa97SGhennadi Procopciuc partition_entry_t fip_part;
89*88b8aa97SGhennadi Procopciuc uintptr_t io_buf_base;
908b81a39eSGhennadi Procopciuc int result __unused;
91*88b8aa97SGhennadi Procopciuc size_t io_buf_size;
92*88b8aa97SGhennadi Procopciuc int ret;
938b81a39eSGhennadi Procopciuc
94*88b8aa97SGhennadi Procopciuc io_buf_base = mmc_dev_spec.buffer.offset;
95*88b8aa97SGhennadi Procopciuc io_buf_size = mmc_dev_spec.buffer.length;
96*88b8aa97SGhennadi Procopciuc
97*88b8aa97SGhennadi Procopciuc ret = mmap_add_dynamic_region(io_buf_base, io_buf_base,
98*88b8aa97SGhennadi Procopciuc io_buf_size,
99*88b8aa97SGhennadi Procopciuc MT_MEMORY | MT_RW | MT_SECURE);
100*88b8aa97SGhennadi Procopciuc if (ret != 0) {
101*88b8aa97SGhennadi Procopciuc ERROR("Failed to map the IO buffer\n");
102*88b8aa97SGhennadi Procopciuc panic();
103*88b8aa97SGhennadi Procopciuc }
104*88b8aa97SGhennadi Procopciuc
105*88b8aa97SGhennadi Procopciuc result = register_io_dev_block(&mmc_dev_con);
1068b81a39eSGhennadi Procopciuc assert(result == 0);
1078b81a39eSGhennadi Procopciuc
108*88b8aa97SGhennadi Procopciuc result = io_dev_open(mmc_dev_con, (uintptr_t)&mmc_dev_spec,
109*88b8aa97SGhennadi Procopciuc &mmc_dev_handle);
1108b81a39eSGhennadi Procopciuc assert(result == 0);
1118b81a39eSGhennadi Procopciuc
1128b81a39eSGhennadi Procopciuc result = register_io_dev_fip(&fip_dev_con);
1138b81a39eSGhennadi Procopciuc assert(result == 0);
1148b81a39eSGhennadi Procopciuc
1158b81a39eSGhennadi Procopciuc result = io_dev_open(fip_dev_con, (uintptr_t)0,
1168b81a39eSGhennadi Procopciuc &fip_dev_handle);
1178b81a39eSGhennadi Procopciuc assert(result == 0);
118*88b8aa97SGhennadi Procopciuc
119*88b8aa97SGhennadi Procopciuc ret = gpt_partition_init();
120*88b8aa97SGhennadi Procopciuc if (ret != 0) {
121*88b8aa97SGhennadi Procopciuc ERROR("Could not load MBR partition table\n");
122*88b8aa97SGhennadi Procopciuc panic();
123*88b8aa97SGhennadi Procopciuc }
124*88b8aa97SGhennadi Procopciuc
125*88b8aa97SGhennadi Procopciuc fip_part = get_partition_entry_list()->list[FIP_PART];
126*88b8aa97SGhennadi Procopciuc fip_mmc_spec.offset = fip_part.start;
127*88b8aa97SGhennadi Procopciuc fip_mmc_spec.length = fip_part.length;
1288b81a39eSGhennadi Procopciuc }
1298b81a39eSGhennadi Procopciuc
plat_get_image_source(unsigned int image_id,uintptr_t * dev_handle,uintptr_t * image_spec)1308b81a39eSGhennadi Procopciuc int plat_get_image_source(unsigned int image_id, uintptr_t *dev_handle,
1318b81a39eSGhennadi Procopciuc uintptr_t *image_spec)
1328b81a39eSGhennadi Procopciuc {
133*88b8aa97SGhennadi Procopciuc static const io_block_spec_t mbr_spec = {
134*88b8aa97SGhennadi Procopciuc .offset = 0,
135*88b8aa97SGhennadi Procopciuc .length = PLAT_PARTITION_BLOCK_SIZE,
1368b81a39eSGhennadi Procopciuc };
1378b81a39eSGhennadi Procopciuc
1388b81a39eSGhennadi Procopciuc static const io_uuid_spec_t bl31_uuid_spec = {
1398b81a39eSGhennadi Procopciuc .uuid = UUID_EL3_RUNTIME_FIRMWARE_BL31,
1408b81a39eSGhennadi Procopciuc };
1418b81a39eSGhennadi Procopciuc
1428b81a39eSGhennadi Procopciuc static const io_uuid_spec_t bl33_uuid_spec = {
1438b81a39eSGhennadi Procopciuc .uuid = UUID_NON_TRUSTED_FIRMWARE_BL33,
1448b81a39eSGhennadi Procopciuc };
1458b81a39eSGhennadi Procopciuc
146*88b8aa97SGhennadi Procopciuc static const struct plat_io_policy policies[GPT_IMAGE_ID + 1] = {
1478b81a39eSGhennadi Procopciuc [FIP_IMAGE_ID] = {
148*88b8aa97SGhennadi Procopciuc .dev_handle = &mmc_dev_handle,
149*88b8aa97SGhennadi Procopciuc .image_spec = (uintptr_t)&fip_mmc_spec,
150*88b8aa97SGhennadi Procopciuc .check = open_mmc,
1518b81a39eSGhennadi Procopciuc },
1528b81a39eSGhennadi Procopciuc [BL31_IMAGE_ID] = {
1538b81a39eSGhennadi Procopciuc .dev_handle = &fip_dev_handle,
1548b81a39eSGhennadi Procopciuc .image_spec = (uintptr_t)&bl31_uuid_spec,
1558b81a39eSGhennadi Procopciuc .check = open_fip,
1568b81a39eSGhennadi Procopciuc },
1578b81a39eSGhennadi Procopciuc [BL33_IMAGE_ID] = {
1588b81a39eSGhennadi Procopciuc .dev_handle = &fip_dev_handle,
1598b81a39eSGhennadi Procopciuc .image_spec = (uintptr_t)&bl33_uuid_spec,
1608b81a39eSGhennadi Procopciuc .check = open_fip,
1618b81a39eSGhennadi Procopciuc },
162*88b8aa97SGhennadi Procopciuc [GPT_IMAGE_ID] = {
163*88b8aa97SGhennadi Procopciuc .dev_handle = &mmc_dev_handle,
164*88b8aa97SGhennadi Procopciuc .image_spec = (uintptr_t)&mbr_spec,
165*88b8aa97SGhennadi Procopciuc .check = open_mmc,
166*88b8aa97SGhennadi Procopciuc },
1678b81a39eSGhennadi Procopciuc };
1688b81a39eSGhennadi Procopciuc const struct plat_io_policy *policy;
1698b81a39eSGhennadi Procopciuc int result;
1708b81a39eSGhennadi Procopciuc
1718b81a39eSGhennadi Procopciuc assert(image_id < ARRAY_SIZE(policies));
1728b81a39eSGhennadi Procopciuc
1738b81a39eSGhennadi Procopciuc policy = &policies[image_id];
1748b81a39eSGhennadi Procopciuc result = policy->check(policy->image_spec);
1758b81a39eSGhennadi Procopciuc assert(result == 0);
1768b81a39eSGhennadi Procopciuc
1778b81a39eSGhennadi Procopciuc *image_spec = policy->image_spec;
1788b81a39eSGhennadi Procopciuc *dev_handle = *policy->dev_handle;
1798b81a39eSGhennadi Procopciuc
1808b81a39eSGhennadi Procopciuc return result;
1818b81a39eSGhennadi Procopciuc }
182