barebox/common/bbu.c
Renaud Barbier 39391164e5 bbu: parameters are inverted in error message
The error message warning the user that the image is too big for
the device has its parameters in the wrong order. Swap the two
parameters so that the output makes sense.

Signed-off-by: Renaud Barbier <renaud.barbier@ge.com>
Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de>
2016-08-18 08:18:32 +02:00

389 lines
7.9 KiB
C

/*
* bbu.c - barebox update functions
*
* Copyright (c) 2012 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix
*
* 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 <bbu.h>
#include <linux/list.h>
#include <errno.h>
#include <readkey.h>
#include <filetype.h>
#include <libfile.h>
#include <fs.h>
#include <fcntl.h>
#include <malloc.h>
#include <linux/stat.h>
#include <image-metadata.h>
static LIST_HEAD(bbu_image_handlers);
int bbu_force(struct bbu_data *data, const char *fmt, ...)
{
va_list args;
printf("UPDATE: ");
va_start(args, fmt);
vprintf(fmt, args);
va_end(args);
if (!(data->flags & BBU_FLAG_FORCE))
goto out;
if (!data->force)
goto out;
data->force--;
printf(" (forced)\n");
return 1;
out:
printf("\n");
return 0;
}
int bbu_confirm(struct bbu_data *data)
{
int key;
if (data->flags & BBU_FLAG_YES)
return 0;
if (data->imagefile)
printf("update barebox from %s using handler %s to %s (y/n)?\n",
data->imagefile, data->handler_name,
data->devicefile);
else
printf("Refresh barebox on %s using handler %s (y/n)?\n",
data->devicefile, data->handler_name);
key = read_key();
if (key == 'y') {
printf("updating barebox...\n");
return 0;
}
return -EINTR;
}
static struct bbu_handler *bbu_find_handler(const char *name)
{
struct bbu_handler *handler;
list_for_each_entry(handler, &bbu_image_handlers, list) {
if (!name) {
if (handler->flags & BBU_HANDLER_FLAG_DEFAULT)
return handler;
continue;
}
if (!strcmp(handler->name, name))
return handler;
}
return NULL;
}
static struct bbu_handler *bbu_find_handler_by_device(const char *devicepath)
{
struct bbu_handler *handler;
if (!devicepath)
return NULL;
list_for_each_entry(handler, &bbu_image_handlers, list)
if (!strcmp(handler->devicefile, devicepath))
return handler;
return NULL;
}
bool barebox_update_handler_exists(struct bbu_data *data)
{
struct bbu_handler *handler;
handler = bbu_find_handler_by_device(data->devicefile);
if (handler)
return true;
if (!data->handler_name)
return false;
return !bbu_find_handler(data->handler_name);
}
static int bbu_check_of_compat(struct bbu_data *data)
{
struct device_node *root_node;
const char *machine, *str;
int ret;
struct imd_header *of_compat;
if (!IS_ENABLED(CONFIG_OFDEVICE) || !IS_ENABLED(CONFIG_IMD))
return 0;
of_compat = imd_find_type(data->imd_data, IMD_TYPE_OF_COMPATIBLE);
if (!of_compat)
return 0;
root_node = of_get_root_node();
if (!root_node)
return 0;
str = imd_string_data(of_compat, 0);
if (of_machine_is_compatible(str)) {
pr_info("Devicetree compatible \"%s\" matches current machine\n", str);
return 0;
}
ret = of_property_read_string(root_node, "compatible", &machine);
if (ret)
return 0;
if (!bbu_force(data, "machine is incompatible with \"%s\", have \"%s\"\n", str, machine))
return -EINVAL;
return 0;
}
static int bbu_check_metadata(struct bbu_data *data)
{
struct imd_header *imd;
int ret;
char *str;
if (!IS_ENABLED(CONFIG_IMD))
return 0;
if (!data->image)
return 0;
data->imd_data = imd_get(data->image, data->len);
if (IS_ERR(data->imd_data)) {
data->imd_data = NULL;
return 0;
}
printf("Image Metadata:\n");
imd_for_each(data->imd_data, imd) {
uint32_t type = imd_read_type(imd);
if (!imd_is_string(type))
continue;
str = imd_concat_strings(imd);
printf(" %s: %s\n", imd_type_to_name(type), str);
free(str);
}
ret = bbu_check_of_compat(data);
if (ret)
return ret;
return 0;
}
/*
* do a barebox update with data from *data
*/
int barebox_update(struct bbu_data *data)
{
struct bbu_handler *handler;
int ret;
handler = bbu_find_handler_by_device(data->devicefile);
if (!handler)
handler = bbu_find_handler(data->handler_name);
if (!handler)
return -ENODEV;
if (!data->image && !data->imagefile &&
!(handler->flags & BBU_HANDLER_CAN_REFRESH)) {
pr_err("No Image file given\n");
return -EINVAL;
}
if (!data->handler_name)
data->handler_name = handler->name;
if (!data->devicefile)
data->devicefile = handler->devicefile;
ret = bbu_check_metadata(data);
if (ret)
return ret;
ret = handler->handler(handler, data);
if (ret == -EINTR)
printf("update aborted\n");
if (!ret)
printf("update succeeded\n");
return ret;
}
/*
* print a list of all registered update handlers
*/
void bbu_handlers_list(void)
{
struct bbu_handler *handler;
if (list_empty(&bbu_image_handlers))
printf("(none)\n");
list_for_each_entry(handler, &bbu_image_handlers, list)
printf("%s%-11s -> %-10s\n",
handler->flags & BBU_HANDLER_FLAG_DEFAULT ?
"* " : " ",
handler->name,
handler->devicefile);
}
/*
* register a new update handler
*/
int bbu_register_handler(struct bbu_handler *handler)
{
if (bbu_find_handler(handler->name))
return -EBUSY;
if (handler->flags & BBU_HANDLER_FLAG_DEFAULT &&
bbu_find_handler(NULL))
return -EBUSY;
list_add_tail(&handler->list, &bbu_image_handlers);
return 0;
}
struct bbu_std {
struct bbu_handler handler;
enum filetype filetype;
};
static int bbu_std_file_handler(struct bbu_handler *handler,
struct bbu_data *data)
{
struct bbu_std *std = container_of(handler, struct bbu_std, handler);
int fd, ret;
enum filetype filetype;
struct stat s;
unsigned oflags = O_WRONLY;
filetype = file_detect_type(data->image, data->len);
if (filetype != std->filetype) {
if (!bbu_force(data, "incorrect image type. Expected: %s, got %s",
file_type_to_string(std->filetype),
file_type_to_string(filetype)))
return -EINVAL;
}
ret = stat(data->devicefile, &s);
if (ret) {
oflags |= O_CREAT;
} else {
if (!S_ISREG(s.st_mode) && s.st_size < data->len) {
printf("Image (%zd) is too big for device (%lld)\n",
data->len, s.st_size);
}
}
ret = bbu_confirm(data);
if (ret)
return ret;
fd = open(data->devicefile, oflags);
if (fd < 0)
return fd;
ret = protect(fd, data->len, 0, 0);
if (ret && ret != -ENOSYS) {
printf("unprotecting %s failed with %s\n", data->devicefile,
strerror(-ret));
goto err_close;
}
ret = erase(fd, data->len, 0);
if (ret && ret != -ENOSYS) {
printf("erasing %s failed with %s\n", data->devicefile,
strerror(-ret));
goto err_close;
}
ret = write_full(fd, data->image, data->len);
if (ret < 0)
goto err_close;
protect(fd, data->len, 0, 1);
ret = 0;
err_close:
close(fd);
return ret;
}
/**
* bbu_register_std_file_update() - register a barebox update handler for a
* standard file-to-device-copy operation
* @name: Name of the handler
* @flags: BBU_HANDLER_FLAG_* flags
* @devicefile: the file to write the update image to
* @imagetype: The filetype that the update image must have
*
* This update handler us suitable for a standard file-to-device copy operation.
* The handler checks for a filetype and unprotects/erases the device if
* necessary. If devicefile belongs to a device then the device is checkd for
* enough space before touching it.
*
* Return: 0 if successful, negative error code otherwise
*/
int bbu_register_std_file_update(const char *name, unsigned long flags,
char *devicefile, enum filetype imagetype)
{
struct bbu_std *std;
struct bbu_handler *handler;
int ret;
std = xzalloc(sizeof(*std));
std->filetype = imagetype;
handler = &std->handler;
handler->flags = flags;
handler->devicefile = devicefile;
handler->name = name;
handler->handler = bbu_std_file_handler;
ret = bbu_register_handler(handler);
if (ret)
free(std);
return ret;
}