Skip to content
Snippets Groups Projects
rkcommon.c 3.2 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*
     * (C) Copyright 2015 Google,  Inc
     * Written by Simon Glass <sjg@chromium.org>
     *
     * SPDX-License-Identifier:	GPL-2.0+
     *
     * Helper functions for Rockchip images
     */
    
    #include "imagetool.h"
    #include <image.h>
    #include <rc4.h>
    #include "mkimage.h"
    #include "rkcommon.h"
    
    enum {
    	RK_SIGNATURE		= 0x0ff0aa55,
    };
    
    /**
     * struct header0_info - header block for boot ROM
     *
     * This is stored at SD card block 64 (where each block is 512 bytes, or at
     * the start of SPI flash. It is encoded with RC4.
     *
     * @signature:		Signature (must be RKSD_SIGNATURE)
     * @disable_rc4:	0 to use rc4 for boot image,  1 to use plain binary
    
     * @init_offset:	Offset in blocks of the SPL code from this header
    
     *			block. E.g. 4 means 2KB after the start of this header.
     * Other fields are not used by U-Boot
     */
    struct header0_info {
    	uint32_t signature;
    	uint8_t reserved[4];
    	uint32_t disable_rc4;
    
    	uint16_t init_offset;
    	uint8_t reserved1[492];
    	uint16_t init_size;
    	uint16_t init_boot_size;
    
    	uint8_t reserved2[2];
    };
    
    
    /**
     * struct spl_info - spl info for each chip
     *
     * @imagename:		Image name(passed by "mkimage -n")
     * @spl_hdr:		Boot ROM requires a 4-bytes spl header
     * @spl_size:		Spl size(include extra 4-bytes spl header)
     */
    struct spl_info {
    	const char *imagename;
    	const char *spl_hdr;
    	const uint32_t spl_size;
    };
    
    static struct spl_info spl_infos[] = {
    	{ "rk3036", "RK30", 0x1000 },
    	{ "rk3288", "RK32", 0x8000 },
    
    	{ "rk3399", "RK33", 0x20000 },
    
    static unsigned char rc4_key[16] = {
    	124, 78, 3, 4, 85, 5, 9, 7,
    	45, 44, 123, 56, 23, 13, 23, 17
    };
    
    
    static struct spl_info *rkcommon_get_spl_info(char *imagename)
    {
    	int i;
    
    	for (i = 0; i < ARRAY_SIZE(spl_infos); i++)
    		if (!strncmp(imagename, spl_infos[i].imagename, 6))
    			return spl_infos + i;
    
    	return NULL;
    }
    
    int rkcommon_check_params(struct image_tool_params *params)
    {
    	int i;
    
    	if (rkcommon_get_spl_info(params->imagename) != NULL)
    		return 0;
    
    	fprintf(stderr, "ERROR: imagename (%s) is not supported!\n",
    		strlen(params->imagename) > 0 ? params->imagename : "NULL");
    
    	fprintf(stderr, "Available imagename:");
    	for (i = 0; i < ARRAY_SIZE(spl_infos); i++)
    		fprintf(stderr, "\t%s", spl_infos[i].imagename);
    	fprintf(stderr, "\n");
    
    	return -1;
    }
    
    const char *rkcommon_get_spl_hdr(struct image_tool_params *params)
    {
    	struct spl_info *info = rkcommon_get_spl_info(params->imagename);
    
    	/*
    	 * info would not be NULL, because of we checked params before.
    	 */
    	return info->spl_hdr;
    }
    
    int rkcommon_get_spl_size(struct image_tool_params *params)
    {
    	struct spl_info *info = rkcommon_get_spl_info(params->imagename);
    
    	/*
    	 * info would not be NULL, because of we checked params before.
    	 */
    	return info->spl_size;
    }
    
    int rkcommon_set_header(void *buf, uint file_size,
    			struct image_tool_params *params)
    
    {
    	struct header0_info *hdr;
    
    
    	if (file_size > rkcommon_get_spl_size(params))
    
    	memset(buf,  '\0', RK_INIT_OFFSET * RK_BLK_SIZE);
    
    	hdr = (struct header0_info *)buf;
    	hdr->signature = RK_SIGNATURE;
    	hdr->disable_rc4 = 1;
    
    	hdr->init_offset = RK_INIT_OFFSET;
    
    	hdr->init_size = (file_size + RK_BLK_SIZE - 1) / RK_BLK_SIZE;
    	hdr->init_size = (hdr->init_size + 3) & ~3;
    	hdr->init_boot_size = hdr->init_size + RK_MAX_BOOT_SIZE / RK_BLK_SIZE;
    
    
    	rc4_encode(buf, RK_BLK_SIZE, rc4_key);
    
    	return 0;
    }