]> git.ipfire.org Git - people/ms/u-boot.git/blame - cmd/ximg.c
env: Rename getenv_hex(), getenv_yesno(), getenv_ulong()
[people/ms/u-boot.git] / cmd / ximg.c
CommitLineData
48abe7bf
WD
1/*
2 * (C) Copyright 2000-2004
3 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
4 *
5 * (C) Copyright 2003
6 * Kai-Uwe Bloem, Auerswald GmbH & Co KG, <linux-development@auerswald.de>
7 *
1a459660 8 * SPDX-License-Identifier: GPL-2.0+
48abe7bf
WD
9 */
10
48abe7bf
WD
11
12/*
13 * Multi Image extract
14 */
15#include <common.h>
16#include <command.h>
17#include <image.h>
0eb25b61 18#include <mapmem.h>
5912d365
WW
19#include <watchdog.h>
20#if defined(CONFIG_BZIP2)
21#include <bzlib.h>
22#endif
48abe7bf 23#include <asm/byteorder.h>
628af179 24#include <asm/io.h>
48abe7bf 25
5912d365
WW
26#ifndef CONFIG_SYS_XIMG_LEN
27/* use 8MByte as default max gunzip size */
28#define CONFIG_SYS_XIMG_LEN 0x800000
29#endif
30
088f1b19 31static int
54841ab5 32do_imgextract(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])
48abe7bf 33{
1b7897f2
MB
34 ulong addr = load_addr;
35 ulong dest = 0;
21d29f7f 36 ulong data, len;
fbe7a155 37 int verify;
1b7897f2 38 int part = 0;
21d29f7f
HS
39#if defined(CONFIG_IMAGE_FORMAT_LEGACY)
40 ulong count;
628af179 41 image_header_t *hdr = NULL;
21d29f7f 42#endif
1b7897f2 43#if defined(CONFIG_FIT)
fbe7a155 44 const char *uname = NULL;
1b7897f2
MB
45 const void* fit_hdr;
46 int noffset;
47 const void *fit_data;
48 size_t fit_len;
49#endif
a92181b3 50#ifdef CONFIG_GZIP
5912d365 51 uint unc_len = CONFIG_SYS_XIMG_LEN;
a92181b3 52#endif
5912d365 53 uint8_t comp;
48abe7bf 54
bfebc8c9 55 verify = env_get_yesno("verify");
48abe7bf
WD
56
57 if (argc > 1) {
58 addr = simple_strtoul(argv[1], NULL, 16);
59 }
60 if (argc > 2) {
61 part = simple_strtoul(argv[2], NULL, 16);
1b7897f2
MB
62#if defined(CONFIG_FIT)
63 uname = argv[2];
64#endif
48abe7bf
WD
65 }
66 if (argc > 3) {
67 dest = simple_strtoul(argv[3], NULL, 16);
68 }
69
712fbcf3 70 switch (genimg_get_format((void *)addr)) {
21d29f7f 71#if defined(CONFIG_IMAGE_FORMAT_LEGACY)
d5934ad7 72 case IMAGE_FORMAT_LEGACY:
48abe7bf 73
1b7897f2
MB
74 printf("## Copying part %d from legacy image "
75 "at %08lx ...\n", part, addr);
76
d5934ad7 77 hdr = (image_header_t *)addr;
712fbcf3 78 if (!image_check_magic(hdr)) {
d5934ad7
MB
79 printf("Bad Magic Number\n");
80 return 1;
81 }
48abe7bf 82
712fbcf3 83 if (!image_check_hcrc(hdr)) {
d5934ad7
MB
84 printf("Bad Header Checksum\n");
85 return 1;
86 }
1b7897f2 87#ifdef DEBUG
712fbcf3 88 image_print_contents(hdr);
1b7897f2 89#endif
48abe7bf 90
83636fa0
PA
91 if (!image_check_type(hdr, IH_TYPE_MULTI) &&
92 !image_check_type(hdr, IH_TYPE_SCRIPT)) {
d5934ad7
MB
93 printf("Wrong Image Type for %s command\n",
94 cmdtp->name);
95 return 1;
96 }
48abe7bf 97
712fbcf3 98 comp = image_get_comp(hdr);
5912d365
WW
99 if ((comp != IH_COMP_NONE) && (argc < 4)) {
100 printf("Must specify load address for %s command "
101 "with compressed image\n",
d5934ad7 102 cmdtp->name);
48abe7bf
WD
103 return 1;
104 }
48abe7bf 105
d5934ad7
MB
106 if (verify) {
107 printf(" Verifying Checksum ... ");
712fbcf3 108 if (!image_check_dcrc(hdr)) {
d5934ad7
MB
109 printf("Bad Data CRC\n");
110 return 1;
48abe7bf 111 }
d5934ad7 112 printf("OK\n");
48abe7bf 113 }
d5934ad7 114
712fbcf3 115 count = image_multi_count(hdr);
1b7897f2 116 if (part >= count) {
d5934ad7
MB
117 printf("Bad Image Part\n");
118 return 1;
119 }
1b7897f2 120
712fbcf3 121 image_multi_getimg(hdr, part, &data, &len);
1b7897f2 122 break;
21d29f7f 123#endif
d5934ad7
MB
124#if defined(CONFIG_FIT)
125 case IMAGE_FORMAT_FIT:
1b7897f2 126 if (uname == NULL) {
712fbcf3 127 puts("No FIT subimage unit name\n");
1b7897f2
MB
128 return 1;
129 }
130
131 printf("## Copying '%s' subimage from FIT image "
132 "at %08lx ...\n", uname, addr);
133
134 fit_hdr = (const void *)addr;
712fbcf3
SW
135 if (!fit_check_format(fit_hdr)) {
136 puts("Bad FIT image format\n");
1b7897f2
MB
137 return 1;
138 }
139
140 /* get subimage node offset */
712fbcf3 141 noffset = fit_image_get_node(fit_hdr, uname);
1b7897f2 142 if (noffset < 0) {
712fbcf3 143 printf("Can't find '%s' FIT subimage\n", uname);
1b7897f2
MB
144 return 1;
145 }
146
712fbcf3 147 if (fit_image_check_comp(fit_hdr, noffset, IH_COMP_NONE)
5912d365
WW
148 && (argc < 4)) {
149 printf("Must specify load address for %s command "
150 "with compressed image\n",
151 cmdtp->name);
1b7897f2
MB
152 return 1;
153 }
154
155 /* verify integrity */
156 if (verify) {
b8da8366 157 if (!fit_image_verify(fit_hdr, noffset)) {
712fbcf3 158 puts("Bad Data Hash\n");
1b7897f2
MB
159 return 1;
160 }
161 }
162
163 /* get subimage data address and length */
712fbcf3 164 if (fit_image_get_data(fit_hdr, noffset,
5912d365 165 &fit_data, &fit_len)) {
712fbcf3 166 puts("Could not find script subimage data\n");
1b7897f2
MB
167 return 1;
168 }
169
712fbcf3
SW
170 if (fit_image_get_comp(fit_hdr, noffset, &comp)) {
171 puts("Could not find script subimage "
5912d365
WW
172 "compression type\n");
173 return 1;
174 }
175
fbe7a155 176 data = (ulong)fit_data;
1b7897f2
MB
177 len = (ulong)fit_len;
178 break;
d5934ad7
MB
179#endif
180 default:
712fbcf3 181 puts("Invalid image type for imxtract\n");
48abe7bf
WD
182 return 1;
183 }
48abe7bf
WD
184
185 if (argc > 3) {
5912d365
WW
186 switch (comp) {
187 case IH_COMP_NONE:
188#if defined(CONFIG_HW_WATCHDOG) || defined(CONFIG_WATCHDOG)
189 {
190 size_t l = len;
191 size_t tail;
192 void *to = (void *) dest;
193 void *from = (void *)data;
194
712fbcf3 195 printf(" Loading part %d ... ", part);
5912d365
WW
196
197 while (l > 0) {
198 tail = (l > CHUNKSZ) ? CHUNKSZ : l;
199 WATCHDOG_RESET();
712fbcf3 200 memmove(to, from, tail);
5912d365
WW
201 to += tail;
202 from += tail;
203 l -= tail;
204 }
205 }
206#else /* !(CONFIG_HW_WATCHDOG || CONFIG_WATCHDOG) */
712fbcf3
SW
207 printf(" Loading part %d ... ", part);
208 memmove((char *) dest, (char *)data, len);
5912d365
WW
209#endif /* CONFIG_HW_WATCHDOG || CONFIG_WATCHDOG */
210 break;
0e0996ef 211#ifdef CONFIG_GZIP
5912d365 212 case IH_COMP_GZIP:
712fbcf3
SW
213 printf(" Uncompressing part %d ... ", part);
214 if (gunzip((void *) dest, unc_len,
215 (uchar *) data, &len) != 0) {
216 puts("GUNZIP ERROR - image not loaded\n");
5912d365
WW
217 return 1;
218 }
219 break;
0e0996ef 220#endif
21d29f7f 221#if defined(CONFIG_BZIP2) && defined(CONFIG_IMAGE_FORMAT_LEGACY)
5912d365 222 case IH_COMP_BZIP2:
5f566f45
WD
223 {
224 int i;
225
712fbcf3 226 printf(" Uncompressing part %d ... ", part);
5f566f45 227 /*
93910edb 228 * If we've got less than 4 MB of malloc()
5f566f45
WD
229 * space, use slower decompression algorithm
230 * which requires at most 2300 KB of memory.
231 */
232 i = BZ2_bzBuffToBuffDecompress(
628af179 233 map_sysmem(ntohl(hdr->ih_load), 0),
5f566f45
WD
234 &unc_len, (char *)data, len,
235 CONFIG_SYS_MALLOC_LEN < (4096 * 1024),
236 0);
237 if (i != BZ_OK) {
712fbcf3 238 printf("BUNZIP2 ERROR %d - "
5f566f45
WD
239 "image not loaded\n", i);
240 return 1;
241 }
5912d365
WW
242 }
243 break;
244#endif /* CONFIG_BZIP2 */
245 default:
712fbcf3 246 printf("Unimplemented compression type %d\n", comp);
5912d365
WW
247 return 1;
248 }
712fbcf3 249 puts("OK\n");
48abe7bf
WD
250 }
251
c72b65cc
PV
252 flush_cache(dest, len);
253
018f5303
SG
254 env_set_hex("fileaddr", data);
255 env_set_hex("filesize", len);
48abe7bf
WD
256
257 return 0;
258}
259
088f1b19
KP
260#ifdef CONFIG_SYS_LONGHELP
261static char imgextract_help_text[] =
a89c33db
WD
262 "addr part [dest]\n"
263 " - extract <part> from legacy image at <addr> and copy to <dest>"
1b7897f2 264#if defined(CONFIG_FIT)
a89c33db
WD
265 "\n"
266 "addr uname [dest]\n"
267 " - extract <uname> subimage from FIT image at <addr> and copy to <dest>"
1b7897f2 268#endif
088f1b19
KP
269 "";
270#endif
271
272U_BOOT_CMD(
273 imxtract, 4, 1, do_imgextract,
274 "extract a part of a multi-image", imgextract_help_text
1b7897f2 275);