blob: 194a366aa152cf4ee620b00cc9b873566366f898 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Jason Hobbs0e3a5932011-08-31 10:37:30 -05002/*
3 * Copyright 2010-2011 Calxeda, Inc.
Bryan Wua1d715b2014-07-31 17:39:59 -07004 * Copyright (c) 2014, NVIDIA CORPORATION. All rights reserved.
Jason Hobbs0e3a5932011-08-31 10:37:30 -05005 */
Wolfgang Denkd79de1d2013-07-08 09:37:19 +02006
Jason Hobbs0e3a5932011-08-31 10:37:30 -05007#include <common.h>
8#include <command.h>
Simon Glass16721112019-12-28 10:44:43 -07009#include <net.h>
Jason Hobbs0e3a5932011-08-31 10:37:30 -050010
Patrice Chotard17e88042019-11-25 09:07:37 +010011#include "pxe_utils.h"
Jason Hobbs0e3a5932011-08-31 10:37:30 -050012
Patrice Chotard17e88042019-11-25 09:07:37 +010013#ifdef CONFIG_CMD_NET
Rob Herringe26e8d62012-12-02 21:00:28 -060014const char *pxe_default_paths[] = {
Joe Hershberger921a71d2013-06-24 17:21:04 -050015#ifdef CONFIG_SYS_SOC
Marek BehĂșn016ed132019-05-02 15:29:12 +020016#ifdef CONFIG_SYS_BOARD
17 "default-" CONFIG_SYS_ARCH "-" CONFIG_SYS_SOC "-" CONFIG_SYS_BOARD,
18#endif
Rob Herringe26e8d62012-12-02 21:00:28 -060019 "default-" CONFIG_SYS_ARCH "-" CONFIG_SYS_SOC,
Joe Hershberger921a71d2013-06-24 17:21:04 -050020#endif
Rob Herringe26e8d62012-12-02 21:00:28 -060021 "default-" CONFIG_SYS_ARCH,
22 "default",
23 NULL
24};
25
Steven Falco7de0fa82013-10-07 09:51:48 -040026static int do_get_tftp(cmd_tbl_t *cmdtp, const char *file_path, char *file_addr)
Rob Herringeee675f2012-05-25 10:47:39 +000027{
28 char *tftp_argv[] = {"tftp", NULL, NULL, NULL};
29
30 tftp_argv[1] = file_addr;
Rob Herring824901c2012-12-02 21:00:21 -060031 tftp_argv[2] = (void *)file_path;
Rob Herringeee675f2012-05-25 10:47:39 +000032
Steven Falco7de0fa82013-10-07 09:51:48 -040033 if (do_tftpb(cmdtp, 0, 3, tftp_argv))
Rob Herringeee675f2012-05-25 10:47:39 +000034 return -ENOENT;
35
36 return 1;
37}
Stephen Warren857291b2014-02-05 20:49:20 -070038
Jason Hobbs0e3a5932011-08-31 10:37:30 -050039/*
40 * Looks for a pxe file with a name based on the pxeuuid environment variable.
41 *
42 * Returns 1 on success or < 0 on error.
43 */
Sjoerd Simons45af07c2015-04-13 22:54:25 +020044static int pxe_uuid_path(cmd_tbl_t *cmdtp, unsigned long pxefile_addr_r)
Jason Hobbs0e3a5932011-08-31 10:37:30 -050045{
46 char *uuid_str;
47
48 uuid_str = from_env("pxeuuid");
49
50 if (!uuid_str)
51 return -ENOENT;
52
Steven Falco7de0fa82013-10-07 09:51:48 -040053 return get_pxelinux_path(cmdtp, uuid_str, pxefile_addr_r);
Jason Hobbs0e3a5932011-08-31 10:37:30 -050054}
55
56/*
57 * Looks for a pxe file with a name based on the 'ethaddr' environment
58 * variable.
59 *
60 * Returns 1 on success or < 0 on error.
61 */
Sjoerd Simons45af07c2015-04-13 22:54:25 +020062static int pxe_mac_path(cmd_tbl_t *cmdtp, unsigned long pxefile_addr_r)
Jason Hobbs0e3a5932011-08-31 10:37:30 -050063{
64 char mac_str[21];
65 int err;
66
67 err = format_mac_pxe(mac_str, sizeof(mac_str));
68
69 if (err < 0)
70 return err;
71
Steven Falco7de0fa82013-10-07 09:51:48 -040072 return get_pxelinux_path(cmdtp, mac_str, pxefile_addr_r);
Jason Hobbs0e3a5932011-08-31 10:37:30 -050073}
74
75/*
76 * Looks for pxe files with names based on our IP address. See pxelinux
77 * documentation for details on what these file names look like. We match
78 * that exactly.
79 *
80 * Returns 1 on success or < 0 on error.
81 */
Sjoerd Simons45af07c2015-04-13 22:54:25 +020082static int pxe_ipaddr_paths(cmd_tbl_t *cmdtp, unsigned long pxefile_addr_r)
Jason Hobbs0e3a5932011-08-31 10:37:30 -050083{
84 char ip_addr[9];
85 int mask_pos, err;
86
Joe Hershberger5874dec2015-04-08 01:41:01 -050087 sprintf(ip_addr, "%08X", ntohl(net_ip.s_addr));
Jason Hobbs0e3a5932011-08-31 10:37:30 -050088
89 for (mask_pos = 7; mask_pos >= 0; mask_pos--) {
Steven Falco7de0fa82013-10-07 09:51:48 -040090 err = get_pxelinux_path(cmdtp, ip_addr, pxefile_addr_r);
Jason Hobbs0e3a5932011-08-31 10:37:30 -050091
92 if (err > 0)
93 return err;
94
95 ip_addr[mask_pos] = '\0';
96 }
97
98 return -ENOENT;
99}
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500100/*
101 * Entry point for the 'pxe get' command.
102 * This Follows pxelinux's rules to download a config file from a tftp server.
103 * The file is stored at the location given by the pxefile_addr_r environment
104 * variable, which must be set.
105 *
106 * UUID comes from pxeuuid env variable, if defined
107 * MAC addr comes from ethaddr env variable, if defined
108 * IP
109 *
110 * see http://syslinux.zytor.com/wiki/index.php/PXELINUX
111 *
112 * Returns 0 on success or 1 on error.
113 */
114static int
115do_pxe_get(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
116{
117 char *pxefile_addr_str;
Jason Hobbs9135c792012-03-05 08:12:28 +0000118 unsigned long pxefile_addr_r;
Rob Herringe26e8d62012-12-02 21:00:28 -0600119 int err, i = 0;
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500120
Rob Herringeee675f2012-05-25 10:47:39 +0000121 do_getfile = do_get_tftp;
122
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500123 if (argc != 1)
Simon Glassa06dfc72011-12-10 08:44:01 +0000124 return CMD_RET_USAGE;
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500125
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500126 pxefile_addr_str = from_env("pxefile_addr_r");
127
128 if (!pxefile_addr_str)
129 return 1;
130
131 err = strict_strtoul(pxefile_addr_str, 16,
Patrice Chotard7f871652019-11-25 09:07:41 +0100132 (unsigned long *)&pxefile_addr_r);
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500133 if (err < 0)
134 return 1;
135
136 /*
137 * Keep trying paths until we successfully get a file we're looking
138 * for.
139 */
Sjoerd Simons45af07c2015-04-13 22:54:25 +0200140 if (pxe_uuid_path(cmdtp, pxefile_addr_r) > 0 ||
141 pxe_mac_path(cmdtp, pxefile_addr_r) > 0 ||
142 pxe_ipaddr_paths(cmdtp, pxefile_addr_r) > 0) {
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500143 printf("Config file found\n");
144
145 return 0;
146 }
147
Rob Herringe26e8d62012-12-02 21:00:28 -0600148 while (pxe_default_paths[i]) {
Steven Falco7de0fa82013-10-07 09:51:48 -0400149 if (get_pxelinux_path(cmdtp, pxe_default_paths[i],
Sjoerd Simons45af07c2015-04-13 22:54:25 +0200150 pxefile_addr_r) > 0) {
Rob Herringe26e8d62012-12-02 21:00:28 -0600151 printf("Config file found\n");
152 return 0;
153 }
154 i++;
155 }
156
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500157 printf("Config file not found\n");
158
Rob Herring58d7f432012-12-02 21:00:22 -0600159 return 1;
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500160}
161
162/*
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500163 * Boots a system using a pxe file
164 *
165 * Returns 0 on success, 1 on error.
166 */
167static int
168do_pxe_boot(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
169{
170 unsigned long pxefile_addr_r;
171 struct pxe_menu *cfg;
172 char *pxefile_addr_str;
173
Rob Herringeee675f2012-05-25 10:47:39 +0000174 do_getfile = do_get_tftp;
175
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500176 if (argc == 1) {
177 pxefile_addr_str = from_env("pxefile_addr_r");
178 if (!pxefile_addr_str)
179 return 1;
180
181 } else if (argc == 2) {
182 pxefile_addr_str = argv[1];
183 } else {
Simon Glassa06dfc72011-12-10 08:44:01 +0000184 return CMD_RET_USAGE;
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500185 }
186
187 if (strict_strtoul(pxefile_addr_str, 16, &pxefile_addr_r) < 0) {
188 printf("Invalid pxefile address: %s\n", pxefile_addr_str);
189 return 1;
190 }
191
Sjoerd Simons45af07c2015-04-13 22:54:25 +0200192 cfg = parse_pxefile(cmdtp, pxefile_addr_r);
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500193
Patrice Chotard7f871652019-11-25 09:07:41 +0100194 if (!cfg) {
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500195 printf("Error parsing config file\n");
196 return 1;
197 }
198
Tom Rini5b5b1762013-09-24 09:05:08 -0400199 handle_pxe_menu(cmdtp, cfg);
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500200
201 destroy_pxe_menu(cfg);
202
Joe Hershberger290c8992015-04-08 01:41:02 -0500203 copy_filename(net_boot_file_name, "", sizeof(net_boot_file_name));
Stephen Warrenc24cbc22014-07-22 18:06:46 -0600204
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500205 return 0;
206}
207
208static cmd_tbl_t cmd_pxe_sub[] = {
209 U_BOOT_CMD_MKENT(get, 1, 1, do_pxe_get, "", ""),
210 U_BOOT_CMD_MKENT(boot, 2, 1, do_pxe_boot, "", "")
211};
212
Jeroen Hofsteef384fbf2014-06-23 00:22:08 +0200213static int do_pxe(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500214{
215 cmd_tbl_t *cp;
216
217 if (argc < 2)
Simon Glassa06dfc72011-12-10 08:44:01 +0000218 return CMD_RET_USAGE;
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500219
Rob Herring759456e2013-10-18 13:04:42 -0500220 is_pxe = true;
221
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500222 /* drop initial "pxe" arg */
223 argc--;
224 argv++;
225
226 cp = find_cmd_tbl(argv[0], cmd_pxe_sub, ARRAY_SIZE(cmd_pxe_sub));
227
228 if (cp)
229 return cp->cmd(cmdtp, flag, argc, argv);
230
Simon Glassa06dfc72011-12-10 08:44:01 +0000231 return CMD_RET_USAGE;
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500232}
233
Patrice Chotard7f871652019-11-25 09:07:41 +0100234U_BOOT_CMD(pxe, 3, 1, do_pxe,
235 "commands to get and boot from pxe files",
236 "get - try to retrieve a pxe file using tftp\n"
237 "pxe boot [pxefile_addr_r] - boot from the pxe file at pxefile_addr_r\n"
Jason Hobbs0e3a5932011-08-31 10:37:30 -0500238);
Stephen Warren857291b2014-02-05 20:49:20 -0700239#endif