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