189 lines
4.7 KiB
C
189 lines
4.7 KiB
C
|
/***************************************************************************
|
||
|
* __________ __ ___.
|
||
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||
|
* \/ \/ \/ \/ \/
|
||
|
* $Id:$
|
||
|
*
|
||
|
* Copyright (C) 2010 by Marcin Bukat
|
||
|
*
|
||
|
* code taken mostly from mkboot.c
|
||
|
* Copyright (C) 2005 by Linus Nielsen Feltzing
|
||
|
*
|
||
|
* 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||
|
* KIND, either express or implied.
|
||
|
*
|
||
|
****************************************************************************/
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include "mkmpioboot.h"
|
||
|
|
||
|
#define OF_FIRMWARE_LEN 0x100000 /* size of HD200_UPG.SYS file */
|
||
|
#define MPIO_STRING_OFFSET 0xfffe0
|
||
|
|
||
|
/* We support only 1.30.05 version of OF for now */
|
||
|
static char *mpio_string = "HD200 HDD Audio Ver113005";
|
||
|
|
||
|
/* MPIO HD200 firmware is plain binary image
|
||
|
* 4 bytes of initial SP (loaded on reset)
|
||
|
* 4 bytes of initial PC (loaded on reset)
|
||
|
* binary image with entry point 0x00000008
|
||
|
*
|
||
|
* We put our bootloader code at 0x000e0000
|
||
|
* and patch reset vector to jump directly
|
||
|
* into our code on reset
|
||
|
*/
|
||
|
|
||
|
static unsigned char image[OF_FIRMWARE_LEN];
|
||
|
|
||
|
static unsigned int get_uint32be(unsigned char* p)
|
||
|
{
|
||
|
return ((p[0] << 24) | (p[1] << 16) | (p[2]<<8) | p[3]);
|
||
|
}
|
||
|
|
||
|
static long checksum(unsigned char* buf, unsigned long length)
|
||
|
{
|
||
|
unsigned long chksum = 69; /* MPIO HD200 model number */
|
||
|
unsigned long i;
|
||
|
|
||
|
if(buf == NULL)
|
||
|
return -1;
|
||
|
|
||
|
for (i = 0; i < length; i++)
|
||
|
{
|
||
|
chksum += *buf++;
|
||
|
}
|
||
|
|
||
|
return chksum;
|
||
|
}
|
||
|
|
||
|
int mkmpioboot(const char* infile, const char* bootfile, const char* outfile, int origin)
|
||
|
{
|
||
|
FILE *f;
|
||
|
int i;
|
||
|
int len;
|
||
|
unsigned long file_checksum;
|
||
|
unsigned char header_checksum[4];
|
||
|
|
||
|
memset(image, 0xff, sizeof(image));
|
||
|
|
||
|
/* First, read the mpio original firmware into the image */
|
||
|
f = fopen(infile, "rb");
|
||
|
if(!f) {
|
||
|
perror(infile);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
i = fread(image, 1, OF_FIRMWARE_LEN, f);
|
||
|
if(i < OF_FIRMWARE_LEN) {
|
||
|
perror(infile);
|
||
|
fclose(f);
|
||
|
return -2;
|
||
|
}
|
||
|
|
||
|
fclose(f);
|
||
|
|
||
|
/* Now check if we have OF file loaded based on presence
|
||
|
* of the version string in firmware
|
||
|
*/
|
||
|
|
||
|
if (strcmp((char*)(image + MPIO_STRING_OFFSET),mpio_string) != 0)
|
||
|
{
|
||
|
perror("Loaded firmware file does not look like MPIO OF file!");
|
||
|
return -3;
|
||
|
}
|
||
|
|
||
|
/* Now, read the boot loader into the image */
|
||
|
f = fopen(bootfile, "rb");
|
||
|
if(!f) {
|
||
|
perror(bootfile);
|
||
|
fclose(f);
|
||
|
return -4;
|
||
|
}
|
||
|
|
||
|
/* get bootloader size
|
||
|
* excluding header
|
||
|
*/
|
||
|
fseek(f, 0, SEEK_END);
|
||
|
len = ftell(f) - 8;
|
||
|
|
||
|
/* Now check if the place we want to put
|
||
|
* our bootloader is free
|
||
|
*/
|
||
|
for(i=0;i<len;i++)
|
||
|
{
|
||
|
if (image[origin+i] != 0)
|
||
|
{
|
||
|
perror("Place for bootloader in OF file not empty");
|
||
|
return -5;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fseek(f, 0, SEEK_SET);
|
||
|
|
||
|
/* get bootloader checksum from the header*/
|
||
|
fread(header_checksum,1,4,f);
|
||
|
|
||
|
/* omit header */
|
||
|
fseek(f, 8, SEEK_SET);
|
||
|
|
||
|
i = fread(image + origin, 1, len, f);
|
||
|
if(i < len) {
|
||
|
perror(bootfile);
|
||
|
fclose(f);
|
||
|
return -6;
|
||
|
}
|
||
|
|
||
|
fclose(f);
|
||
|
|
||
|
/* calculate checksum and compare with data
|
||
|
* from header
|
||
|
*/
|
||
|
file_checksum = checksum(image + origin, len);
|
||
|
|
||
|
if ( file_checksum != get_uint32be(header_checksum) )
|
||
|
{
|
||
|
printf("Bootloader checksum error\n");
|
||
|
return -7;
|
||
|
}
|
||
|
|
||
|
f = fopen(outfile, "wb");
|
||
|
if(!f) {
|
||
|
perror(outfile);
|
||
|
return -8;
|
||
|
}
|
||
|
|
||
|
/* Patch the stack pointer address */
|
||
|
image[0] = image[origin + 0];
|
||
|
image[1] = image[origin + 1];
|
||
|
image[2] = image[origin + 2];
|
||
|
image[3] = image[origin + 3];
|
||
|
|
||
|
/* Patch the reset vector to start the boot loader */
|
||
|
image[4] = image[origin + 4];
|
||
|
image[5] = image[origin + 5];
|
||
|
image[6] = image[origin + 6];
|
||
|
image[7] = image[origin + 7];
|
||
|
|
||
|
i = fwrite(image, 1, OF_FIRMWARE_LEN, f);
|
||
|
if(i < OF_FIRMWARE_LEN) {
|
||
|
perror(outfile);
|
||
|
fclose(f);
|
||
|
return -9;
|
||
|
}
|
||
|
|
||
|
printf("Wrote 0x%x bytes in %s\n", OF_FIRMWARE_LEN, outfile);
|
||
|
|
||
|
fclose(f);
|
||
|
|
||
|
return 0;
|
||
|
}
|