123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347 |
- /*
- * Copyright (c) International Business Machines Corp., 2006
- *
- * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
- /*
- * An utility to update UBI volumes.
- *
- * Authors: Frank Haverkamp
- * Joshua W. Boyer
- * Artem Bityutskiy
- */
- #define PROGRAM_NAME "ubiupdatevol"
- #include <fcntl.h>
- #include <stdio.h>
- #include <stdint.h>
- #include <getopt.h>
- #include <stdarg.h>
- #include <stdlib.h>
- #include <string.h>
- #include <unistd.h>
- #include <sys/stat.h>
- #include <libubi.h>
- #include "common.h"
- struct args {
- int truncate;
- const char *node;
- const char *img;
- /* For deprecated -d and -B options handling */
- char dev_name[256];
- long long size;
- long long skip;
- int use_stdin;
- };
- static struct args args;
- static const char doc[] = PROGRAM_NAME " version " VERSION
- " - a tool to write data to UBI volumes.";
- static const char optionsstr[] =
- "-t, --truncate truncate volume (wipe it out)\n"
- "-s, --size=<bytes> bytes to read from input\n"
- " --skip=<bytes> leading bytes to skip from input\n"
- "-h, --help print help message\n"
- "-V, --version print program version";
- static const char usage[] =
- "Usage: " PROGRAM_NAME " <UBI volume node file name> [-t] [-s <size>] [-h] [-V] [--truncate]\n"
- "\t\t\t[--size=<size>] [--help] [--version] <image file>\n\n"
- "Example 1: " PROGRAM_NAME " /dev/ubi0_1 fs.img - write file \"fs.img\" to UBI volume /dev/ubi0_1\n"
- "Example 2: " PROGRAM_NAME " /dev/ubi0_1 -t - wipe out UBI volume /dev/ubi0_1";
- static const struct option long_options[] = {
- /* Order matters for opts w/val=0; see option_index below. */
- { .name = "skip", .has_arg = 1, .flag = NULL, .val = 0 },
- { .name = "truncate", .has_arg = 0, .flag = NULL, .val = 't' },
- { .name = "help", .has_arg = 0, .flag = NULL, .val = 'h' },
- { .name = "version", .has_arg = 0, .flag = NULL, .val = 'V' },
- { .name = "size", .has_arg = 1, .flag = NULL, .val = 's' },
- { NULL, 0, NULL, 0}
- };
- static int parse_opt(int argc, char * const argv[])
- {
- while (1) {
- int option_index, key, error = 0;
- key = getopt_long(argc, argv, "ts:h?V", long_options, &option_index);
- if (key == -1)
- break;
- switch (key) {
- case 0:
- switch (option_index) {
- case 0: /* --skip */
- args.skip = simple_strtoull(optarg, &error);
- if (error || args.skip < 0)
- return errmsg("bad skip: " "\"%s\"", optarg);
- break;
- }
- break;
- case 't':
- args.truncate = 1;
- break;
- case 's':
- args.size = simple_strtoull(optarg, &error);
- if (error || args.size < 0)
- return errmsg("bad size: " "\"%s\"", optarg);
- break;
- case 'h':
- printf("%s\n\n", doc);
- printf("%s\n\n", usage);
- printf("%s\n", optionsstr);
- exit(EXIT_SUCCESS);
- case '?':
- printf("%s\n\n", doc);
- printf("%s\n\n", usage);
- printf("%s\n", optionsstr);
- return -1;
- case 'V':
- common_print_version();
- exit(EXIT_SUCCESS);
- case ':':
- return errmsg("parameter is missing");
- default:
- fprintf(stderr, "Use -h for help\n");
- return -1;
- }
- }
- if (optind == argc)
- return errmsg("UBI device name was not specified (use -h for help)");
- else if (optind != argc - 2 && !args.truncate)
- return errmsg("specify UBI device name and image file name as first 2 "
- "parameters (use -h for help)");
- args.node = argv[optind];
- args.img = argv[optind + 1];
- if (args.img && args.truncate)
- return errmsg("You can't truncate and specify an image (use -h for help)");
- if (args.img && !args.truncate) {
- if (strcmp(args.img, "-") == 0)
- args.use_stdin = 1;
- if (args.use_stdin && !args.size)
- return errmsg("file size must be specified if input is stdin");
- }
- return 0;
- }
- static int truncate_volume(libubi_t libubi)
- {
- int err, fd;
- fd = open(args.node, O_RDWR);
- if (fd == -1)
- return sys_errmsg("cannot open \"%s\"", args.node);
- err = ubi_update_start(libubi, fd, 0);
- if (err) {
- sys_errmsg("cannot truncate volume \"%s\"", args.node);
- close(fd);
- return -1;
- }
- close(fd);
- return 0;
- }
- static int ubi_write(int fd, const void *buf, int len)
- {
- int ret;
- while (len) {
- ret = write(fd, buf, len);
- if (ret < 0) {
- if (errno == EINTR) {
- warnmsg("do not interrupt me!");
- continue;
- }
- return sys_errmsg("cannot write %d bytes to volume \"%s\"",
- len, args.node);
- }
- if (ret == 0)
- return errmsg("cannot write %d bytes to volume \"%s\"", len, args.node);
- len -= ret;
- buf += ret;
- }
- return 0;
- }
- static int update_volume(libubi_t libubi, struct ubi_vol_info *vol_info)
- {
- int err, fd, ifd;
- long long bytes;
- char *buf;
- buf = malloc(vol_info->leb_size);
- if (!buf)
- return errmsg("cannot allocate %d bytes of memory", vol_info->leb_size);
- if (!args.size) {
- struct stat st;
- err = stat(args.img, &st);
- if (err < 0) {
- errmsg("stat failed on \"%s\"", args.img);
- goto out_free;
- }
- bytes = st.st_size - args.skip;
- } else
- bytes = args.size;
- if (bytes > vol_info->rsvd_bytes) {
- errmsg("\"%s\" (size %lld) will not fit volume \"%s\" (size %lld)",
- args.img, bytes, args.node, vol_info->rsvd_bytes);
- goto out_free;
- }
- fd = open(args.node, O_RDWR);
- if (fd == -1) {
- sys_errmsg("cannot open UBI volume \"%s\"", args.node);
- goto out_free;
- }
- if (args.use_stdin) {
- ifd = STDIN_FILENO;
- if (args.skip) {
- errmsg("seeking stdin not supported");
- goto out_close1;
- }
- } else {
- ifd = open(args.img, O_RDONLY);
- if (ifd == -1) {
- sys_errmsg("cannot open \"%s\"", args.img);
- goto out_close1;
- }
- if (args.skip && lseek(ifd, args.skip, SEEK_CUR) == -1) {
- sys_errmsg("lseek input by %lld failed", args.skip);
- goto out_close;
- }
- }
- err = ubi_update_start(libubi, fd, bytes);
- if (err) {
- sys_errmsg("cannot start volume \"%s\" update", args.node);
- goto out_close;
- }
- while (bytes) {
- ssize_t ret;
- int to_copy = min(vol_info->leb_size, bytes);
- ret = read(ifd, buf, to_copy);
- if (ret <= 0) {
- if (errno == EINTR) {
- warnmsg("do not interrupt me!");
- continue;
- } else {
- sys_errmsg("cannot read %d bytes from \"%s\"",
- to_copy, args.img);
- goto out_close;
- }
- }
- err = ubi_write(fd, buf, ret);
- if (err)
- goto out_close;
- bytes -= ret;
- }
- close(ifd);
- close(fd);
- free(buf);
- return 0;
- out_close:
- close(ifd);
- out_close1:
- close(fd);
- out_free:
- free(buf);
- return -1;
- }
- int main(int argc, char * const argv[])
- {
- int err;
- libubi_t libubi;
- struct ubi_vol_info vol_info;
- err = parse_opt(argc, argv);
- if (err)
- return -1;
- libubi = libubi_open();
- if (!libubi) {
- if (errno == 0)
- errmsg("UBI is not present in the system");
- return sys_errmsg("cannot open libubi");
- }
- err = ubi_probe_node(libubi, args.node);
- if (err == 1) {
- errmsg("\"%s\" is an UBI device node, not an UBI volume node",
- args.node);
- goto out_libubi;
- } else if (err < 0) {
- if (errno == ENODEV)
- errmsg("\"%s\" is not an UBI volume node", args.node);
- else
- sys_errmsg("error while probing \"%s\"", args.node);
- goto out_libubi;
- }
- err = ubi_get_vol_info(libubi, args.node, &vol_info);
- if (err) {
- sys_errmsg("cannot get information about UBI volume \"%s\"",
- args.node);
- goto out_libubi;
- }
- if (args.truncate)
- err = truncate_volume(libubi);
- else
- err = update_volume(libubi, &vol_info);
- if (err)
- goto out_libubi;
- libubi_close(libubi);
- return 0;
- out_libubi:
- libubi_close(libubi);
- return -1;
- }
|