blob: 54ffd16470e8c9d0c43ba1c209fee241163bdcd1 [file] [log] [blame]
wdenkd0dd1072002-10-20 17:17:16 +00001/*
2 * (C) Copyright 2001
3 * Kyle Harris, kharris@nexus-tech.net
4 *
Wolfgang Denkd79de1d2013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
wdenkd0dd1072002-10-20 17:17:16 +00006 */
7
8/*
Wolfgang Denk85c25df2009-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 Glass7e5581e2013-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 Denk85c25df2009-04-01 23:34:12 +020029source (ulong addr, const char *fit_uname)
wdenkd0dd1072002-10-20 17:17:16 +000030{
Wolfgang Denka1be4762008-05-20 16:00:29 +020031 ulong len;
Simon Glass7e5581e2013-04-20 08:42:49 +000032 const image_header_t *hdr;
Marian Balakowicz23b77a22008-03-12 10:33:00 +010033 ulong *data;
Marian Balakowicz23b77a22008-03-12 10:33:00 +010034 int verify;
Simon Glass7e5581e2013-04-20 08:42:49 +000035 void *buf;
Marian Balakowicz23b77a22008-03-12 10:33:00 +010036#if defined(CONFIG_FIT)
37 const void* fit_hdr;
38 int noffset;
39 const void *fit_data;
40 size_t fit_len;
41#endif
wdenkd0dd1072002-10-20 17:17:16 +000042
Bartlomiej Sieka47868592008-04-18 12:39:23 +020043 verify = getenv_yesno ("verify");
wdenkd0dd1072002-10-20 17:17:16 +000044
Simon Glass7e5581e2013-04-20 08:42:49 +000045 buf = map_sysmem(addr, 0);
46 switch (genimg_get_format(buf)) {
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010047 case IMAGE_FORMAT_LEGACY:
Simon Glass7e5581e2013-04-20 08:42:49 +000048 hdr = buf;
wdenkd0dd1072002-10-20 17:17:16 +000049
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010050 if (!image_check_magic (hdr)) {
51 puts ("Bad magic number\n");
52 return 1;
53 }
wdenkd0dd1072002-10-20 17:17:16 +000054
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010055 if (!image_check_hcrc (hdr)) {
56 puts ("Bad header crc\n");
wdenkd0dd1072002-10-20 17:17:16 +000057 return 1;
58 }
wdenkd0dd1072002-10-20 17:17:16 +000059
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010060 if (verify) {
61 if (!image_check_dcrc (hdr)) {
62 puts ("Bad data crc\n");
63 return 1;
64 }
65 }
wdenkd0dd1072002-10-20 17:17:16 +000066
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010067 if (!image_check_type (hdr, IH_TYPE_SCRIPT)) {
68 puts ("Bad image type\n");
69 return 1;
70 }
71
72 /* get length of script */
73 data = (ulong *)image_get_data (hdr);
wdenkd0dd1072002-10-20 17:17:16 +000074
Marian Balakowiczd7c88a42008-02-29 14:58:34 +010075 if ((len = uimage_to_cpu (*data)) == 0) {
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010076 puts ("Empty Script\n");
77 return 1;
78 }
Bartlomiej Siekae67c8272008-03-20 23:10:19 +010079
80 /*
81 * scripts are just multi-image files with one component, seek
82 * past the zero-terminated sequence of image lengths to get
83 * to the actual image data
84 */
85 while (*data++);
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +010086 break;
87#if defined(CONFIG_FIT)
88 case IMAGE_FORMAT_FIT:
Marian Balakowicz23b77a22008-03-12 10:33:00 +010089 if (fit_uname == NULL) {
90 puts ("No FIT subimage unit name\n");
91 return 1;
92 }
93
Simon Glass7e5581e2013-04-20 08:42:49 +000094 fit_hdr = buf;
Marian Balakowicz23b77a22008-03-12 10:33:00 +010095 if (!fit_check_format (fit_hdr)) {
96 puts ("Bad FIT image format\n");
97 return 1;
98 }
99
100 /* get script component image node offset */
101 noffset = fit_image_get_node (fit_hdr, fit_uname);
102 if (noffset < 0) {
103 printf ("Can't find '%s' FIT subimage\n", fit_uname);
104 return 1;
105 }
106
107 if (!fit_image_check_type (fit_hdr, noffset, IH_TYPE_SCRIPT)) {
108 puts ("Not a image image\n");
109 return 1;
110 }
111
112 /* verify integrity */
113 if (verify) {
Simon Glass7428ad12013-05-07 06:11:57 +0000114 if (!fit_image_verify(fit_hdr, noffset)) {
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100115 puts ("Bad Data Hash\n");
116 return 1;
117 }
118 }
119
120 /* get script subimage data address and length */
121 if (fit_image_get_data (fit_hdr, noffset, &fit_data, &fit_len)) {
122 puts ("Could not find script subimage data\n");
123 return 1;
124 }
125
126 data = (ulong *)fit_data;
127 len = (ulong)fit_len;
128 break;
Marian Balakowiczdbdd16a2008-02-04 08:28:09 +0100129#endif
130 default:
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200131 puts ("Wrong image format for \"source\" command\n");
wdenkd0dd1072002-10-20 17:17:16 +0000132 return 1;
133 }
134
wdenk699b13a2002-11-03 18:03:52 +0000135 debug ("** Script length: %ld\n", len);
Simon Glassa6642e02012-03-30 21:30:55 +0000136 return run_command_list((char *)data, len, 0);
wdenkd0dd1072002-10-20 17:17:16 +0000137}
138
wdenk57b2d802003-06-27 21:31:46 +0000139/**************************************************/
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200140#if defined(CONFIG_CMD_SOURCE)
wdenkd0dd1072002-10-20 17:17:16 +0000141int
Wolfgang Denk6262d0212010-06-28 22:00:46 +0200142do_source (cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
wdenkd0dd1072002-10-20 17:17:16 +0000143{
144 ulong addr;
145 int rcode;
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100146 const char *fit_uname = NULL;
wdenkd0dd1072002-10-20 17:17:16 +0000147
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100148 /* Find script image */
wdenkd0dd1072002-10-20 17:17:16 +0000149 if (argc < 2) {
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200150 addr = CONFIG_SYS_LOAD_ADDR;
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200151 debug ("* source: default load address = 0x%08lx\n", addr);
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100152#if defined(CONFIG_FIT)
153 } else if (fit_parse_subimage (argv[1], load_addr, &addr, &fit_uname)) {
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200154 debug ("* source: subimage '%s' from FIT image at 0x%08lx\n",
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100155 fit_uname, addr);
156#endif
wdenkd0dd1072002-10-20 17:17:16 +0000157 } else {
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100158 addr = simple_strtoul(argv[1], NULL, 16);
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200159 debug ("* source: cmdline image address = 0x%08lx\n", addr);
wdenkd0dd1072002-10-20 17:17:16 +0000160 }
161
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100162 printf ("## Executing script at %08lx\n", addr);
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200163 rcode = source (addr, fit_uname);
wdenkd0dd1072002-10-20 17:17:16 +0000164 return rcode;
165}
wdenk57b2d802003-06-27 21:31:46 +0000166
Kim Phillipsdc00a682012-10-29 13:34:31 +0000167#ifdef CONFIG_SYS_LONGHELP
168static char source_help_text[] =
Wolfgang Denk85c25df2009-04-01 23:34:12 +0200169 "[addr]\n"
170 "\t- run script starting at addr\n"
Wolfgang Denkc54781c2009-05-24 17:06:54 +0200171 "\t- A valid image header must be present"
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100172#if defined(CONFIG_FIT)
Wolfgang Denkc54781c2009-05-24 17:06:54 +0200173 "\n"
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100174 "For FIT format uImage addr must include subimage\n"
Wolfgang Denkc54781c2009-05-24 17:06:54 +0200175 "unit name in the form of addr:<subimg_uname>"
Jon Loeligerd704d912007-07-10 11:02:44 -0500176#endif
Kim Phillipsdc00a682012-10-29 13:34:31 +0000177 "";
178#endif
179
180U_BOOT_CMD(
181 source, 2, 0, do_source,
182 "run script from memory", source_help_text
Marian Balakowicz23b77a22008-03-12 10:33:00 +0100183);
Jon Loeligerd704d912007-07-10 11:02:44 -0500184#endif