]> git.dujemihanovic.xyz Git - u-boot.git/commitdiff
cmd: efidebug: add uri device path
authorMasahisa Kojima <masahisa.kojima@linaro.org>
Fri, 10 Nov 2023 04:25:41 +0000 (13:25 +0900)
committerIlias Apalodimas <ilias.apalodimas@linaro.org>
Sat, 18 Nov 2023 08:08:09 +0000 (10:08 +0200)
This adds the URI device path option for 'boot add' subcommand.
User can add the URI load option for downloading ISO image file
or EFI application through network. Currently HTTP is only supported.

Signed-off-by: Masahisa Kojima <masahisa.kojima@linaro.org>
Signed-off-by: Ilias Apalodimas <ilias.apalodimas@linaro.org>
cmd/efidebug.c
include/net.h
net/wget.c

index 201531ac19fc66b1f22fb54868259858750debdd..78ef16f4cb5cfc9d96c419136f707a501a362a49 100644 (file)
@@ -19,6 +19,7 @@
 #include <log.h>
 #include <malloc.h>
 #include <mapmem.h>
+#include <net.h>
 #include <part.h>
 #include <search.h>
 #include <linux/ctype.h>
@@ -707,6 +708,65 @@ out:
        return initrd_dp;
 }
 
+/**
+ * efi_boot_add_uri() - set URI load option
+ *
+ * @argc:              Number of arguments
+ * @argv:              Argument array
+ * @var_name16:                variable name buffer
+ * @var_name16_size:   variable name buffer size
+ * @lo:                        pointer to the load option
+ * @file_path:         buffer to set the generated device path pointer
+ * @fp_size:           file_path size
+ * Return:             CMD_RET_SUCCESS on success,
+ *                     CMD_RET_USAGE or CMD_RET_RET_FAILURE on failure
+ */
+static int efi_boot_add_uri(int argc, char *const argv[], u16 *var_name16,
+                           size_t var_name16_size, struct efi_load_option *lo,
+                           struct efi_device_path **file_path,
+                           efi_uintn_t *fp_size)
+{
+       int id;
+       char *pos;
+       char *endp;
+       u16 *label;
+       efi_uintn_t uridp_len;
+       struct efi_device_path_uri *uridp;
+
+       if (argc < 3 || lo->label)
+               return CMD_RET_USAGE;
+
+       id = (int)hextoul(argv[1], &endp);
+       if (*endp != '\0' || id > 0xffff)
+               return CMD_RET_USAGE;
+
+       label = efi_convert_string(argv[2]);
+       if (!label)
+               return CMD_RET_FAILURE;
+
+       if (!wget_validate_uri(argv[3])) {
+               printf("ERROR: invalid URI\n");
+               return CMD_RET_FAILURE;
+       }
+
+       efi_create_indexed_name(var_name16, var_name16_size, "Boot", id);
+       lo->label = label;
+
+       uridp_len = sizeof(struct efi_device_path) + strlen(argv[3]) + 1;
+       uridp = efi_alloc(uridp_len + sizeof(END));
+       uridp->dp.type = DEVICE_PATH_TYPE_MESSAGING_DEVICE;
+       uridp->dp.sub_type = DEVICE_PATH_SUB_TYPE_MSG_URI;
+       uridp->dp.length = uridp_len;
+       strcpy(uridp->uri, argv[3]);
+       pos = (char *)uridp + uridp_len;
+       memcpy(pos, &END, sizeof(END));
+
+       *file_path = &uridp->dp;
+       *fp_size += uridp_len + sizeof(END);
+
+       return CMD_RET_SUCCESS;
+}
+
 /**
  * do_efi_boot_add() - set UEFI load option
  *
@@ -829,6 +889,21 @@ static int do_efi_boot_add(struct cmd_tbl *cmdtp, int flag,
                        argc -= 1;
                        argv += 1;
                        break;
+               case 'u':
+                       if (IS_ENABLED(CONFIG_EFI_HTTP_BOOT)) {
+                               r = efi_boot_add_uri(argc, argv, var_name16,
+                                                    sizeof(var_name16), &lo,
+                                                    &file_path, &fp_size);
+                               if (r != CMD_RET_SUCCESS)
+                                       goto out;
+                               fp_free = file_path;
+                               argc -= 3;
+                               argv += 3;
+                       } else{
+                               r = CMD_RET_USAGE;
+                               goto out;
+                       }
+                       break;
                default:
                        r = CMD_RET_USAGE;
                        goto out;
@@ -1491,6 +1566,9 @@ U_BOOT_LONGHELP(efidebug,
        "  -b|-B <bootid> <label> <interface> <devnum>[:<part>] <file path>\n"
        "  -i|-I <interface> <devnum>[:<part>] <initrd file path>\n"
        "  (-b, -i for short form device path)\n"
+#if (IS_ENABLED(CONFIG_EFI_HTTP_BOOT))
+       "  -u <bootid> <label> <uri>\n"
+#endif
        "  -s '<optional data>'\n"
        "efidebug boot rm <bootid#1> [<bootid#2> [<bootid#3> [...]]]\n"
        "  - delete UEFI BootXXXX variables\n"
index 61f9018769a3b7726577c05338bd6ef4272d9460..ac511eab103c0025050b8529a22fc3fb442f02bd 100644 (file)
@@ -939,4 +939,12 @@ static inline void eth_set_enable_bootdevs(bool enable) {}
  */
 int wget_with_dns(ulong dst_addr, char *uri);
 
