xref: /openbmc/u-boot/tools/env/fw_env_main.c (revision d51f71c1d7d5b23a9a27e3ebaefdb4dda78cacb2)
183d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
26aff3115Swdenk /*
3bc11756dSGrant Erickson  * (C) Copyright 2000-2008
46aff3115Swdenk  * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
56aff3115Swdenk  */
66aff3115Swdenk 
76aff3115Swdenk /*
83bac3513Swdenk  * Command line user interface to firmware (=U-Boot) environment.
96aff3115Swdenk  *
106aff3115Swdenk  * Implements:
11a8a752c0SMarek Vasut  *	fw_printenv [ -a key ] [[ -n name ] | [ name ... ]]
12bc11756dSGrant Erickson  *              - prints the value of a single environment variable
13bc11756dSGrant Erickson  *                "name", the ``name=value'' pairs of one or more
14bc11756dSGrant Erickson  *                environment variables "name", or the whole
15bc11756dSGrant Erickson  *                environment if no names are specified.
16a8a752c0SMarek Vasut  *	fw_setenv [ -a key ] name [ value ... ]
176aff3115Swdenk  *		- If a name without any values is given, the variable
186aff3115Swdenk  *		  with this name is deleted from the environment;
196aff3115Swdenk  *		  otherwise, all "value" arguments are concatenated,
20bc11756dSGrant Erickson  *		  separated by single blank characters, and the
216aff3115Swdenk  *		  resulting string is assigned to the environment
226aff3115Swdenk  *		  variable "name"
23a8a752c0SMarek Vasut  *
24a8a752c0SMarek Vasut  * If '-a key' is specified, the env block is encrypted with AES 128 CBC.
25a8a752c0SMarek Vasut  * The 'key' argument is in the format of 32 hexadecimal numbers (16 bytes
26a8a752c0SMarek Vasut  * of AES key), eg. '-a aabbccddeeff00112233445566778899'.
276aff3115Swdenk  */
286aff3115Swdenk 
29e4a223f0SJoe Hershberger #include <fcntl.h>
30e4a223f0SJoe Hershberger #include <getopt.h>
316aff3115Swdenk #include <stdio.h>
326aff3115Swdenk #include <string.h>
336aff3115Swdenk #include <stdlib.h>
34e4a223f0SJoe Hershberger #include <sys/file.h>
35e4a223f0SJoe Hershberger #include <unistd.h>
3684d46e7eSStefan Agner #include <version.h>
379d80b49aSStefano Babic #include "fw_env_private.h"
386aff3115Swdenk #include "fw_env.h"
396aff3115Swdenk 
406aff3115Swdenk #define CMD_PRINTENV	"fw_printenv"
416aff3115Swdenk #define CMD_SETENV	"fw_setenv"
42b92ae3afSAndreas Fenkart static int do_printenv;
436aff3115Swdenk 
44bd7b26f8SStefano Babic static struct option long_options[] = {
45b92ae3afSAndreas Fenkart 	{"config", required_argument, NULL, 'c'},
46bd7b26f8SStefano Babic 	{"help", no_argument, NULL, 'h'},
47b92ae3afSAndreas Fenkart 	{"script", required_argument, NULL, 's'},
48d877a6c5SAlex Kiernan 	{"noheader", no_argument, NULL, 'n'},
49d40dbfb7SB, Ravi 	{"lock", required_argument, NULL, 'l'},
5084d46e7eSStefan Agner 	{"version", no_argument, NULL, 'v'},
51bd7b26f8SStefano Babic 	{NULL, 0, NULL, 0}
52bd7b26f8SStefano Babic };
53bd7b26f8SStefano Babic 
5481974f44SAndreas Fenkart static struct env_opts env_opts;
5581974f44SAndreas Fenkart 
5681974f44SAndreas Fenkart /* setenv options */
5781974f44SAndreas Fenkart static int noheader;
5881974f44SAndreas Fenkart 
5981974f44SAndreas Fenkart /* getenv options */
6081974f44SAndreas Fenkart static char *script_file;
6181974f44SAndreas Fenkart 
usage_printenv(void)62b92ae3afSAndreas Fenkart void usage_printenv(void)
63bd7b26f8SStefano Babic {
64bd7b26f8SStefano Babic 
65b92ae3afSAndreas Fenkart 	fprintf(stderr,
66b92ae3afSAndreas Fenkart 		"Usage: fw_printenv [OPTIONS]... [VARIABLE]...\n"
67b92ae3afSAndreas Fenkart 		"Print variables from U-Boot environment\n"
68b92ae3afSAndreas Fenkart 		"\n"
69b92ae3afSAndreas Fenkart 		" -h, --help           print this help.\n"
7084d46e7eSStefan Agner 		" -v, --version        display version\n"
71b92ae3afSAndreas Fenkart #ifdef CONFIG_FILE
72b92ae3afSAndreas Fenkart 		" -c, --config         configuration file, default:" CONFIG_FILE "\n"
73b92ae3afSAndreas Fenkart #endif
74b92ae3afSAndreas Fenkart 		" -n, --noheader       do not repeat variable name in output\n"
75*d51f71c1STim Lee 		" -l, --lock           lock node, default:/run\n"
76b92ae3afSAndreas Fenkart 		"\n");
77b92ae3afSAndreas Fenkart }
78b92ae3afSAndreas Fenkart 
usage_env_set(void)79382bee57SSimon Glass void usage_env_set(void)
80b92ae3afSAndreas Fenkart {
81b92ae3afSAndreas Fenkart 	fprintf(stderr,
82b92ae3afSAndreas Fenkart 		"Usage: fw_setenv [OPTIONS]... [VARIABLE]...\n"
83b92ae3afSAndreas Fenkart 		"Modify variables in U-Boot environment\n"
84bd7b26f8SStefano Babic 		"\n"
85b92ae3afSAndreas Fenkart 		" -h, --help           print this help.\n"
8684d46e7eSStefan Agner 		" -v, --version        display version\n"
87b92ae3afSAndreas Fenkart #ifdef CONFIG_FILE
88b92ae3afSAndreas Fenkart 		" -c, --config         configuration file, default:" CONFIG_FILE "\n"
89b92ae3afSAndreas Fenkart #endif
90*d51f71c1STim Lee 		" -l, --lock           lock node, default:/run\n"
91b92ae3afSAndreas Fenkart 		" -s, --script         batch mode to minimize writes\n"
92bd7b26f8SStefano Babic 		"\n"
93b92ae3afSAndreas Fenkart 		"Examples:\n"
94b92ae3afSAndreas Fenkart 		"  fw_setenv foo bar   set variable foo equal bar\n"
95b92ae3afSAndreas Fenkart 		"  fw_setenv foo       clear variable foo\n"
96b92ae3afSAndreas Fenkart 		"  fw_setenv --script file run batch script\n"
97b92ae3afSAndreas Fenkart 		"\n"
98b92ae3afSAndreas Fenkart 		"Script Syntax:\n"
99b92ae3afSAndreas Fenkart 		"  key [space] value\n"
1000e280659SVagrant Cascadian 		"  lines starting with '#' are treated as comment\n"
101b92ae3afSAndreas Fenkart 		"\n"
102b92ae3afSAndreas Fenkart 		"  A variable without value will be deleted. Any number of spaces are\n"
103b92ae3afSAndreas Fenkart 		"  allowed between key and value. Space inside of the value is treated\n"
104b92ae3afSAndreas Fenkart 		"  as part of the value itself.\n"
105b92ae3afSAndreas Fenkart 		"\n"
106b92ae3afSAndreas Fenkart 		"Script Example:\n"
107b92ae3afSAndreas Fenkart 		"  netdev         eth0\n"
108b92ae3afSAndreas Fenkart 		"  kernel_addr    400000\n"
109b92ae3afSAndreas Fenkart 		"  foo            empty empty empty    empty empty empty\n"
110b92ae3afSAndreas Fenkart 		"  bar\n"
111b92ae3afSAndreas Fenkart 		"\n");
112bd7b26f8SStefano Babic }
113bd7b26f8SStefano Babic 
parse_common_args(int argc,char * argv[])114af93e3d8SAndreas Fenkart static void parse_common_args(int argc, char *argv[])
1156aff3115Swdenk {
116bd7b26f8SStefano Babic 	int c;
1176aff3115Swdenk 
118371ee137SAndreas Fenkart #ifdef CONFIG_FILE
11981974f44SAndreas Fenkart 	env_opts.config_file = CONFIG_FILE;
120371ee137SAndreas Fenkart #endif
121371ee137SAndreas Fenkart 
12284d46e7eSStefan Agner 	while ((c = getopt_long(argc, argv, ":a:c:l:h:v", long_options, NULL)) !=
123af93e3d8SAndreas Fenkart 	       EOF) {
124bd7b26f8SStefano Babic 		switch (c) {
125371ee137SAndreas Fenkart #ifdef CONFIG_FILE
1269884f44cSMichael Heimpold 		case 'c':
12781974f44SAndreas Fenkart 			env_opts.config_file = optarg;
1289884f44cSMichael Heimpold 			break;
129371ee137SAndreas Fenkart #endif
130d40dbfb7SB, Ravi 		case 'l':
131d40dbfb7SB, Ravi 			env_opts.lockname = optarg;
132d40dbfb7SB, Ravi 			break;
13307ce9440SAndreas Fenkart 		case 'h':
134382bee57SSimon Glass 			do_printenv ? usage_printenv() : usage_env_set();
13507ce9440SAndreas Fenkart 			exit(EXIT_SUCCESS);
13607ce9440SAndreas Fenkart 			break;
13784d46e7eSStefan Agner 		case 'v':
13884d46e7eSStefan Agner 			fprintf(stderr, "Compiled with " U_BOOT_VERSION "\n");
13984d46e7eSStefan Agner 			exit(EXIT_SUCCESS);
14084d46e7eSStefan Agner 			break;
141af93e3d8SAndreas Fenkart 		default:
142af93e3d8SAndreas Fenkart 			/* ignore unknown options */
143af93e3d8SAndreas Fenkart 			break;
144af93e3d8SAndreas Fenkart 		}
145af93e3d8SAndreas Fenkart 	}
146af93e3d8SAndreas Fenkart 
147af93e3d8SAndreas Fenkart 	/* Reset getopt for the next pass. */
148af93e3d8SAndreas Fenkart 	opterr = 1;
149af93e3d8SAndreas Fenkart 	optind = 1;
150af93e3d8SAndreas Fenkart }
151af93e3d8SAndreas Fenkart 
parse_printenv_args(int argc,char * argv[])152af93e3d8SAndreas Fenkart int parse_printenv_args(int argc, char *argv[])
153af93e3d8SAndreas Fenkart {
154af93e3d8SAndreas Fenkart 	int c;
155af93e3d8SAndreas Fenkart 
156af93e3d8SAndreas Fenkart 	parse_common_args(argc, argv);
157af93e3d8SAndreas Fenkart 
15884d46e7eSStefan Agner 	while ((c = getopt_long(argc, argv, "a:c:ns:l:h:v", long_options, NULL))
159d40dbfb7SB, Ravi 		!= EOF) {
160af93e3d8SAndreas Fenkart 		switch (c) {
161af93e3d8SAndreas Fenkart 		case 'n':
16281974f44SAndreas Fenkart 			noheader = 1;
163af93e3d8SAndreas Fenkart 			break;
164af93e3d8SAndreas Fenkart 		case 'a':
165af93e3d8SAndreas Fenkart 		case 'c':
166af93e3d8SAndreas Fenkart 		case 'h':
167d40dbfb7SB, Ravi 		case 'l':
168af93e3d8SAndreas Fenkart 			/* ignore common options */
169af93e3d8SAndreas Fenkart 			break;
17007ce9440SAndreas Fenkart 		default: /* '?' */
171b92ae3afSAndreas Fenkart 			usage_printenv();
17207ce9440SAndreas Fenkart 			exit(EXIT_FAILURE);
17307ce9440SAndreas Fenkart 			break;
17407ce9440SAndreas Fenkart 		}
17507ce9440SAndreas Fenkart 	}
17607ce9440SAndreas Fenkart 	return 0;
17707ce9440SAndreas Fenkart }
17807ce9440SAndreas Fenkart 
parse_setenv_args(int argc,char * argv[])17907ce9440SAndreas Fenkart int parse_setenv_args(int argc, char *argv[])
18007ce9440SAndreas Fenkart {
18107ce9440SAndreas Fenkart 	int c;
18207ce9440SAndreas Fenkart 
183af93e3d8SAndreas Fenkart 	parse_common_args(argc, argv);
184371ee137SAndreas Fenkart 
18584d46e7eSStefan Agner 	while ((c = getopt_long(argc, argv, "a:c:ns:l:h:v", long_options, NULL))
186d40dbfb7SB, Ravi 		!= EOF) {
18707ce9440SAndreas Fenkart 		switch (c) {
188bd7b26f8SStefano Babic 		case 's':
18981974f44SAndreas Fenkart 			script_file = optarg;
190bd7b26f8SStefano Babic 			break;
191af93e3d8SAndreas Fenkart 		case 'a':
192af93e3d8SAndreas Fenkart 		case 'c':
193bd7b26f8SStefano Babic 		case 'h':
194d40dbfb7SB, Ravi 		case 'l':
195af93e3d8SAndreas Fenkart 			/* ignore common options */
19607ce9440SAndreas Fenkart 			break;
19729ccd7c3SDaniel Hobi 		default: /* '?' */
198382bee57SSimon Glass 			usage_env_set();
19907ce9440SAndreas Fenkart 			exit(EXIT_FAILURE);
20007ce9440SAndreas Fenkart 			break;
201bd7b26f8SStefano Babic 		}
202bd7b26f8SStefano Babic 	}
20307ce9440SAndreas Fenkart 	return 0;
20407ce9440SAndreas Fenkart }
20507ce9440SAndreas Fenkart 
main(int argc,char * argv[])20607ce9440SAndreas Fenkart int main(int argc, char *argv[])
20707ce9440SAndreas Fenkart {
208*d51f71c1STim Lee 	char *lockname = "/run/" CMD_PRINTENV ".lock";
20907ce9440SAndreas Fenkart 	int lockfd = -1;
21007ce9440SAndreas Fenkart 	int retval = EXIT_SUCCESS;
211b92ae3afSAndreas Fenkart 	char *_cmdname;
21207ce9440SAndreas Fenkart 
213b92ae3afSAndreas Fenkart 	_cmdname = *argv;
214b92ae3afSAndreas Fenkart 	if (strrchr(_cmdname, '/') != NULL)
215b92ae3afSAndreas Fenkart 		_cmdname = strrchr(_cmdname, '/') + 1;
21607ce9440SAndreas Fenkart 
217b92ae3afSAndreas Fenkart 	if (strcmp(_cmdname, CMD_PRINTENV) == 0) {
218b92ae3afSAndreas Fenkart 		do_printenv = 1;
219b92ae3afSAndreas Fenkart 	} else if (strcmp(_cmdname, CMD_SETENV) == 0) {
220b92ae3afSAndreas Fenkart 		do_printenv = 0;
22107ce9440SAndreas Fenkart 	} else {
22207ce9440SAndreas Fenkart 		fprintf(stderr,
22307ce9440SAndreas Fenkart 			"Identity crisis - may be called as `%s' or as `%s' but not as `%s'\n",
224b92ae3afSAndreas Fenkart 			CMD_PRINTENV, CMD_SETENV, _cmdname);
225b92ae3afSAndreas Fenkart 		exit(EXIT_FAILURE);
226b92ae3afSAndreas Fenkart 	}
227b92ae3afSAndreas Fenkart 
228b92ae3afSAndreas Fenkart 	if (do_printenv) {
229b92ae3afSAndreas Fenkart 		if (parse_printenv_args(argc, argv))
230b92ae3afSAndreas Fenkart 			exit(EXIT_FAILURE);
231b92ae3afSAndreas Fenkart 	} else {
232b92ae3afSAndreas Fenkart 		if (parse_setenv_args(argc, argv))
23307ce9440SAndreas Fenkart 			exit(EXIT_FAILURE);
23407ce9440SAndreas Fenkart 	}
23507ce9440SAndreas Fenkart 
2361ce68697SAndreas Fenkart 	/* shift parsed flags, jump to non-option arguments */
2371ce68697SAndreas Fenkart 	argc -= optind;
2381ce68697SAndreas Fenkart 	argv += optind;
2391ce68697SAndreas Fenkart 
240d40dbfb7SB, Ravi 	if (env_opts.lockname) {
2418a0b827bSKristian Amlie 		lockname = malloc(strlen(env_opts.lockname) +
242d40dbfb7SB, Ravi 				sizeof(CMD_PRINTENV) + 10);
243d40dbfb7SB, Ravi 		if (!lockname) {
244d40dbfb7SB, Ravi 			fprintf(stderr, "Unable allocate memory");
245d40dbfb7SB, Ravi 			exit(EXIT_FAILURE);
246d40dbfb7SB, Ravi 		}
247d40dbfb7SB, Ravi 
248d40dbfb7SB, Ravi 		sprintf(lockname, "%s/%s.lock",
249d40dbfb7SB, Ravi 			env_opts.lockname, CMD_PRINTENV);
250d40dbfb7SB, Ravi 	}
251d40dbfb7SB, Ravi 
25207ce9440SAndreas Fenkart 	lockfd = open(lockname, O_WRONLY | O_CREAT | O_TRUNC, 0666);
25307ce9440SAndreas Fenkart 	if (-1 == lockfd) {
25407ce9440SAndreas Fenkart 		fprintf(stderr, "Error opening lock file %s\n", lockname);
25507ce9440SAndreas Fenkart 		return EXIT_FAILURE;
25607ce9440SAndreas Fenkart 	}
25707ce9440SAndreas Fenkart 
25807ce9440SAndreas Fenkart 	if (-1 == flock(lockfd, LOCK_EX)) {
25907ce9440SAndreas Fenkart 		fprintf(stderr, "Error locking file %s\n", lockname);
26007ce9440SAndreas Fenkart 		close(lockfd);
26107ce9440SAndreas Fenkart 		return EXIT_FAILURE;
26207ce9440SAndreas Fenkart 	}
263bd7b26f8SStefano Babic 
264b92ae3afSAndreas Fenkart 	if (do_printenv) {
26581974f44SAndreas Fenkart 		if (fw_printenv(argc, argv, noheader, &env_opts) != 0)
266e4a223f0SJoe Hershberger 			retval = EXIT_FAILURE;
267b92ae3afSAndreas Fenkart 	} else {
26881974f44SAndreas Fenkart 		if (!script_file) {
269382bee57SSimon Glass 			if (fw_env_set(argc, argv, &env_opts) != 0)
270e4a223f0SJoe Hershberger 				retval = EXIT_FAILURE;
271bd7b26f8SStefano Babic 		} else {
27281974f44SAndreas Fenkart 			if (fw_parse_script(script_file, &env_opts) != 0)
273e4a223f0SJoe Hershberger 				retval = EXIT_FAILURE;
274bd7b26f8SStefano Babic 		}
275e4a223f0SJoe Hershberger 	}
276e4a223f0SJoe Hershberger 
277d40dbfb7SB, Ravi 	if (env_opts.lockname)
278d40dbfb7SB, Ravi 		free(lockname);
279d40dbfb7SB, Ravi 
280e4a223f0SJoe Hershberger 	flock(lockfd, LOCK_UN);
281e4a223f0SJoe Hershberger 	close(lockfd);
282e4a223f0SJoe Hershberger 	return retval;
2836aff3115Swdenk }
284