xref: /rk3399_ARM-atf/plat/nxp/s32/s32g274ardb2/plat_io_storage.c (revision d0ce1ac58476f546ee87233112b59f4c73e11228)
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