+/**
+ * wget_validate_uri() - varidate the uri
+ *
+ * @uri:       uri string of target file of wget
+ * Return:     true if uri is valid, false if uri is invalid
+ */
+bool wget_validate_uri(char *uri);
+
 #endif /* __NET_H__ */
index 2087146b3771d1124c32d59be0bbd8cd031d2065..6ae2237a0a8c5276a928a5301cf2f4dfb4e83417 100644 (file)
@@ -566,3 +566,74 @@ out:
        return ret;
 }
 #endif
+
+/**
+ * wget_validate_uri() - validate the uri for wget
+ *
+ * @uri:       uri string
+ *
+ * This function follows the current U-Boot wget implementation.
+ * scheme: only "http:" is supported
+ * authority:
+ *   - user information: not supported
+ *   - host: supported
+ *   - port: not supported(always use the default port)
+ *
+ * Uri is expected to be correctly percent encoded.
+ * This is the minimum check, control codes(0x1-0x19, 0x7F, except '\0')
+ * and space character(0x20) are not allowed.
+ *
+ * TODO: stricter uri conformance check
+ *
+ * Return:     true on success, false on failure
+ */
+bool wget_validate_uri(char *uri)
+{
+       char c;
+       bool ret = true;
+       char *str_copy, *s, *authority;
+
+       for (c = 0x1; c < 0x21; c++) {
+               if (strchr(uri, c)) {
+                       log_err("invalid character is used\n");
+                       return false;
+               }
+       }
+       if (strchr(uri, 0x7f)) {
+               log_err("invalid character is used\n");
+               return false;
+       }
+
+       if (strncmp(uri, "http://", 7)) {
+               log_err("only http:// is supported\n");
+               return false;
+       }
+       str_copy = strdup(uri);
+       if (!str_copy)
+               return false;
+
+       s = str_copy + strlen("http://");
+       authority = strsep(&s, "/");
+       if (!s) {
+               log_err("invalid uri, no file path\n");
+               ret = false;
+               goto out;
+       }
+       s = strchr(authority, '@');
+       if (s) {
+               log_err("user information is not supported\n");
+               ret = false;
+               goto out;
+       }
+       s = strchr(authority, ':');
+       if (s) {
+               log_err("user defined port is not supported\n");
+               ret = false;
+               goto out;
+       }
+
+out:
+       free(str_copy);
+
+       return ret;
+}