Newer
Older
arm-trusted-firmware / drivers / brcm / scp.c
@Sheetal Tigadoli Sheetal Tigadoli on 3 Apr 2020 2 KB Add BL2 support for Broadcom stingray platform
/*
 * Copyright (c) 2017 - 2020, Broadcom
 *
 * SPDX-License-Identifier: BSD-3-Clause
 */

#include <string.h>

#include <arch_helpers.h>
#include <common/debug.h>

/* MCU binary image structure: <header> <data>
 *
 * Header structure:
 * <magic-start>
 * <num-sections>
 * {<src-offset> <src-size> <dst-addr>}*
 * <magic-end>
 *
 * MCU data (<data>) consists of several sections of code/data, to be
 *             installed (copied) into MCU memories.
 * Header (<header>) gives information about sections contained in <data>.
 *
 * The installer code iterates over sections in MCU binary.
 * For each section, it copies the section into MCU memory.
 *
 * The header contains:
 *	 - <magic-start> - 32-bit magic number to mark header start
 *	 - <num-sections> - number of sections in <data>
 *	 - <num-sections> tuples. Each tuple describes a section.
 *			 A tuple contains three 32-bit words.
 *	 - <magic-end> - 32-bit magic number to mark header end
 *
 * Each section is describes by a tuple, consisting of three 32-bit words:
 *	 - offset of section within MCU binary (relative to beginning of <data>)
 *	 - section size (in bytes) in MCU binary
 *	 - target address (in MCU memory). Section is copied to this location.
 *
 * All fields are 32-bit unsigned integers in little endian format.
 * All sizes are assumed to be 32-bit aligned.
 */

#define SCP_BIN_HEADER_MAGIC_START 0xfa587D01
#define SCP_BIN_HEADER_MAGIC_END 0xf3e06a85

int download_scp_patch(void *image, unsigned int image_size)
{
	unsigned int *pheader = (unsigned int *)(image);
	unsigned int header_size;
	unsigned char *pdata;
	void *dest;
	unsigned int num_sections;
	unsigned int section_src_offset;
	unsigned int section_size;

	if (pheader && (pheader[0] != SCP_BIN_HEADER_MAGIC_START)) {
		ERROR("SCP: Could not find SCP header.\n");
		return -1;
	}

	num_sections = pheader[1];
	INFO("...Number of sections: %d\n", num_sections);
	header_size = 4 * (1 + 1 + 3 * num_sections + 1);

	if (image_size < header_size) {
		ERROR("SCP: Wrong size.\n");
		return -1;
	}

	if (*(pheader + header_size/4 - 1) != SCP_BIN_HEADER_MAGIC_END) {
		ERROR("SCP: Could not find SCP footer.\n");
		return -1;
	}

	VERBOSE("SCP image header validated successfully\n");
	pdata = (unsigned char *)pheader + header_size;

	for (pheader += 2; num_sections > 0; num_sections--) {

		section_src_offset = pheader[0];
		section_size = pheader[1];
		dest = (void *)(unsigned long)pheader[2];

		INFO("section: src:0x%x, size:%d, dst:0x%x\n",
				section_src_offset, section_size, pheader[2]);

		if ((section_src_offset + section_size) > image_size) {
			ERROR("SCP: Section points to outside of patch.\n");
			return -1;
		}

		/* copy from source to target section */
		memcpy(dest, pdata + section_src_offset, section_size);
		flush_dcache_range((uintptr_t)dest, section_size);

		/* next section */
		pheader += 3;
	}
	return 0;
}