sdk-hwV1.3/lichee/brandy-2.0/u-boot-2018/sprite/sprite_download.c

1137 lines
35 KiB
C

/*
* (C) Copyright 2007-2013
* Allwinner Technology Co., Ltd. <www.allwinnertech.com>
* Jerry Wang <wangflord@allwinnertech.com>
*
* See file CREDITS for list of people who contributed to this
* project.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of
* the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*/
#include <config.h>
#include <common.h>
#include <private_toc.h>
#include <private_boot0.h>
#include <private_uboot.h>
#include <sunxi_mbr.h>
#include <sunxi_nand.h>
#include <sunxi_flash.h>
#include <sunxi_board.h>
#include <malloc.h>
#include <sprite_verify.h>
#include <part_efi.h>
#include <fdt_support.h>
#include <sunxi_image_verifier.h>
#include <asm/arch/rtc.h>
#include <sys_partition.h>
DECLARE_GLOBAL_DATA_PTR;
#define GPT_BUFF_SIZE (8*1024)
int sunxi_mbr_convert_to_gpt(void *sunxi_mbr_buf, char *gpt_buf,int storage_type);
int download_standard_gpt(void *sunxi_mbr_buf, size_t buf_size, int storage_type);
extern int sunxi_set_secure_mode(void);
static void dump_dram_para(void* dram, uint size)
{
int i;
uint *addr = (uint *)dram;
for(i = 0; i < size; i++) {
printf("dram para[%d] = %x\n", i, addr[i]);
}
}
static void prepare_backup_gpt_header(gpt_header *gpt_h)
{
uint32_t calc_crc32;
uint64_t val;
/* recalculate the values for the Backup GPT Header */
val = le64_to_cpu(gpt_h->my_lba);
gpt_h->my_lba = gpt_h->alternate_lba; /*total_sectors - 1*/
gpt_h->alternate_lba = cpu_to_le64(val);
gpt_h->partition_entry_lba =
cpu_to_le64(le64_to_cpu(gpt_h->last_usable_lba) + 1);
gpt_h->header_crc32 = 0;
calc_crc32 = crc32(0,(const unsigned char *)gpt_h,
le32_to_cpu(gpt_h->header_size));
gpt_h->header_crc32 = cpu_to_le32(calc_crc32);
}
#ifdef CONFIG_SUNXI_COMPARE_IMAGE_AND_FLASH
static int compare_image_and_flash_size(u8 *buf)
{
sunxi_mbr_t *mbr_info = (sunxi_mbr_t *)buf;
sunxi_partition *part_info;
u32 i;
char buffer[32];
uint logic_size = 0;
uint flash_size = 0;
for (part_info = mbr_info->array, i = 0; i < mbr_info->PartCount;
i++, part_info++) {
memset(buffer, 0, 32);
memcpy(buffer, part_info->name, 16);
//NOTE : Use UDISK addrlo as the size of image
if ((!strcmp(buffer, "UDISK")) || (!strcmp(buffer, CONFIG_LAST_PARTITION_NAME))) {
logic_size = part_info->addrlo;
}
}
flash_size = sunxi_flash_size();
pr_msg("logic size : 0x%x sector(512 byte)\n", logic_size);
pr_msg("flash size : 0x%x sector(512 byte)\n", flash_size);
if (flash_size == 0) {
pr_msg("flash size is zero, ignore check\n");
return 0;
}
if (flash_size > logic_size) {
return 0;
} else {
pr_err("error : logic_size biger than flash_size\n");
return -1;
}
}
#endif
int sunxi_sprite_download_mbr(void *buffer, uint buffer_size)
{
int ret = 0;
int storage_type = 0;
int mbr_num = SUNXI_MBR_COPY_NUM;
if (get_boot_storage_type() == STORAGE_NOR) {
mbr_num = 1;
}
if(buffer_size != (SUNXI_MBR_SIZE * mbr_num)) {
printf("the mbr size is bad\n");
return -1;
}
#ifdef CONFIG_SUNXI_COMPARE_IMAGE_AND_FLASH
if (compare_image_and_flash_size(buffer)) {
pr_err("please check you image and flash size!\n");
pr_err("=============Exit burning now=============\n");
return -1;
}
#endif
storage_type = get_boot_storage_type();
if ((storage_type == STORAGE_NAND) && (sunxi_sprite_init(0))) {
return -2;
}
/*write GPT Table*/
ret = download_standard_gpt(buffer,buffer_size,storage_type);
if(ret) {
return -3;
}
pr_msg("update partition map\n");
sunxi_probe_partition_map();
return ret;
}
int sunxi_sprite_download_uboot(void *buffer, int production_media, int generate_checksum)
{
u32 length = 0;
sbrom_toc1_head_info_t *toc1 = (sbrom_toc1_head_info_t *)buffer;
if(toc1->magic != TOC_MAIN_INFO_MAGIC)
{
printf("sunxi sprite: toc magic is error\n");
printf("need %s image\n", gd->bootfile_mode == SUNXI_BOOT_FILE_TOC ? "secure" : "normal");
return -1;
}
length = toc1->valid_len;
if(generate_checksum) {
toc1->add_sum = sunxi_sprite_generate_checksum(buffer,
toc1->valid_len,toc1->add_sum);
}
printf("uboot size = 0x%x\n", length);
printf("storage type = %d\n", production_media);
#if defined(CONFIG_SUNXI_BURN_ROTPK_ON_SPRITE) || \
defined(CONFIG_SUNXI_ROTPK_BURN_ENABLE_BY_TOOL)
sunxi_verify_preserve_toc1(buffer);
#endif
return sunxi_sprite_download_toc(buffer, length, production_media);
}
int sunxi_sprite_upload_uboot(void *buffer, uint len)
{
return sunxi_sprite_upload_toc(buffer, len);
}
#if defined(CONFIG_SUNXI_ROTPK_BURN_ENABLE_BY_TOOL)
int set_rotpk_flag(unsigned char *flag)
{
if ((uboot_spare_head.boot_data.func_mask &
UBOOT_FUNC_MASK_BIT_BURN_ROTPK) !=
UBOOT_FUNC_MASK_BIT_BURN_ROTPK) {
printf("tool did not set rotpk burn flag, skip rotpk burn\n");
*flag = 0;
return 0;
}
if (gd->securemode == SUNXI_NORMAL_MODE) {
printf("normal mode, don't need set rotpk flag\n");
*flag = 0;
return 0;
}
printf("set rotpk flag to toc0 header\n");
*flag = 1;
return 0;
}
#endif
int download_secure_boot0(void *buffer, int production_media)
{
toc0_private_head_t *toc0 = (toc0_private_head_t *)buffer;
sbrom_toc0_config_t *toc0_config = NULL;
int ret = 0;
if (toc0->items_nr == 3)
toc0_config = (sbrom_toc0_config_t *)(buffer + 0xa0);
else
toc0_config = (sbrom_toc0_config_t *)(buffer + 0x80);
debug("%s\n", (char *)toc0->name);
if(strncmp((const char *)toc0->name, TOC0_MAGIC, MAGIC_SIZE))
{
printf("sunxi sprite: toc0 magic is error, need secure image\n");
return -1;
}
if(sunxi_sprite_verify_checksum(buffer, toc0->length, toc0->check_sum))
{
printf("sunxi sprite: toc0 checksum is error\n");
return -1;
}
#ifdef CONFIG_SUNXI_TURNNING_FLASH
//update flash param
if(!production_media)
{
#ifdef CONFIG_SUNXI_UBIFS
if (nand_use_ubi()) {
ubi_nand_get_flash_info(
(void *)toc0_config->storage_data,
STORAGE_BUFFER_SIZE);
}
#elif defined(CONFIG_SUNXI_COMM_NAND_V1) || defined(CONFIG_SUNXI_COMM_NAND)
nand_uboot_get_flash_info(
(void *)toc0_config->storage_data,
STORAGE_BUFFER_SIZE);
#else
printf("not define RAWNAND & SPINAND\n");
return -1;
#endif
} else {
#ifdef CONFIG_SUNXI_SDMMC
//storage_data[384]; // 0-159:nand info 160-255:card info
if (production_media == STORAGE_EMMC) {
if (mmc_write_info(2, (void *)(toc0_config->storage_data+160), 384-160)) {
printf("add sdmmc2 gpio info fail!\n");
return -1;
}
} else if (production_media == STORAGE_EMMC3) {
if (mmc_write_info(3, (void *)(toc0_config->storage_data+160), 384-160)) {
printf("add sdmmc3 gpio info fail!\n");
return -1;
}
} else if (production_media == STORAGE_EMMC0) {
if (mmc_write_info(0, (void *)(toc0_config->storage_data+160), 384-160)) {
printf("add sdmmc0 gpio info fail!\n");
return -1;
}
}
#endif
}
#endif
#ifdef CONFIG_SUNXI_TURNNING_DRAM
//update dram param
if (uboot_spare_head.boot_data.work_mode == WORK_MODE_CARD_PRODUCT ||
(uboot_spare_head.boot_data.work_mode == WORK_MODE_BOOT && get_boot_dram_update_flag())) {
memcpy((void *)toc0_config->dram_para, (void *)(uboot_spare_head.boot_data.dram_para), 32 * 4);
/*update dram flag*/
set_boot_dram_update_flag(toc0_config->dram_para);
} else if (uboot_spare_head.boot_data.work_mode == WORK_MODE_UDISK_UPDATE ||
uboot_spare_head.boot_data.work_mode == WORK_MODE_CARD_UPDATE ||
uboot_spare_head.boot_data.work_mode == WORK_MODE_SPRITE_RECOVERY ||
uboot_spare_head.boot_data.work_mode == WORK_MODE_BOOT) {
printf("skip memcpy dram para for work_mode: %d \n", uboot_spare_head.boot_data.work_mode);
} else {
memcpy((void *)toc0_config->dram_para, (void *)CONFIG_DRAM_PARA_ADDR, 32 * 4);
/*update dram flag*/
set_boot_dram_update_flag(toc0_config->dram_para);
}
#endif
dump_dram_para(toc0_config->dram_para, 32);
#if defined(CONFIG_SUNXI_ROTPK_BURN_ENABLE_BY_TOOL)
set_rotpk_flag(&toc0_config->rotpk_flag);
#endif
/* regenerate check sum */
toc0->check_sum = sunxi_sprite_generate_checksum(buffer, toc0->length, toc0->check_sum);
if (sunxi_sprite_verify_checksum(buffer, toc0->length, toc0->check_sum)) {
printf("sunxi sprite: boot0 checksum is error\n");
return -1;
}
printf("storage type = %d\n", production_media);
ret = sunxi_sprite_download_spl(buffer, toc0->length, production_media);
if (!ret) {
printf("burn sboot ok, set secure bit\n");
ret = sunxi_set_secure_mode();
}
return ret;
}
int download_normal_boot0(void *buffer, int production_media)
{
boot0_file_head_t *boot0 = (boot0_file_head_t *)buffer;
debug("%s\n", boot0->boot_head.magic);
if (strncmp((const char *)boot0->boot_head.magic, BOOT0_MAGIC, MAGIC_SIZE)) {
printf("sunxi sprite: boot0 magic is error\n");
return -1;
}
if (sunxi_sprite_verify_checksum(buffer, boot0->boot_head.length, boot0->boot_head.check_sum)) {
printf("sunxi sprite: boot0 checksum is error\n");
return -1;
}
#ifdef CONFIG_SUNXI_TURNNING_FLASH
if(!production_media) {
#ifdef CONFIG_SUNXI_UBIFS
if (nand_use_ubi()) {
ubi_nand_get_flash_info(
(void *)boot0->prvt_head.storage_data,
STORAGE_BUFFER_SIZE);
}
#elif defined(CONFIG_SUNXI_COMM_NAND_V1) || defined(CONFIG_SUNXI_COMM_NAND)
nand_uboot_get_flash_info(
(void *)boot0->prvt_head.storage_data,
STORAGE_BUFFER_SIZE);
#else
printf("not define RAWNAND & SPINAND\n");
return -1;
#endif
} else {
#ifdef CONFIG_SUNXI_SDMMC
if (production_media == STORAGE_EMMC) {
if (mmc_write_info(2, (void *)boot0->prvt_head.storage_data, STORAGE_BUFFER_SIZE)) {
printf("add sdmmc2 private info fail!\n");
return -1;
}
} else if (production_media == STORAGE_EMMC3) {
if (mmc_write_info(3, (void *)boot0->prvt_head.storage_data, STORAGE_BUFFER_SIZE)) {
printf("add sdmmc3 private info fail!\n");
return -1;
}
} else if (production_media == STORAGE_EMMC0) {
if (mmc_write_info(0, (void *)boot0->prvt_head.storage_data, STORAGE_BUFFER_SIZE)) {
printf("add sdmmc0 private info fail!\n");
return -1;
}
}
#endif
}
#endif
#ifdef CONFIG_SUNXI_TURNNING_DRAM
if (uboot_spare_head.boot_data.work_mode == WORK_MODE_CARD_PRODUCT ||
(uboot_spare_head.boot_data.work_mode == WORK_MODE_BOOT && get_boot_dram_update_flag())) {
memcpy((void *)&boot0->prvt_head.dram_para, (void *)(uboot_spare_head.boot_data.dram_para), 32 * 4);
/*update dram flag*/
set_boot_dram_update_flag(boot0->prvt_head.dram_para);
} else if (uboot_spare_head.boot_data.work_mode == WORK_MODE_UDISK_UPDATE ||
uboot_spare_head.boot_data.work_mode == WORK_MODE_CARD_UPDATE ||
uboot_spare_head.boot_data.work_mode == WORK_MODE_SPRITE_RECOVERY ||
uboot_spare_head.boot_data.work_mode == WORK_MODE_BOOT) {
printf("skip memcpy dram para for work_mode: %d \n", uboot_spare_head.boot_data.work_mode);
} else {
memcpy((void *)boot0->prvt_head.dram_para, (void *)CONFIG_DRAM_PARA_ADDR, 32 * 4);
/*update dram flag*/
set_boot_dram_update_flag(boot0->prvt_head.dram_para);
}
#endif
dump_dram_para(boot0->prvt_head.dram_para,32);
/* regenerate check sum */
boot0->boot_head.check_sum = sunxi_sprite_generate_checksum(buffer, boot0->boot_head.length, boot0->boot_head.check_sum);
if(sunxi_sprite_verify_checksum(buffer, boot0->boot_head.length, boot0->boot_head.check_sum)) {
printf("sunxi sprite: boot0 checksum is error\n");
return -1;
}
printf("storage type = %d\n", production_media);
return sunxi_sprite_download_spl(buffer, boot0->boot_head.length, production_media);
}
int sunxi_sprite_download_boot0(void *buffer, int production_media)
{
if(gd->bootfile_mode == SUNXI_BOOT_FILE_NORMAL || gd->bootfile_mode == SUNXI_BOOT_FILE_PKG){
return download_normal_boot0(buffer, production_media);
} else {
return download_secure_boot0(buffer, production_media);
}
/* usb dma recv time out maybe enter usb product twice
* brom enter fel mode or boot0 enter fel mode
*/
rtc_set_bootmode_flag(0);
}
void sunxi_get_logical_offset_param(int storage_type, u32 *logic_offset,
int *total_sectors)
{
/*
* for nand, no phyread available, so this is not relevant
* for mmc, part offset is physical offset, offset should be taken care in start value, refer by logic_offset here
* for nor, part offset is logical offset, offset should be taken care in end value, refer by total_sectors here
*/
if (storage_type == STORAGE_EMMC || storage_type == STORAGE_EMMC3 ||
storage_type == STORAGE_SD || storage_type == STORAGE_EMMC0) {
*logic_offset = sunxi_flashmap_logical_offset(FLASHMAP_SDMMC, LINUX_LOGIC_OFFSET);
} else {
*logic_offset = 0;
}
*total_sectors = sunxi_sprite_size();
#ifdef CONFIG_SUNXI_SPINOR
if (storage_type == STORAGE_NOR) {
#ifdef CONFIG_SUNXI_RTOS
*total_sectors -= sunxi_flashmap_logical_offset(FLASHMAP_SPI_NOR, RTOS_LOGIC_OFFSET);
#else
*total_sectors -= sunxi_flashmap_logical_offset(FLASHMAP_SPI_NOR, LINUX_LOGIC_OFFSET);
#endif
}
#endif
}
int gpt_convert_to_sunxi_mbr(void *sunxi_mbr_buf, char *gpt_buf, int storage_type)
{
u32 data_len = 0;
char *pbuf = gpt_buf;
gpt_entry *pgpt_entry = NULL;
char *gpt_entry_start = NULL;
int PartCount = 0;
int pos = 0;
int i, j;
int crc32_total;
int mbr_size = 0;
u64 start_sector;
int total_sectors;
u32 logic_offset = 0;
/* mbr_size = 256; [> hardcode, TODO: fixit <] */
/* mbr_size = mbr_size * (1024/512); */
sunxi_mbr_t *sunxi_mbr = (sunxi_mbr_t *)sunxi_mbr_buf;
memset(sunxi_mbr, 0, sizeof(sunxi_mbr_t));
sunxi_mbr->version = 0x00000200;
memcpy(sunxi_mbr->magic, SUNXI_MBR_MAGIC, 8);
sunxi_mbr->copy = 1;
data_len = 0;
data_len += 512; /* 0 to gpt->head */
data_len += 512; /* gpt->head to gpt_entry */
gpt_entry_start = (pbuf + data_len);
while (1) {
pgpt_entry = (gpt_entry *)(gpt_entry_start + (pos)*GPT_ENTRY_SIZE);
if ((long)(pgpt_entry->starting_lba) > 0) {
printf("gpt pos:%d s:0x%llx e:0x%llx\n", pos, pgpt_entry->starting_lba, pgpt_entry->ending_lba);
PartCount++;
} else
break;
pos++;
}
sunxi_mbr->PartCount = PartCount;
printf("PartCount = %d\n", PartCount);
sunxi_mbr->PartCount += 1; //need UDISK
printf("fix, now PartCount = %d\n", sunxi_mbr->PartCount);
for (i = sunxi_mbr->PartCount - 1; i >= 0; i--) {
/* udisk is the first part */
/* pos = (i == sunxi_mbr->PartCount-1) ? 0: i+1; */
pos = i;
pgpt_entry = (gpt_entry *)(gpt_entry_start + (pos)*GPT_ENTRY_SIZE);
sunxi_mbr->array[i].lenhi = ((pgpt_entry->ending_lba - pgpt_entry->starting_lba + 1) >> 32) & 0xffffffff;
sunxi_mbr->array[i].lenlo = ((pgpt_entry->ending_lba - pgpt_entry->starting_lba + 1) >> 0) & 0xffffffff;
if (i == sunxi_mbr->PartCount - 1) {
sunxi_mbr->array[i].lenhi = 0;
sunxi_mbr->array[i].lenlo = 0;
};
printf("%d starting_lba:%llx ending_lba:%llx\n", i, pgpt_entry->starting_lba, pgpt_entry->ending_lba);
if (i == 0) {
/* mbr_size = pgpt_entry->ending_lba - 511; */
sunxi_get_logical_offset_param(storage_type, &logic_offset, &total_sectors);
mbr_size = pgpt_entry->starting_lba - logic_offset;
printf("mbr sector size:%x\n", mbr_size);
}
if (pgpt_entry->attributes.fields.type_guid_specific == 0x6000)
sunxi_mbr->array[i].ro = 1;
else if (pgpt_entry->attributes.fields.type_guid_specific == 0x8000)
sunxi_mbr->array[i].ro = 0;
strcpy((char *)sunxi_mbr->array[i].classname, "DISK");
memset(sunxi_mbr->array[i].name, 0, 16);
for (j = 0; j < 16; j++) {
if (pgpt_entry->partition_name[j])
sunxi_mbr->array[i].name[j] = pgpt_entry->partition_name[j];
else
break;
}
}
for (i = 0; i < sunxi_mbr->PartCount; i++) {
if (i == 0) {
sunxi_mbr->array[i].addrhi = 0;
sunxi_mbr->array[i].addrlo = mbr_size;
} else {
start_sector = sunxi_mbr->array[i - 1].addrlo;
start_sector |= (u64)sunxi_mbr->array[i - 1].addrhi << 32;
start_sector += sunxi_mbr->array[i - 1].lenlo;
sunxi_mbr->array[i].addrlo = (u32)(start_sector & 0xffffffff);
sunxi_mbr->array[i].addrhi = (u32)((start_sector >> 32) & 0xffffffff);
}
printf("i=%d, addrhi=%d addrlo=0x%x, lenhi=0x%x, lenlo=0x%x, name=%s\n", i, sunxi_mbr->array[i].addrhi,
sunxi_mbr->array[i].addrlo, sunxi_mbr->array[i].lenhi, sunxi_mbr->array[i].lenlo, sunxi_mbr->array[i].name);
}
crc32_total = crc32(0, (const unsigned char *)(sunxi_mbr_buf + 4), SUNXI_MBR_SIZE - 4);
sunxi_mbr->crc32 = crc32_total;
return 0;
}
#if 0
__weak int sunxi_sprite_erase_flash(void *mbr)
{
int nodeoffset;
uint32_t erase_flag = 0;
nodeoffset = fdt_path_offset (working_fdt,"/soc/platform" );
fdt_getprop_u32(working_fdt,nodeoffset,"eraseflag",&erase_flag);
sunxi_flash_erase(erase_flag, mbr);
return 0;
}
#endif
int sunxi_mbr_convert_to_gpt(void *sunxi_mbr_buf, char *gpt_buf,int storage_type)
{
legacy_mbr *remain_mbr;
sunxi_mbr_t *sunxi_mbr = (sunxi_mbr_t *)sunxi_mbr_buf;
char *pbuf = gpt_buf;
gpt_header *gpt_head;
gpt_entry *pgpt_entry = NULL;
char* gpt_entry_start=NULL;
u32 data_len = 0;
int total_sectors;
u32 logic_offset = 0;
int i,j = 0;
unsigned char guid[16] = {0x88,0x38,0x6f,0xab,0x9a,0x56,0x26,0x49,0x96,0x68,0x80,0x94,0x1d,0xcb,0x40,0xbc};
unsigned char part_guid[16] = {0x46,0x55,0x08,0xa0,0x66,0x41,0x4a,0x74,0xa3,0x53,0xfc,0xa9,0x27,0x2b,0x8e,0x45};
efi_guid_t basic_data_guid = PARTITION_BASIC_DATA_GUID;
if(strncmp((const char*)sunxi_mbr->magic, SUNXI_MBR_MAGIC, 8)) {
debug("%s:not sunxi mbr, can't convert to GPT partition\n", __func__);
return 0;
}
if(crc32(0, (const unsigned char *)(sunxi_mbr_buf + 4), SUNXI_MBR_SIZE - 4) != sunxi_mbr->crc32)
{
debug("%s:sunxi mbr crc error, can't convert to GPT partition\n",__func__);
return 0;
}
sunxi_get_logical_offset_param(storage_type, &logic_offset, &total_sectors);
/* 1. LBA0: write legacy mbr,part type must be 0xee */
remain_mbr = (legacy_mbr *)pbuf;
memset(remain_mbr, 0x0, 512);
remain_mbr->partition_record[0].sector = 0x2;
remain_mbr->partition_record[0].cyl = 0x0;
remain_mbr->partition_record[0].sys_ind = EFI_PMBR_OSTYPE_EFI_GPT;
remain_mbr->partition_record[0].end_head = 0xFF;
remain_mbr->partition_record[0].end_sector = 0xFF;
remain_mbr->partition_record[0].end_cyl = 0xFF;
remain_mbr->partition_record[0].start_sect = 1UL;
remain_mbr->partition_record[0].nr_sects = 0xffffffff;
remain_mbr->signature = MSDOS_MBR_SIGNATURE;
data_len += 512;
/* 2. LBA1: fill primary gpt header */
gpt_head = (gpt_header *)(pbuf + data_len);
gpt_head->signature= GPT_HEADER_SIGNATURE;
gpt_head->revision = GPT_HEADER_REVISION_V1;
gpt_head->header_size = sizeof(gpt_header);
gpt_head->header_crc32 = 0x00;
gpt_head->reserved1 = 0x0;
gpt_head->my_lba = 0x01;
gpt_head->alternate_lba = total_sectors - 1;
gpt_head->first_usable_lba = sunxi_mbr->array[0].addrlo + logic_offset;
if (storage_type == STORAGE_NOR) {
/*spinor do not have much space, drop backup gpt to enlarge UDISK*/
gpt_head->last_usable_lba = total_sectors - 1;
} else {
/*room for backup GPT consider "unsable":1 GPT head + 32 GPT entry*/
gpt_head->last_usable_lba = total_sectors - (1 + 32) - 1;
}
memcpy(gpt_head->disk_guid.b,guid,16);
gpt_head->partition_entry_lba = 2;
gpt_head->num_partition_entries = sunxi_mbr->PartCount;
gpt_head->sizeof_partition_entry = GPT_ENTRY_SIZE;
gpt_head->partition_entry_array_crc32 = 0;
data_len += 512;
/* 3. LBA2~LBAn: fill gpt entry */
gpt_entry_start = (pbuf + data_len);
for(i=0;i<sunxi_mbr->PartCount;i++) {
pgpt_entry = (gpt_entry *)(gpt_entry_start + (i)*GPT_ENTRY_SIZE);
memcpy((void*)&(pgpt_entry->partition_type_guid),(void*)&basic_data_guid,sizeof(basic_data_guid));
memcpy(pgpt_entry->unique_partition_guid.b,part_guid,16);
pgpt_entry->unique_partition_guid.b[15] = part_guid[15]+i;
pgpt_entry->starting_lba = ((u64)sunxi_mbr->array[i].addrhi<<32) + sunxi_mbr->array[i].addrlo + logic_offset;
pgpt_entry->ending_lba = pgpt_entry->starting_lba \
+((u64)sunxi_mbr->array[i].lenhi<<32) \
+ sunxi_mbr->array[i].lenlo-1;
/* UDISK partition */
if(i == sunxi_mbr->PartCount-1) {
pgpt_entry->ending_lba = gpt_head->last_usable_lba;
#ifdef CONFIG_SUNXI_UBIFS
if (((STORAGE_SPI_NAND == get_boot_storage_type_ext()) ||
(STORAGE_NAND == get_boot_storage_type_ext())) && nand_use_ubi()) {
/* backup gpt not belong to any volumes */
pgpt_entry->ending_lba = pgpt_entry->starting_lba + sunxi_mbr->array[i].lenlo - 1;
}
#endif
}
debug("GPT:%-12s: %-12llx %-12llx\n", sunxi_mbr->array[i].name, pgpt_entry->starting_lba, pgpt_entry->ending_lba);
if(sunxi_mbr->array[i].ro == 1) {
pgpt_entry->attributes.fields.type_guid_specific = 0x6000;
}
else {
pgpt_entry->attributes.fields.type_guid_specific = 0x8000;
}
if(sunxi_mbr->array[i].keydata == 0x8000){
pgpt_entry->attributes.fields.keydata = 1;
}
//ASCII to unicode
memset(pgpt_entry->partition_name, 0,PARTNAME_SZ*sizeof(efi_char16_t));
if (!strncmp((char *)sunxi_mbr->array[i].name, "UDISK", sizeof("UDISK"))) {
char temp_partition_name[16] = {CONFIG_LAST_PARTITION_NAME};
for (j = 0; j < strlen((const char *)temp_partition_name); j++) {
pgpt_entry->partition_name[j] = (efi_char16_t)temp_partition_name[j];
}
/* update last partiton name ok set gpt_head->reserved1 = 0x1 */
gpt_head->reserved1 = 0x1;
} else {
for (j = 0; j < strlen((const char *)sunxi_mbr->array[i].name); j++) {
pgpt_entry->partition_name[j] = (efi_char16_t)sunxi_mbr->array[i].name[j];
}
}
data_len += GPT_ENTRY_SIZE;
}
//entry crc
gpt_head->partition_entry_array_crc32 = crc32(0, (unsigned char const *)gpt_entry_start,
(gpt_head->num_partition_entries)*(gpt_head->sizeof_partition_entry));
debug("gpt_head->partition_entry_array_crc32 = 0x%x\n",gpt_head->partition_entry_array_crc32);
//gpt crc
gpt_head->header_crc32 = crc32(0,(const unsigned char *)gpt_head, sizeof(gpt_header));
debug("gpt_head->header_crc32 = 0x%x\n",gpt_head->header_crc32);
/* 4. LBA-1: the last sector fill backup gpt header */
return data_len;
}
int download_standard_gpt(void *sunxi_mbr_buf, size_t buf_size, int storage_type)
{
typedef int (*FLASH_WIRTE)(uint start_block, uint nblock, void *buffer);
FLASH_WIRTE flash_write_pt = NULL;
char *gpt_buf = NULL;
int data_len = 0;
int ret = 0;
gpt_header *gpt_head;
int gpt_buf_len = 8*1024;
#ifdef CONFIG_SUNXI_UBIFS
if (((STORAGE_SPI_NAND == get_boot_storage_type_ext()) ||
(STORAGE_NAND == get_boot_storage_type_ext())) && nand_use_ubi()) {
printf("force mbr\n");
ret = sunxi_sprite_write(0, buf_size>>9, sunxi_mbr_buf);
if (!ret) {
printf("%s:write mbr sectors fail ret = %d\n", __func__, ret);
sunxi_sprite_write_end();
sunxi_sprite_flush();
return -1;
}
sunxi_sprite_write_end();
sunxi_sprite_flush();
return 0;
}
#endif
gpt_buf = memalign(CONFIG_SYS_CACHELINE_SIZE, ALIGN(gpt_buf_len, CONFIG_SYS_CACHELINE_SIZE));
if(gpt_buf == NULL) {
debug("malloc for GPT fail\n");
return -1;
}
memset(gpt_buf, 0x0, gpt_buf_len);
data_len = sunxi_mbr_convert_to_gpt(sunxi_mbr_buf, gpt_buf, storage_type);
if(data_len == 0) {
goto __err_end;
}
/*write GPT for u-boot use*/
ret = sunxi_sprite_write(0, (data_len+511)>>9, gpt_buf);
if(!ret) {
debug("%s:write gpt sectors fail\n",__func__);
goto __err_end;
}
/*write GPT for kenerl use if sdmmc*/
if (STORAGE_EMMC == storage_type || STORAGE_EMMC3 == storage_type
|| storage_type == STORAGE_EMMC0) {
ret = sunxi_sprite_phywrite(0, (data_len + 511) >> 9, gpt_buf);
if (!ret)
goto __err_end;
}
printf("write primary GPT success\n");
gpt_head = (gpt_header *)(gpt_buf + GPT_HEAD_OFFSET);
prepare_backup_gpt_header(gpt_head);
if (STORAGE_NOR == storage_type) {
printf("spinor: skip backup GPT\n");
} else {
if (STORAGE_EMMC == storage_type ||
STORAGE_EMMC3 == storage_type ||
storage_type == STORAGE_EMMC0)
flash_write_pt = sunxi_sprite_phywrite;
else
flash_write_pt = sunxi_sprite_write;
/* write back-up gpt PTE */
ret = flash_write_pt(gpt_head->last_usable_lba + 1,
(GPT_BUFF_SIZE - GPT_ENTRY_OFFSET) / 512,
gpt_buf + GPT_ENTRY_OFFSET);
if (!ret) {
goto __err_end;
}
/* write back-up gpt HEAD */
ret = flash_write_pt(gpt_head->my_lba, 1,
gpt_buf + GPT_HEAD_OFFSET);
if (!ret) {
goto __err_end;
}
printf("write Backup GPT success\n");
}
free(gpt_buf);
return 0;
__err_end:
if(gpt_buf)
free(gpt_buf);
return -1;
}
#define GPT_UPDATE_BUF_SIZE (GPT_BUFF_SIZE - GPT_ENTRY_OFFSET)
#ifdef CONFIG_SUNXI_UPDATE_GPT
int sunxi_update_gpt(void)
{
#ifdef CONFIG_SUNXI_UBIFS
if (((STORAGE_SPI_NAND == get_boot_storage_type_ext()) ||
(STORAGE_NAND == get_boot_storage_type_ext())) && nand_use_ubi()) {
return 0;
}
#endif
typedef int (*FLASH_WIRTE)(uint start_block, uint nblock, void *buffer);
FLASH_WIRTE flash_write_pt = NULL;
char *buf;
int data_len = GPT_UPDATE_BUF_SIZE;
int ret = 0;
int storage_type = get_boot_storage_type();
int i, j;
char char8_name[PARTNAME_SZ] = { 0 };
buf = (char *)memalign(CONFIG_SYS_CACHELINE_SIZE, GPT_UPDATE_BUF_SIZE);
if (buf == NULL) {
pr_err("malloc fail\n");
return -1;
}
char *gpt_buf = buf;
gpt_header *gpt_head = (gpt_header *)(buf + GPT_HEAD_OFFSET);
gpt_entry *entry = (gpt_entry *)(buf + GPT_ENTRY_OFFSET);
if ((storage_type == STORAGE_SD) || (storage_type == STORAGE_EMMC)
|| (storage_type == STORAGE_EMMC0))
sunxi_flash_phyread(0, GPT_BUFF_SIZE/512, buf);
else
sunxi_flash_read(0, GPT_BUFF_SIZE/512, buf);
if (gpt_head->reserved1 != 1) {
if (gpt_head->signature == GPT_HEADER_SIGNATURE) {
u32 calc_crc32 = 0;
u32 backup_crc32 = 0;
backup_crc32 = gpt_head->header_crc32;
gpt_head->header_crc32 = 0;
calc_crc32 = crc32(0, (const unsigned char *)gpt_head,
sizeof(gpt_header));
gpt_head->header_crc32 = backup_crc32;
if (calc_crc32 != backup_crc32) {
printf("the GPT table is bad\n");
goto __err_end;
}
/* gpt_show_partition_info(buffer); */
}
for (i = 0; i < gpt_head->num_partition_entries; i++) {
for (j = 0; j < PARTNAME_SZ; j++) {
char8_name[j] = (char)(entry[i].partition_name[j]);
}
if ((get_boot_work_mode() == WORK_MODE_BOOT) && (!strncmp(char8_name, "UDISK", sizeof("UDISK")))) {
char temp_partition_name[16] = {CONFIG_LAST_PARTITION_NAME};
for (j = 0; j < strlen((const char *)temp_partition_name); j++) {
entry[i].partition_name[j] = (efi_char16_t)temp_partition_name[j];
}
/* update last partiton name ok set gpt_head->reserved1 = 0x1 */
gpt_head->reserved1 = 0x1;
for (j = 0; j < PARTNAME_SZ; j++) {
char8_name[j] = (char)(entry[i].partition_name[j]);
}
pr_msg("GPT:%-12s: %-12llx %-12llx\n", char8_name,
entry[i].starting_lba, entry[i].ending_lba);
break;
}
pr_msg("GPT:%-12s: %-12llx %-12llx\n", char8_name,
entry[i].starting_lba, entry[i].ending_lba);
}
gpt_head->partition_entry_array_crc32 = 0;
gpt_head->partition_entry_array_crc32 = crc32(0, (unsigned char const *)entry,
(gpt_head->num_partition_entries)*(gpt_head->sizeof_partition_entry));
//gpt crc32
gpt_head->header_crc32 = 0;
gpt_head->header_crc32 = crc32(0, (const unsigned char *)gpt_head, gpt_head->header_size);
/*write GPT for u-boot use*/
ret = sunxi_sprite_write(0, (data_len+511)>>9, gpt_buf);
if (!ret) {
debug("%s:write gpt sectors fail\n", __func__);
goto __err_end;
}
/*write GPT for kenerl use if sdmmc*/
if (STORAGE_EMMC == storage_type || STORAGE_EMMC3 == storage_type
|| storage_type == STORAGE_EMMC0 || storage_type == STORAGE_SD) {
ret = sunxi_sprite_phywrite(0, (data_len + 511) >> 9, gpt_buf);
if (!ret)
goto __err_end;
}
pr_msg("write primary GPT success\n");
prepare_backup_gpt_header(gpt_head);
if (STORAGE_NOR == storage_type) {
pr_msg("spinor: skip backup GPT\n");
} else {
if (STORAGE_EMMC == storage_type ||
STORAGE_EMMC3 == storage_type ||
storage_type == STORAGE_EMMC0 ||
storage_type == STORAGE_SD)
flash_write_pt = sunxi_sprite_phywrite;
else
flash_write_pt = sunxi_sprite_write;
ret = flash_write_pt((u32)(gpt_head->last_usable_lba + 1),
(GPT_BUFF_SIZE - GPT_ENTRY_OFFSET) / 512,
gpt_buf + GPT_ENTRY_OFFSET);
if (!ret) {
goto __err_end;
}
/* write back-up gpt HEAD */
ret = flash_write_pt((u32)gpt_head->my_lba, 1,
gpt_buf + GPT_HEAD_OFFSET);
if (!ret) {
goto __err_end;
}
pr_msg("write Backup GPT success\n");
}
}
free(buf);
return 0;
__err_end:
free(buf);
pr_err("update gpt fail\n");
return -1;
}
#endif
#ifdef CONFIG_OFFLINE_BURN
static int gpt_fix_flash_size(char *gpt_buf, int storage_type)
{
int total_sectors;
u32 logic_offset = 0;
gpt_header *gpt_head;
gpt_entry *pgpt_entry = NULL;
char *gpt_entry_start = NULL;
int i = 0;
sunxi_get_logical_offset_param(storage_type, &logic_offset, &total_sectors);
/*LBA1: fill primary gpt header */
gpt_head = (gpt_header *)(gpt_buf + GPT_HEAD_OFFSET);
gpt_head->alternate_lba = total_sectors - 1;
gpt_head->first_usable_lba += logic_offset;
if (storage_type == STORAGE_NOR) {
/*spinor do not have much space, drop backup gpt to enlarge UDISK*/
gpt_head->last_usable_lba = total_sectors - 1;
} else {
/*room for backup GPT consider "unsable":1 GPT head + 32 GPT entry*/
gpt_head->last_usable_lba = total_sectors - (1 + 32) - 1;
}
/* fix entry logic offset*/
gpt_entry_start = (gpt_buf + GPT_ENTRY_OFFSET);
for(i = 0; i < gpt_head->num_partition_entries; i++ ){
pgpt_entry = (gpt_entry *)(gpt_entry_start + (i)*GPT_ENTRY_SIZE);
pgpt_entry->starting_lba += logic_offset;
pgpt_entry->ending_lba += logic_offset;
/*find last part */
if(i == gpt_head->num_partition_entries - 1) {
pgpt_entry->ending_lba = gpt_head->last_usable_lba;
}
}
/* entry crc */
gpt_head->partition_entry_array_crc32 = 0;
gpt_head->partition_entry_array_crc32 = crc32(0, (unsigned char const *)gpt_entry_start,
(gpt_head->num_partition_entries)*(gpt_head->sizeof_partition_entry));
printf("gpt_head->partition_entry_array_crc32 = 0x%x\n",gpt_head->partition_entry_array_crc32);
/*gpt crc */
gpt_head->header_crc32 = 0;
gpt_head->header_crc32 = crc32(0,(const unsigned char *)gpt_head, sizeof(gpt_header));
printf("gpt_head->header_crc32 = 0x%x\n",gpt_head->header_crc32);
return 0;
}
int sunxi_sprite_download_mbr(void *buffer, uint buffer_size)
{
typedef int (*FLASH_WIRTE)(uint start_block, uint nblock, void *buffer);
FLASH_WIRTE flash_write_pt = NULL;
int ret = 0;
int storage_type = 0;
gpt_header *gpt_head = (gpt_header*)(buffer + GPT_HEAD_OFFSET);
aastorage_type = get_boot_storage_type();
if ((storage_type == STORAGE_NAND) && (sunxi_flash_init())) {
return -1;
}
if (buffer_size != GPT_BUFF_SIZE) {
printf("the GPT size is bad\n");
return -1;
}
if(gpt_head->signature != GPT_HEADER_SIGNATURE) {
printf("gpt magic error, %llx != %llx\n",gpt_head->signature,
GPT_HEADER_SIGNATURE);
return 0;
}
gpt_fix_flash_size(buffer, storage_type);
/*write to logic addr for uboot use*/
ret = sunxi_flash_write(0, (buffer_size+511)>>9, buffer);
if(!ret) {
debug("%s:write gpt sectors fail\n",__func__);
goto __err_end;
}
/*write to phy addr for kernel use*/
if (STORAGE_EMMC == storage_type || STORAGE_EMMC3 == storage_type
|| storage_type == STORAGE_EMMC0) {
ret = sunxi_flash_phywrite(0, buffer_size>>9, buffer);
if(!ret)
goto __err_end;
}
printf("write primary GPT success\n");
prepare_backup_gpt_header(gpt_head);
if (STORAGE_EMMC == storage_type || STORAGE_EMMC3 == storage_type
storage_type == STORAGE_EMMC0)
flash_write_pt = sunxi_flash_phywrite;
else
flash_write_pt = sunxi_flash_write;
if (STORAGE_NOR == storage_type) {
printf("spinor: skip backup GPT\n");
} else {
/* write back-up gpt PTE */
ret = flash_write_pt(gpt_head->last_usable_lba + 1,
(GPT_BUFF_SIZE - GPT_ENTRY_OFFSET) / 512,
buffer + GPT_ENTRY_OFFSET);
if (!ret) {
goto __err_end;
}
/* write back-up gpt HEAD */
ret = flash_write_pt(gpt_head->my_lba, 1,
buffer + GPT_HEAD_OFFSET);
if (!ret) {
goto __err_end;
}
printf("write Backup GPT success\n");
}
return 0;
__err_end:
return -1;
}
static int gpt_show_partition_info(char *buf)
{
int i, j;
char char8_name[PARTNAME_SZ] = {0};
gpt_header *gpt_head = (gpt_header*)(buf + GPT_HEAD_OFFSET);
gpt_entry *entry = (gpt_entry*)(buf + GPT_ENTRY_OFFSET);
for(i = 0; i < gpt_head->num_partition_entries; i++ ){
for(j = 0; j < PARTNAME_SZ; j++ ) {
char8_name[j] = (char)(entry[i].partition_name[j]);
}
printf("GPT:%-12s: %-12llx %-12llx\n", char8_name, entry[i].starting_lba, entry[i].ending_lba);
}
return 0;
}
static int is_gpt_valid(void *buffer)
{
u32 calc_crc32 = 0;
u32 backup_crc32 = 0;
gpt_header *gpt_head = (gpt_header *)(buffer + GPT_HEAD_OFFSET);
if(gpt_head->signature != GPT_HEADER_SIGNATURE)
{
printf("gpt magic error, %llx != %llx\n",gpt_head->signature, GPT_HEADER_SIGNATURE);
return 0;
}
gpt_show_partition_info(buffer);
return 1;
backup_crc32 = gpt_head->header_crc32;
gpt_head->header_crc32 = 0;
calc_crc32 = crc32(0,(const unsigned char *)gpt_head, sizeof(gpt_header));
gpt_head->header_crc32 = backup_crc32;
if(calc_crc32 == backup_crc32)
{
gpt_show_partition_info(buffer);
return 1;
}
printf("gpt crc error, 0x%x != 0x%x\n",backup_crc32, calc_crc32);
return 0;
}
int nand_get_mbr(char *buf, uint len)
{
int i, j;
char char8_name[PARTNAME_SZ] = {0};
gpt_header *gpt_head = (gpt_header*)(buf + 512);
gpt_entry *entry = (gpt_entry*)(buf + 1024);
int index = 0;
nand_mbr.PartCount = gpt_head->num_partition_entries + 1;
nand_mbr.array[index].addr = 0;
nand_mbr.array[index].len = entry[0].starting_lba;
nand_mbr.array[index].user_type = 0x8000;
nand_mbr.array[index].classname[0] = 'm';
nand_mbr.array[index].classname[1] = 'b';
nand_mbr.array[index].classname[2] = 'r';
nand_mbr.array[index].classname[3] = '\0';
index++;
for(i = 0; i < gpt_head->num_partition_entries; i++, index++ ){
for(j = 0; j < 16; j++ ) {
nand_mbr.array[index].classname[j] = (char)(entry[i].partition_name[j]);
}
nand_mbr.array[index].addr = nand_mbr.array[i].addr + nand_mbr.array[i-1].len;
nand_mbr.array[index].len = entry[i].ending_lba - entry[i].starting_lba;
nand_mbr.array[index].user_type = entry[i].attributes.fields.user_type;
if(i == 0)
nand_mbr.array[0].user_type = nand_mbr.array[1].user_type;
}
/* for DEBUG */
{
printf("total part: %d\n", nand_mbr.PartCount);
for(i=0; i < nand_mbr.PartCount; i++)
{
printf("%s %d, %x, %x\n",nand_mbr.array[i].classname,i, nand_mbr.array[i].len, nand_mbr.array[i].user_type);
}
}
return 0;
}
#endif