blob: 6881bc9ddd3382c02c4c6130f313e5efca0066f3 [file] [log] [blame]
wdenkd0dd1072002-10-20 17:17:16 +00001/*
2 * (C) Copyright 2001
3 * Kyle Harris, kharris@nexus-tech.net
4 *
Wolfgang Denk1a459662013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
wdenkd0dd1072002-10-20 17:17:16 +00006 */
7
8/*
Wolfgang Denk74de7ae2009-04-01 23:34:12 +02009 * The "source" command allows to define "script images", i. e. files
10 * that contain command sequences that can be executed by the command
11 * interpreter. It returns the exit status of the last command
12 * executed from the script. This is very similar to running a shell
13 * script in a UNIX shell, hence the name for the command.
wdenkd0dd1072002-10-20 17:17:16 +000014 */
15
16/* #define DEBUG */
17
18#include <common.h>
19#include <command.h>
20#include <image.h>
21#include <malloc.h>
22#include <asm/byteorder.h>
Simon Glass4ca30d62013-04-20 08:42:49 +000023#include <asm/io.h>
wdenkd0dd1072002-10-20 17:17:16 +000024#if defined(CONFIG_8xx)
25#include <mpc8xx.h>
26#endif
wdenkd0dd1072002-10-20 17:17:16 +000027
wdenkd0dd1072002-10-20 17:17:16 +000028int
Wolfgang Denk74de7ae2009-04-01 23:34:12 +020029source (ulong addr, const char *fit_uname)
wdenkd0dd1072002-10-20 17:17:16 +000030{
Wolfgang Denk53677ef2008-05-20 16:00:29 +020031 ulong len;
Heiko Schocher21d29f72014-05-28 11:33:33 +020032#if defined(CONFIG_IMAGE_FORMAT_LEGACY)
Simon Glass4ca30d62013-04-20 08:42:49 +000033 const image_header_t *hdr;
Heiko Schocher21d29f72014-05-28 11:33:33 +020034#endif
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010035 ulong *data;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010036 int verify;
Simon Glass4ca30d62013-04-20 08:42:49 +000037 void *buf;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010038#if defined(CONFIG_FIT)
39 const void* fit_hdr;
40 int noffset;
41 const void *fit_data;
42 size_t fit_len;
43#endif
wdenkd0dd1072002-10-20 17:17:16 +000044
Bartlomiej Siekaedbed242008-04-18 12:39:23 +020045 verify = getenv_yesno ("verify");
wdenkd0dd1072002-10-20 17:17:16 +000046
Simon Glass4ca30d62013-04-20 08:42:49 +000047 buf = map_sysmem(addr, 0);
48 switch (genimg_get_format(buf)) {
Heiko Schocher21d29f72014-05-28 11:33:33 +020049#if defined(CONFIG_IMAGE_FORMAT_LEGACY)
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010050 case IMAGE_FORMAT_LEGACY:
Simon Glass4ca30d62013-04-20 08:42:49 +000051 hdr = buf;
wdenkd0dd1072002-10-20 17:17:16 +000052
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010053 if (!image_check_magic (hdr)) {
54 puts ("Bad magic number\n");
wdenkd0dd1072002-10-20 17:17:16 +000055 return 1;
56 }
wdenkd0dd1072002-10-20 17:17:16 +000057
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010058 if (!image_check_hcrc (hdr)) {
59 puts ("Bad header crc\n");
60 return 1;
61 }
62
63 if (verify) {
64 if (!image_check_dcrc (hdr)) {
65 puts ("Bad data crc\n");
66 return 1;
67 }
68 }
69
70 if (!image_check_type (hdr, IH_TYPE_SCRIPT)) {
71 puts ("Bad image type\n");
72 return 1;
73 }
74
75 /* get length of script */
76 data = (ulong *)image_get_data (hdr);
77
Marian Balakowicz9a4daad2008-02-29 14:58:34 +010078 if ((len = uimage_to_cpu (*data)) == 0) {
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010079 puts ("Empty Script\n");
80 return 1;
81 }
Bartlomiej Sieka36cc8cb2008-03-20 23:10:19 +010082
83 /*
84 * scripts are just multi-image files with one component, seek
85 * past the zero-terminated sequence of image lengths to get
86 * to the actual image data
87 */
88 while (*data++);
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010089 break;
Heiko Schocher21d29f72014-05-28 11:33:33 +020090#endif
Marian Balakowiczd5934ad2008-02-04 08:28:09 +010091#if defined(CONFIG_FIT)
92 case IMAGE_FORMAT_FIT:
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010093 if (fit_uname == NULL) {
94 puts ("No FIT subimage unit name\n");
95 return 1;
96 }
97
Simon Glass4ca30d62013-04-20 08:42:49 +000098 fit_hdr = buf;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +010099 if (!fit_check_format (fit_hdr)) {
100 puts ("Bad FIT image format\n");
101 return 1;
102 }
103
104 /* get script component image node offset */
105 noffset = fit_image_get_node (fit_hdr, fit_uname);
106 if (noffset < 0) {
107 printf ("Can't find '%s' FIT subimage\n", fit_uname);
108 return 1;
109 }
110
111 if (!fit_image_check_type (fit_hdr, noffset, IH_TYPE_SCRIPT)) {
112 puts ("Not a image image\n");
113 return 1;
114 }
115
116 /* verify integrity */
117 if (verify) {
Simon Glassb8da8362013-05-07 06:11:57 +0000118 if (!fit_image_verify(fit_hdr, noffset)) {
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100119 puts ("Bad Data Hash\n");
120 return 1;
121 }
122 }
123
124 /* get script subimage data address and length */
125 if (fit_image_get_data (fit_hdr, noffset, &fit_data, &fit_len)) {
126 puts ("Could not find script subimage data\n");
127 return 1;
128 }
129
130 data = (ulong *)fit_data;
131 len = (ulong)fit_len;
132 break;
Marian Balakowiczd5934ad2008-02-04 08:28:09 +0100133#endif
134 default:
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200135 puts ("Wrong image format for \"source\" command\n");
wdenkd0dd1072002-10-20 17:17:16 +0000136 return 1;
137 }
138
wdenk699b13a2002-11-03 18:03:52 +0000139 debug ("** Script length: %ld\n", len);
Simon Glassd51004a2012-03-30 21:30:55 +0000140 return run_command_list((char *)data, len, 0);
wdenkd0dd1072002-10-20 17:17:16 +0000141}
142
wdenk8bde7f72003-06-27 21:31:46 +0000143/**************************************************/
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200144#if defined(CONFIG_CMD_SOURCE)
Jeroen Hofstee0e350f82014-06-23 00:22:08 +0200145static int do_source(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
wdenkd0dd1072002-10-20 17:17:16 +0000146{
147 ulong addr;
148 int rcode;
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100149 const char *fit_uname = NULL;
wdenkd0dd1072002-10-20 17:17:16 +0000150
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100151 /* Find script image */
wdenkd0dd1072002-10-20 17:17:16 +0000152 if (argc < 2) {
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200153 addr = CONFIG_SYS_LOAD_ADDR;
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200154 debug ("* source: default load address = 0x%08lx\n", addr);
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100155#if defined(CONFIG_FIT)
156 } else if (fit_parse_subimage (argv[1], load_addr, &addr, &fit_uname)) {
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200157 debug ("* source: subimage '%s' from FIT image at 0x%08lx\n",
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100158 fit_uname, addr);
159#endif
wdenkd0dd1072002-10-20 17:17:16 +0000160 } else {
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100161 addr = simple_strtoul(argv[1], NULL, 16);
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200162 debug ("* source: cmdline image address = 0x%08lx\n", addr);
wdenkd0dd1072002-10-20 17:17:16 +0000163 }
164
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100165 printf ("## Executing script at %08lx\n", addr);
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200166 rcode = source (addr, fit_uname);
wdenkd0dd1072002-10-20 17:17:16 +0000167 return rcode;
168}
wdenk8bde7f72003-06-27 21:31:46 +0000169
Kim Phillips088f1b12012-10-29 13:34:31 +0000170#ifdef CONFIG_SYS_LONGHELP
171static char source_help_text[] =
Wolfgang Denk74de7ae2009-04-01 23:34:12 +0200172 "[addr]\n"
173 "\t- run script starting at addr\n"
Wolfgang Denka89c33d2009-05-24 17:06:54 +0200174 "\t- A valid image header must be present"
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100175#if defined(CONFIG_FIT)
Wolfgang Denka89c33d2009-05-24 17:06:54 +0200176 "\n"
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100177 "For FIT format uImage addr must include subimage\n"
Wolfgang Denka89c33d2009-05-24 17:06:54 +0200178 "unit name in the form of addr:<subimg_uname>"
Jon Loeliger90253172007-07-10 11:02:44 -0500179#endif
Kim Phillips088f1b12012-10-29 13:34:31 +0000180 "";
181#endif
182
183U_BOOT_CMD(
184 source, 2, 0, do_source,
185 "run script from memory", source_help_text
Marian Balakowicz424c4ab2008-03-12 10:33:00 +0100186);
Jon Loeliger90253172007-07-10 11:02:44 -0500187#endif