204 lines
4.0 KiB
C
204 lines
4.0 KiB
C
/*
|
|
* Copyright © 2016 Keith Packard <keithp@keithp.com>
|
|
*
|
|
* 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 <stdarg.h>
|
|
#include <ctype.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <errno.h>
|
|
#include "ao-hex.h"
|
|
#include "ao-dfu.h"
|
|
|
|
static uint32_t dfu_crc;
|
|
static FILE *dfu_file;
|
|
static int dfu_failed;
|
|
static int dfu_error;
|
|
|
|
static uint32_t update_crc(uint32_t crc, uint8_t byte)
|
|
{
|
|
int j;
|
|
uint32_t mask;
|
|
|
|
crc = crc ^ byte;
|
|
for (j = 0; j < 8; j++) {
|
|
mask = -(crc & 1);
|
|
crc = (crc >> 1) ^ (0xEDB88320 & mask);
|
|
}
|
|
return crc;
|
|
}
|
|
|
|
static void dfu_init(FILE *file)
|
|
{
|
|
dfu_crc = 0xffffffff;
|
|
dfu_file = file;
|
|
dfu_failed = 0;
|
|
dfu_error = 0;
|
|
}
|
|
|
|
static int dfu_fini(void)
|
|
{
|
|
if (fflush(dfu_file) == EOF) {
|
|
if (!dfu_failed) {
|
|
dfu_failed = 1;
|
|
dfu_error = errno;
|
|
}
|
|
}
|
|
if (dfu_failed)
|
|
errno = dfu_error;
|
|
return !dfu_failed;
|
|
}
|
|
|
|
static void dfu_8(uint8_t byte) {
|
|
if (putc(byte, dfu_file) == EOF) {
|
|
if (!dfu_failed) {
|
|
dfu_failed = 1;
|
|
dfu_error = errno;
|
|
}
|
|
}
|
|
dfu_crc = update_crc(dfu_crc, byte);
|
|
}
|
|
|
|
static void dfu_pad(int len) {
|
|
while (len--)
|
|
dfu_8(0);
|
|
}
|
|
|
|
static void dfu_string(char *string) {
|
|
char c;
|
|
|
|
while ((c = *string++))
|
|
dfu_8((uint8_t) c);
|
|
}
|
|
|
|
static void dfu_string_pad(char *string, int len) {
|
|
char c;
|
|
|
|
while ((c = *string++)) {
|
|
dfu_8((uint8_t) c);
|
|
len--;
|
|
}
|
|
dfu_pad(len);
|
|
}
|
|
|
|
static void dfu_block(uint8_t *bytes, int len) {
|
|
while (len--)
|
|
dfu_8(*bytes++);
|
|
}
|
|
|
|
static void dfu_lsb16(uint16_t value) {
|
|
dfu_8(value);
|
|
dfu_8(value>>8);
|
|
}
|
|
|
|
static void dfu_lsb32(uint32_t value) {
|
|
dfu_8(value);
|
|
dfu_8(value >> 8);
|
|
dfu_8(value >> 16);
|
|
dfu_8(value >> 24);
|
|
}
|
|
|
|
static uint32_t dfu_image_size(struct ao_hex_image *image) {
|
|
return 8 + image->length;
|
|
}
|
|
|
|
static uint32_t dfu_images_size(int num_image, struct ao_hex_image images[])
|
|
{
|
|
uint32_t size = 0;
|
|
int i;
|
|
|
|
for (i = 0; i < num_image; i++)
|
|
size += dfu_image_size(&images[i]);
|
|
return size;
|
|
}
|
|
|
|
static void dfu_image(struct ao_hex_image *image)
|
|
{
|
|
dfu_lsb32(image->address);
|
|
dfu_lsb32(image->length);
|
|
dfu_block(image->data, image->length);
|
|
}
|
|
|
|
static void dfu_target(char *name, int num_image, struct ao_hex_image images[])
|
|
{
|
|
uint32_t images_size = dfu_images_size(num_image, images);
|
|
int i;
|
|
|
|
dfu_string("Target");
|
|
dfu_8(0);
|
|
if (name) {
|
|
dfu_8(1);
|
|
dfu_pad(3);
|
|
dfu_string_pad(name, 255);
|
|
} else {
|
|
dfu_8(0);
|
|
dfu_pad(3);
|
|
dfu_pad(255);
|
|
}
|
|
dfu_lsb32(images_size);
|
|
dfu_lsb32(num_image);
|
|
for (i = 0; i < num_image; i++)
|
|
dfu_image(&images[i]);
|
|
}
|
|
|
|
static uint32_t dfu_target_size(int num_image, struct ao_hex_image images[])
|
|
{
|
|
return 274 + dfu_images_size(num_image, images);
|
|
}
|
|
|
|
static uint32_t
|
|
dfu_size(int num_image, struct ao_hex_image images[])
|
|
{
|
|
uint32_t size = 0;
|
|
size += 11; /* DFU Prefix */
|
|
|
|
size += dfu_target_size(num_image, images);
|
|
|
|
return size;
|
|
}
|
|
|
|
int
|
|
ao_dfu_write(FILE *file, struct ao_dfu_info *info, int num_image, struct ao_hex_image images[])
|
|
{
|
|
uint32_t total_size;
|
|
|
|
total_size = dfu_size(num_image, images);
|
|
|
|
dfu_init(file);
|
|
/* DFU Prefix */
|
|
dfu_string(DFU_SIGNATURE);
|
|
dfu_8(0x01);
|
|
dfu_lsb32(total_size);
|
|
dfu_8(0x01);
|
|
|
|
dfu_target("ST...", num_image, images);
|
|
|
|
/* DFU Suffix */
|
|
dfu_lsb16(info->bcdDevice);
|
|
dfu_lsb16(info->idProduct);
|
|
dfu_lsb16(info->idVendor);
|
|
dfu_lsb16(DFU_SPEC_VERSION);
|
|
dfu_string("UFD");
|
|
dfu_8(16);
|
|
dfu_lsb32(dfu_crc);
|
|
return dfu_fini();
|
|
}
|
|
|