blob: e5a9bc02db23b05374a572cd99b97e856839f72a [file] [log] [blame]
wdenkd0dd1072002-10-20 17:17:16 +00001/*
2 * (C) Copyright 2001
3 * Kyle Harris, kharris@nexus-tech.net
4 *
5 * See file CREDITS for list of people who contributed to this
6 * project.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License as
10 * published by the Free Software Foundation; either version 2 of
11 * the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
21 * MA 02111-1307 USA
22 */
23
24/*
25 * autoscript allows a remote host to download a command file and,
26 * optionally, binary data for automatically updating the target. For
27 * example, you create a new kernel image and want the user to be
28 * able to simply download the image and the machine does the rest.
29 * The kernel image is postprocessed with mkimage, which creates an
30 * image with a script file prepended. If enabled, autoscript will
31 * verify the script and contents of the download and execute the
32 * script portion. This would be responsible for erasing flash,
33 * copying the new image, and rebooting the machine.
34 */
35
36/* #define DEBUG */
37
38#include <common.h>
39#include <command.h>
40#include <image.h>
41#include <malloc.h>
42#include <asm/byteorder.h>
wdenkd0dd1072002-10-20 17:17:16 +000043#if defined(CONFIG_8xx)
44#include <mpc8xx.h>
45#endif
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020046#ifdef CONFIG_SYS_HUSH_PARSER
wdenkd0dd1072002-10-20 17:17:16 +000047#include <hush.h>
48#endif
49
wdenkd0dd1072002-10-20 17:17:16 +000050int
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010051autoscript (ulong addr, const char *fit_uname)
wdenkd0dd1072002-10-20 17:17:16 +000052{
Wolfgang Denk53677ef2008-05-20 16:00:29 +020053 ulong len;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010054 image_header_t *hdr;
55 ulong *data;
56 char *cmd;
57 int rcode = 0;
58 int verify;
59#if defined(CONFIG_FIT)
60 const void* fit_hdr;
61 int noffset;
62 const void *fit_data;
63 size_t fit_len;
64#endif
wdenkd0dd1072002-10-20 17:17:16 +000065
Bartlomiej Siekaedbed242008-04-18 12:39:23 +020066 verify = getenv_yesno ("verify");
wdenkd0dd1072002-10-20 17:17:16 +000067
Marian Balakowicz9a4daad2008-02-29 14:58:34 +010068 switch (genimg_get_format ((void *)addr)) {
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010069 case IMAGE_FORMAT_LEGACY:
70 hdr = (image_header_t *)addr;
wdenkd0dd1072002-10-20 17:17:16 +000071
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010072 if (!image_check_magic (hdr)) {
73 puts ("Bad magic number\n");
wdenkd0dd1072002-10-20 17:17:16 +000074 return 1;
75 }
wdenkd0dd1072002-10-20 17:17:16 +000076
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010077 if (!image_check_hcrc (hdr)) {
78 puts ("Bad header crc\n");
79 return 1;
80 }
81
82 if (verify) {
83 if (!image_check_dcrc (hdr)) {
84 puts ("Bad data crc\n");
85 return 1;
86 }
87 }
88
89 if (!image_check_type (hdr, IH_TYPE_SCRIPT)) {
90 puts ("Bad image type\n");
91 return 1;
92 }
93
94 /* get length of script */
95 data = (ulong *)image_get_data (hdr);
96
Marian Balakowicz9a4daad2008-02-29 14:58:34 +010097 if ((len = uimage_to_cpu (*data)) == 0) {
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010098 puts ("Empty Script\n");
99 return 1;
100 }
Bartlomiej Sieka36cc8cb2008-03-20 23:10:19 +0100101
102 /*
103 * scripts are just multi-image files with one component, seek
104 * past the zero-terminated sequence of image lengths to get
105 * to the actual image data
106 */
107 while (*data++);
Marian Balakowiczd5934ad2008-02-04 08:28:09 +0100108 break;
109#if defined(CONFIG_FIT)
110 case IMAGE_FORMAT_FIT:
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100111 if (fit_uname == NULL) {
112 puts ("No FIT subimage unit name\n");
113 return 1;
114 }
115
116 fit_hdr = (const void *)addr;
117 if (!fit_check_format (fit_hdr)) {
118 puts ("Bad FIT image format\n");
119 return 1;
120 }
121
122 /* get script component image node offset */
123 noffset = fit_image_get_node (fit_hdr, fit_uname);
124 if (noffset < 0) {
125 printf ("Can't find '%s' FIT subimage\n", fit_uname);
126 return 1;
127 }
128
129 if (!fit_image_check_type (fit_hdr, noffset, IH_TYPE_SCRIPT)) {
130 puts ("Not a image image\n");
131 return 1;
132 }
133
134 /* verify integrity */
135 if (verify) {
136 if (!fit_image_check_hashes (fit_hdr, noffset)) {
137 puts ("Bad Data Hash\n");
138 return 1;
139 }
140 }
141
142 /* get script subimage data address and length */
143 if (fit_image_get_data (fit_hdr, noffset, &fit_data, &fit_len)) {
144 puts ("Could not find script subimage data\n");
145 return 1;
146 }
147
148 data = (ulong *)fit_data;
149 len = (ulong)fit_len;
150 break;
Marian Balakowiczd5934ad2008-02-04 08:28:09 +0100151#endif
152 default:
153 puts ("Wrong image format for autoscript\n");
wdenkd0dd1072002-10-20 17:17:16 +0000154 return 1;
155 }
156
wdenk699b13a2002-11-03 18:03:52 +0000157 debug ("** Script length: %ld\n", len);
wdenkd0dd1072002-10-20 17:17:16 +0000158
159 if ((cmd = malloc (len + 1)) == NULL) {
160 return 1;
161 }
162
wdenkd0dd1072002-10-20 17:17:16 +0000163 /* make sure cmd is null terminated */
Marian Balakowiczf13e7b22008-01-08 18:12:17 +0100164 memmove (cmd, (char *)data, len);
wdenkd0dd1072002-10-20 17:17:16 +0000165 *(cmd + len) = 0;
166
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200167#ifdef CONFIG_SYS_HUSH_PARSER /*?? */
wdenkd0dd1072002-10-20 17:17:16 +0000168 rcode = parse_string_outer (cmd, FLAG_PARSE_SEMICOLON);
169#else
170 {
171 char *line = cmd;
172 char *next = cmd;
173
174 /*
175 * break into individual lines,
176 * and execute each line;
177 * terminate on error.
178 */
179 while (*next) {
180 if (*next == '\n') {
181 *next = '\0';
182 /* run only non-empty commands */
Petri Lehtinen35912932008-09-10 09:43:49 +0300183 if (*line) {
wdenkd0dd1072002-10-20 17:17:16 +0000184 debug ("** exec: \"%s\"\n",
185 line);
186 if (run_command (line, 0) < 0) {
187 rcode = 1;
188 break;
189 }
190 }
191 line = next + 1;
192 }
193 ++next;
194 }
Petri Lehtinen35912932008-09-10 09:43:49 +0300195 if (rcode == 0 && *line)
196 rcode = (run_command(line, 0) >= 0);
wdenkd0dd1072002-10-20 17:17:16 +0000197 }
198#endif
199 free (cmd);
200 return rcode;
201}
202
wdenk8bde7f72003-06-27 21:31:46 +0000203/**************************************************/
Jon Loeligerbaa26db2007-07-08 17:51:39 -0500204#if defined(CONFIG_CMD_AUTOSCRIPT)
wdenkd0dd1072002-10-20 17:17:16 +0000205int
206do_autoscript (cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
207{
208 ulong addr;
209 int rcode;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100210 const char *fit_uname = NULL;
wdenkd0dd1072002-10-20 17:17:16 +0000211
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100212 /* Find script image */
wdenkd0dd1072002-10-20 17:17:16 +0000213 if (argc < 2) {
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200214 addr = CONFIG_SYS_LOAD_ADDR;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100215 debug ("* autoscr: default load address = 0x%08lx\n", addr);
216#if defined(CONFIG_FIT)
217 } else if (fit_parse_subimage (argv[1], load_addr, &addr, &fit_uname)) {
218 debug ("* autoscr: subimage '%s' from FIT image at 0x%08lx\n",
219 fit_uname, addr);
220#endif
wdenkd0dd1072002-10-20 17:17:16 +0000221 } else {
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100222 addr = simple_strtoul(argv[1], NULL, 16);
223 debug ("* autoscr: cmdline image address = 0x%08lx\n", addr);
wdenkd0dd1072002-10-20 17:17:16 +0000224 }
225
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100226 printf ("## Executing script at %08lx\n", addr);
227 rcode = autoscript (addr, fit_uname);
wdenkd0dd1072002-10-20 17:17:16 +0000228 return rcode;
229}
wdenk8bde7f72003-06-27 21:31:46 +0000230
wdenk0d498392003-07-01 21:06:45 +0000231U_BOOT_CMD(
232 autoscr, 2, 0, do_autoscript,
Peter Tyser2fb26042009-01-27 18:03:12 -0600233 "run script from memory",
wdenk8bde7f72003-06-27 21:31:46 +0000234 "[addr] - run script starting at addr"
235 " - A valid autoscr header must be present\n"
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100236#if defined(CONFIG_FIT)
237 "For FIT format uImage addr must include subimage\n"
238 "unit name in the form of addr:<subimg_uname>\n"
Jon Loeliger90253172007-07-10 11:02:44 -0500239#endif
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100240);
Jon Loeliger90253172007-07-10 11:02:44 -0500241#endif