9
0
Fork 0
barebox/common/uimage.c

529 lines
12 KiB
C

/*
* uimage.c - uimage handling code
*
* Copyright (c) 2011 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix
*
* partly based on U-Boot uImage code
*
* 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 version 2
* as published by the Free Software Foundation.
*
* 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.
*/
#include <common.h>
#include <image.h>
#include <malloc.h>
#include <errno.h>
#include <libbb.h>
#include <uncompress.h>
#include <fcntl.h>
#include <fs.h>
#include <rtc.h>
#include <filetype.h>
#include <memory.h>
static inline int uimage_is_multi_image(struct uimage_handle *handle)
{
return (handle->header.ih_type == IH_TYPE_MULTI) ? 1 : 0;
}
void uimage_print_contents(struct uimage_handle *handle)
{
struct image_header *hdr = &handle->header;
#if defined(CONFIG_TIMESTAMP)
struct rtc_time tm;
#endif
printf(" Image Name: %.*s\n", IH_NMLEN, hdr->ih_name);
#if defined(CONFIG_TIMESTAMP)
printf(" Created: ");
to_tm(hdr->ih_time, &tm);
printf("%4d-%02d-%02d %2d:%02d:%02d UTC\n",
tm.tm_year, tm.tm_mon, tm.tm_mday,
tm.tm_hour, tm.tm_min, tm.tm_sec);
#endif
#if defined(CONFIG_CMD_BOOTM_SHOW_TYPE)
printf(" OS: %s\n", image_get_os_name(hdr->ih_os));
printf(" Architecture: %s\n", image_get_arch_name(hdr->ih_arch));
printf(" Type: %s\n", image_get_type_name(hdr->ih_type));
printf(" Compression: %s\n", image_get_comp_name(hdr->ih_comp));
#endif
printf(" Data Size: %d Bytes = %s\n", hdr->ih_size,
size_human_readable(hdr->ih_size));
printf(" Load Address: %08x\n", hdr->ih_load);
printf(" Entry Point: %08x\n", hdr->ih_ep);
if (uimage_is_multi_image(handle)) {
int i;
printf(" Contents:\n");
for (i = 0; i < handle->nb_data_entries; i++) {
struct uimage_handle_data *data = &handle->ihd[i];
printf(" Image %d: %ld (%s)\n", i,
data->len, size_human_readable(data->len));
}
}
}
EXPORT_SYMBOL(uimage_print_contents);
ssize_t uimage_get_size(struct uimage_handle *handle, unsigned int image_no)
{
if (image_no >= handle->nb_data_entries)
return -EINVAL;
return handle->ihd[image_no].len;
}
EXPORT_SYMBOL(uimage_get_size);
static const char uimage_tmp[] = "/.uImage_tmp";
/*
* open a uimage. This will check the header contents and
* return a handle to the uImage
*/
struct uimage_handle *uimage_open(const char *filename)
{
int fd;
uint32_t checksum;
struct uimage_handle *handle;
struct image_header *header;
int i;
int ret;
struct stat s;
again:
fd = open(filename, O_RDONLY);
if (fd < 0) {
printf("could not open: %s\n", errno_str());
return NULL;
}
/*
* Hack around tftp fs. We need lseek for uImage support, but
* this cannot be implemented in tftp fs, so we detect this
* by doing a test lseek and copy the file to ram if it fails
*/
if (IS_BUILTIN(CONFIG_FS_TFTP) && lseek(fd, 0, SEEK_SET)) {
close(fd);
ret = copy_file(filename, uimage_tmp, 0);
if (ret)
return NULL;
filename = uimage_tmp;
goto again;
}
handle = xzalloc(sizeof(struct uimage_handle));
header = &handle->header;
if (read(fd, header, sizeof(*header)) < 0) {
printf("could not read: %s\n", errno_str());
goto err_out;
}
if (uimage_to_cpu(header->ih_magic) != IH_MAGIC) {
printf("Bad Magic Number\n");
goto err_out;
}
checksum = uimage_to_cpu(header->ih_hcrc);
header->ih_hcrc = 0;
if (crc32(0, header, sizeof(*header)) != checksum) {
printf("Bad Header Checksum\n");
goto err_out;
}
/* convert header to cpu native endianess */
header->ih_magic = uimage_to_cpu(header->ih_magic);
header->ih_hcrc = uimage_to_cpu(header->ih_hcrc);
header->ih_time = uimage_to_cpu(header->ih_time);
header->ih_size = uimage_to_cpu(header->ih_size);
header->ih_load = uimage_to_cpu(header->ih_load);
header->ih_ep = uimage_to_cpu(header->ih_ep);
header->ih_dcrc = uimage_to_cpu(header->ih_dcrc);
if (header->ih_name[0]) {
handle->name = xzalloc(IH_NMLEN + 1);
strncpy(handle->name, header->ih_name, IH_NMLEN);
} else {
handle->name = xstrdup(filename);
}
if (uimage_is_multi_image(handle)) {
size_t offset;
for (i = 0; i < MAX_MULTI_IMAGE_COUNT; i++) {
u32 size;
ret = read(fd, &size, sizeof(size));
if (ret < 0)
goto err_out;
if (!size)
break;
handle->ihd[i].len = uimage_to_cpu(size);
}
handle->nb_data_entries = i;
/* offset of the first image in a multifile image */
offset = 0;
for (i = 0; i < handle->nb_data_entries; i++) {
handle->ihd[i].offset = offset;
offset += (handle->ihd[i].len + 3) & ~3;
}
handle->data_offset = sizeof(struct image_header) +
sizeof(u32) * (handle->nb_data_entries + 1);
} else {
handle->ihd[0].offset = 0;
handle->ihd[0].len = header->ih_size;
handle->nb_data_entries = 1;
handle->data_offset = sizeof(struct image_header);
}
/*
* fd is now at the first data word
*/
handle->fd = fd;
return handle;
err_out:
close(fd);
free(handle);
if (IS_BUILTIN(CONFIG_FS_TFTP) && !stat(uimage_tmp, &s))
unlink(uimage_tmp);
return NULL;
}
EXPORT_SYMBOL(uimage_open);
/*
* close a uImage previously opened with uimage_open
*/
void uimage_close(struct uimage_handle *handle)
{
struct stat s;
close(handle->fd);
free(handle->name);
free(handle);
if (IS_BUILTIN(CONFIG_FS_TFTP) && !stat(uimage_tmp, &s))
unlink(uimage_tmp);
}
EXPORT_SYMBOL(uimage_close);
static int uimage_fd;
static int uimage_fill(void *buf, unsigned int len)
{
return read_full(uimage_fd, buf, len);
}
static int uncompress_copy(unsigned char *inbuf_unused, int len,
int(*fill)(void*, unsigned int),
int(*flush)(void*, unsigned int),
unsigned char *outbuf_unused,
int *pos,
void(*error_fn)(char *x))
{
int ret;
void *buf = xmalloc(PAGE_SIZE);
while (len) {
int now = min(len, PAGE_SIZE);
ret = fill(buf, now);
if (ret < 0)
goto err;
ret = flush(buf, now);
if (ret < 0)
goto err;
len -= now;
}
ret = 0;
err:
free(buf);
return ret;
}
/*
* Verify the data crc of an uImage
*/
int uimage_verify(struct uimage_handle *handle)
{
u32 crc = 0;
int len, ret;
void *buf;
ret = lseek(handle->fd, sizeof(struct image_header), SEEK_SET);
if (ret < 0)
return ret;
buf = xmalloc(PAGE_SIZE);
len = handle->header.ih_size;
while (len) {
int now = min(len, PAGE_SIZE);
ret = read(handle->fd, buf, now);
if (ret < 0)
goto err;
crc = crc32(crc, buf, now);
len -= ret;
}
if (crc != handle->header.ih_dcrc) {
printf("Bad Data CRC: 0x%08x != 0x%08x\n",
crc, handle->header.ih_dcrc);
ret = -EINVAL;
goto err;
}
ret = 0;
err:
free(buf);
return ret;
}
EXPORT_SYMBOL(uimage_verify);
/*
* Load a uimage, flushing output to flush function
*/
int uimage_load(struct uimage_handle *handle, unsigned int image_no,
int(*flush)(void*, unsigned int))
{
image_header_t *hdr = &handle->header;
struct uimage_handle_data *iha;
int ret;
int (*uncompress_fn)(unsigned char *inbuf, int len,
int(*fill)(void*, unsigned int),
int(*flush)(void*, unsigned int),
unsigned char *output,
int *pos,
void(*error)(char *x));
if (image_no >= handle->nb_data_entries)
return -EINVAL;
iha = &handle->ihd[image_no];
ret = lseek(handle->fd, iha->offset + handle->data_offset,
SEEK_SET);
if (ret < 0)
return ret;
/* if ramdisk U-Boot expect to ignore the compression type */
if (hdr->ih_comp == IH_COMP_NONE || hdr->ih_type == IH_TYPE_RAMDISK)
uncompress_fn = uncompress_copy;
else
uncompress_fn = uncompress;
uimage_fd = handle->fd;
ret = uncompress_fn(NULL, iha->len, uimage_fill, flush,
NULL, NULL,
uncompress_err_stdout);
return ret;
}
EXPORT_SYMBOL(uimage_load);
static void *uimage_buf;
static size_t uimage_size;
static struct resource *uimage_resource;
static int uimage_sdram_flush(void *buf, unsigned int len)
{
if (uimage_size + len > resource_size(uimage_resource)) {
resource_size_t start = uimage_resource->start;
resource_size_t size = resource_size(uimage_resource) + len;
release_sdram_region(uimage_resource);
uimage_resource = request_sdram_region("uimage",
start, size);
if (!uimage_resource) {
printf("unable to request SDRAM "
PRINTF_CONVERSION_RESOURCE "-"
PRINTF_CONVERSION_RESOURCE "\n",
start, start + size - 1);
return -ENOMEM;
}
}
memcpy(uimage_buf + uimage_size, buf, len);
uimage_size += len;
return len;
}
#define BUFSIZ (PAGE_SIZE * 32)
struct resource *file_to_sdram(const char *filename, unsigned long adr)
{
struct resource *res;
size_t size = BUFSIZ;
size_t ofs = 0;
ssize_t now;
int fd;
fd = open(filename, O_RDONLY);
if (fd < 0)
return NULL;
while (1) {
res = request_sdram_region("image", adr, size);
if (!res) {
printf("unable to request SDRAM 0x%08lx-0x%08lx\n",
adr, adr + size - 1);
goto out;
}
now = read_full(fd, (void *)(res->start + ofs), BUFSIZ);
if (now < 0) {
release_sdram_region(res);
res = NULL;
goto out;
}
if (now < BUFSIZ) {
release_sdram_region(res);
res = request_sdram_region("image", adr, ofs + now);
goto out;
}
release_sdram_region(res);
ofs += BUFSIZ;
size += BUFSIZ;
}
out:
close(fd);
return res;
}
/*
* Load an uImage to a dynamically allocated sdram resource.
* the resource must be freed afterwards with release_sdram_region
*/
struct resource *uimage_load_to_sdram(struct uimage_handle *handle,
int image_no, unsigned long load_address)
{
int ret;
ssize_t size;
resource_size_t start = (resource_size_t)load_address;
uimage_buf = (void *)load_address;
uimage_size = 0;
size = uimage_get_size(handle, image_no);
if (size < 0)
return NULL;
uimage_resource = request_sdram_region("uimage",
start, size);
if (!uimage_resource) {
printf("unable to request SDRAM 0x%08llx-0x%08llx\n",
(unsigned long long)start,
(unsigned long long)start + size - 1);
return NULL;
}
ret = uimage_load(handle, image_no, uimage_sdram_flush);
if (ret) {
release_sdram_region(uimage_resource);
return NULL;
}
return uimage_resource;
}
EXPORT_SYMBOL(uimage_load_to_sdram);
void *uimage_load_to_buf(struct uimage_handle *handle, int image_no,
size_t *outsize)
{
u32 size;
int ret;
struct uimage_handle_data *ihd;
char ftbuf[128];
enum filetype ft;
void *buf;
if (image_no >= handle->nb_data_entries)
return NULL;
ihd = &handle->ihd[image_no];
ret = lseek(handle->fd, ihd->offset + handle->data_offset,
SEEK_SET);
if (ret < 0)
return NULL;
if (handle->header.ih_comp == IH_COMP_NONE) {
buf = malloc(ihd->len);
if (!buf)
return NULL;
ret = read_full(handle->fd, buf, ihd->len);
if (ret < ihd->len) {
free(buf);
return NULL;
}
size = ihd->len;
goto out;
}
ret = read(handle->fd, ftbuf, 128);
if (ret < 0)
return NULL;
ft = file_detect_type(ftbuf, 128);
if ((int)ft < 0)
return NULL;
if (ft != filetype_gzip)
return NULL;
ret = lseek(handle->fd, ihd->offset + handle->data_offset +
ihd->len - 4,
SEEK_SET);
if (ret < 0)
return NULL;
ret = read(handle->fd, &size, 4);
if (ret < 0)
return NULL;
size = le32_to_cpu(size);
ret = lseek(handle->fd, ihd->offset + handle->data_offset,
SEEK_SET);
if (ret < 0)
return NULL;
buf = malloc(size);
ret = uncompress_fd_to_buf(handle->fd, buf, uncompress_err_stdout);
if (ret) {
free(buf);
return NULL;
}
out:
if (outsize)
*outsize = size;
return buf;
}