123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275 |
- /*
- * (C) Copyright 2000-2004
- * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
- *
- * (C) Copyright 2003
- * Kai-Uwe Bloem, Auerswald GmbH & Co KG, <linux-development@auerswald.de>
- *
- * 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 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
- */
- /*
- * Multi Image extract
- */
- #include <common.h>
- #include <command.h>
- #include <image.h>
- #include <watchdog.h>
- #if defined(CONFIG_BZIP2)
- #include <bzlib.h>
- #endif
- #include <asm/byteorder.h>
- #ifndef CONFIG_SYS_XIMG_LEN
- /* use 8MByte as default max gunzip size */
- #define CONFIG_SYS_XIMG_LEN 0x800000
- #endif
- int
- do_imgextract(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])
- {
- ulong addr = load_addr;
- ulong dest = 0;
- ulong data, len, count;
- int verify;
- int part = 0;
- char pbuf[10];
- image_header_t *hdr;
- #if defined(CONFIG_FIT)
- const char *uname = NULL;
- const void* fit_hdr;
- int noffset;
- const void *fit_data;
- size_t fit_len;
- #endif
- uint unc_len = CONFIG_SYS_XIMG_LEN;
- uint8_t comp;
- verify = getenv_yesno ("verify");
- if (argc > 1) {
- addr = simple_strtoul(argv[1], NULL, 16);
- }
- if (argc > 2) {
- part = simple_strtoul(argv[2], NULL, 16);
- #if defined(CONFIG_FIT)
- uname = argv[2];
- #endif
- }
- if (argc > 3) {
- dest = simple_strtoul(argv[3], NULL, 16);
- }
- switch (genimg_get_format ((void *)addr)) {
- case IMAGE_FORMAT_LEGACY:
- printf("## Copying part %d from legacy image "
- "at %08lx ...\n", part, addr);
- hdr = (image_header_t *)addr;
- if (!image_check_magic (hdr)) {
- printf("Bad Magic Number\n");
- return 1;
- }
- if (!image_check_hcrc (hdr)) {
- printf("Bad Header Checksum\n");
- return 1;
- }
- #ifdef DEBUG
- image_print_contents (hdr);
- #endif
- if (!image_check_type (hdr, IH_TYPE_MULTI)) {
- printf("Wrong Image Type for %s command\n",
- cmdtp->name);
- return 1;
- }
- comp = image_get_comp (hdr);
- if ((comp != IH_COMP_NONE) && (argc < 4)) {
- printf("Must specify load address for %s command "
- "with compressed image\n",
- cmdtp->name);
- return 1;
- }
- if (verify) {
- printf(" Verifying Checksum ... ");
- if (!image_check_dcrc (hdr)) {
- printf("Bad Data CRC\n");
- return 1;
- }
- printf("OK\n");
- }
- count = image_multi_count (hdr);
- if (part >= count) {
- printf("Bad Image Part\n");
- return 1;
- }
- image_multi_getimg (hdr, part, &data, &len);
- break;
- #if defined(CONFIG_FIT)
- case IMAGE_FORMAT_FIT:
- if (uname == NULL) {
- puts ("No FIT subimage unit name\n");
- return 1;
- }
- printf("## Copying '%s' subimage from FIT image "
- "at %08lx ...\n", uname, addr);
- fit_hdr = (const void *)addr;
- if (!fit_check_format (fit_hdr)) {
- puts ("Bad FIT image format\n");
- return 1;
- }
- /* get subimage node offset */
- noffset = fit_image_get_node (fit_hdr, uname);
- if (noffset < 0) {
- printf ("Can't find '%s' FIT subimage\n", uname);
- return 1;
- }
- if (fit_image_check_comp (fit_hdr, noffset, IH_COMP_NONE)
- && (argc < 4)) {
- printf("Must specify load address for %s command "
- "with compressed image\n",
- cmdtp->name);
- return 1;
- }
- /* verify integrity */
- if (verify) {
- if (!fit_image_check_hashes (fit_hdr, noffset)) {
- puts ("Bad Data Hash\n");
- return 1;
- }
- }
- /* get subimage data address and length */
- if (fit_image_get_data (fit_hdr, noffset,
- &fit_data, &fit_len)) {
- puts ("Could not find script subimage data\n");
- return 1;
- }
- if (fit_image_get_comp (fit_hdr, noffset, &comp)) {
- puts ("Could not find script subimage "
- "compression type\n");
- return 1;
- }
- data = (ulong)fit_data;
- len = (ulong)fit_len;
- break;
- #endif
- default:
- puts ("Invalid image type for imxtract\n");
- return 1;
- }
- if (argc > 3) {
- switch (comp) {
- case IH_COMP_NONE:
- #if defined(CONFIG_HW_WATCHDOG) || defined(CONFIG_WATCHDOG)
- {
- size_t l = len;
- size_t tail;
- void *to = (void *) dest;
- void *from = (void *)data;
- printf (" Loading part %d ... ", part);
- while (l > 0) {
- tail = (l > CHUNKSZ) ? CHUNKSZ : l;
- WATCHDOG_RESET();
- memmove (to, from, tail);
- to += tail;
- from += tail;
- l -= tail;
- }
- }
- #else /* !(CONFIG_HW_WATCHDOG || CONFIG_WATCHDOG) */
- printf (" Loading part %d ... ", part);
- memmove ((char *) dest, (char *)data, len);
- #endif /* CONFIG_HW_WATCHDOG || CONFIG_WATCHDOG */
- break;
- case IH_COMP_GZIP:
- printf (" Uncompressing part %d ... ", part);
- if (gunzip ((void *) dest, unc_len,
- (uchar *) data, &len) != 0) {
- puts ("GUNZIP ERROR - image not loaded\n");
- return 1;
- }
- break;
- #if defined(CONFIG_BZIP2)
- case IH_COMP_BZIP2:
- {
- int i;
- printf (" Uncompressing part %d ... ", part);
- /*
- * If we've got less than 4 MB of malloc()
- * space, use slower decompression algorithm
- * which requires at most 2300 KB of memory.
- */
- i = BZ2_bzBuffToBuffDecompress(
- (char*)ntohl(hdr->ih_load),
- &unc_len, (char *)data, len,
- CONFIG_SYS_MALLOC_LEN < (4096 * 1024),
- 0);
- if (i != BZ_OK) {
- printf ("BUNZIP2 ERROR %d - "
- "image not loaded\n", i);
- return 1;
- }
- }
- break;
- #endif /* CONFIG_BZIP2 */
- default:
- printf ("Unimplemented compression type %d\n", comp);
- return 1;
- }
- puts ("OK\n");
- }
- sprintf(pbuf, "%8lx", data);
- setenv("fileaddr", pbuf);
- sprintf(pbuf, "%8lx", len);
- setenv("filesize", pbuf);
- return 0;
- }
- U_BOOT_CMD(
- imxtract, 4, 1, do_imgextract,
- "extract a part of a multi-image",
- "addr part [dest]\n"
- " - extract <part> from legacy image at <addr> and copy to <dest>"
- #if defined(CONFIG_FIT)
- "\n"
- "addr uname [dest]\n"
- " - extract <uname> subimage from FIT image at <addr> and copy to <dest>"
- #endif
- );
|