*/
 efi_status_t efi_parse_options(char const *cmdline)
 {
-       char *str;
-
-       str = strstr(cmdline, "nokaslr");
-       if (str == cmdline || (str && str > cmdline && *(str - 1) == ' '))
-               efi_nokaslr = true;
-
-       str = strstr(cmdline, "quiet");
-       if (str == cmdline || (str && str > cmdline && *(str - 1) == ' '))
-               efi_quiet = true;
-
-       /*
-        * If no EFI parameters were specified on the cmdline we've got
-        * nothing to do.
-        */
-       str = strstr(cmdline, "efi=");
-       if (!str)
-               return EFI_SUCCESS;
-
-       /* Skip ahead to first argument */
-       str += strlen("efi=");
-
-       /*
-        * Remember, because efi= is also used by the kernel we need to
-        * skip over arguments we don't understand.
-        */
-       while (*str && *str != ' ') {
-               if (!strncmp(str, "nochunk", 7)) {
-                       str += strlen("nochunk");
-                       efi_nochunk = true;
-               }
+       size_t len = strlen(cmdline) + 1;
+       efi_status_t status;
+       char *str, *buf;
 
-               if (!strncmp(str, "novamap", 7)) {
-                       str += strlen("novamap");
-                       efi_novamap = true;
-               }
+       status = efi_bs_call(allocate_pool, EFI_LOADER_DATA, len, (void **)&buf);
+       if (status != EFI_SUCCESS)
+               return status;
 
-               if (IS_ENABLED(CONFIG_EFI_SOFT_RESERVE) &&
-                   !strncmp(str, "nosoftreserve", 7)) {
-                       str += strlen("nosoftreserve");
-                       efi_nosoftreserve = true;
-               }
+       str = skip_spaces(memcpy(buf, cmdline, len));
 
-               if (!strncmp(str, "disable_early_pci_dma", 21)) {
-                       str += strlen("disable_early_pci_dma");
-                       efi_disable_pci_dma = true;
-               }
+       while (*str) {
+               char *param, *val;
 
-               if (!strncmp(str, "no_disable_early_pci_dma", 24)) {
-                       str += strlen("no_disable_early_pci_dma");
-                       efi_disable_pci_dma = false;
-               }
+               str = next_arg(str, ¶m, &val);
 
-               /* Group words together, delimited by "," */
-               while (*str && *str != ' ' && *str != ',')
-                       str++;
+               if (!strcmp(param, "nokaslr")) {
+                       efi_nokaslr = true;
+               } else if (!strcmp(param, "quiet")) {
+                       efi_quiet = true;
+               } else if (!strcmp(param, "efi") && val) {
+                       efi_nochunk = parse_option_str(val, "nochunk");
+                       efi_novamap = parse_option_str(val, "novamap");
 
-               if (*str == ',')
-                       str++;
-       }
+                       efi_nosoftreserve = IS_ENABLED(CONFIG_EFI_SOFT_RESERVE) &&
+                                           parse_option_str(val, "nosoftreserve");
 
+                       if (parse_option_str(val, "disable_early_pci_dma"))
+                               efi_disable_pci_dma = true;
+                       if (parse_option_str(val, "no_disable_early_pci_dma"))
+                               efi_disable_pci_dma = false;
+               }
+       }
+       efi_bs_call(free_pool, buf);
        return EFI_SUCCESS;
 }
 
 
  *  Copyright (C) 1991, 1992  Linus Torvalds
  */
 
+#include <linux/ctype.h>
 #include <linux/types.h>
 #include <linux/string.h>
 
        return 0;
 }
 #endif
+
+/* Works only for digits and letters, but small and fast */
+#define TOLOWER(x) ((x) | 0x20)
+
+static unsigned int simple_guess_base(const char *cp)
+{
+       if (cp[0] == '0') {
+               if (TOLOWER(cp[1]) == 'x' && isxdigit(cp[2]))
+                       return 16;
+               else
+                       return 8;
+       } else {
+               return 10;
+       }
+}
+
+/**
+ * simple_strtoull - convert a string to an unsigned long long
+ * @cp: The start of the string
+ * @endp: A pointer to the end of the parsed string will be placed here
+ * @base: The number base to use
+ */
+
+unsigned long long simple_strtoull(const char *cp, char **endp, unsigned int base)
+{
+       unsigned long long result = 0;
+
+       if (!base)
+               base = simple_guess_base(cp);
+
+       if (base == 16 && cp[0] == '0' && TOLOWER(cp[1]) == 'x')
+               cp += 2;
+
+       while (isxdigit(*cp)) {
+               unsigned int value;
+
+               value = isdigit(*cp) ? *cp - '0' : TOLOWER(*cp) - 'a' + 10;
+               if (value >= base)
+                       break;
+               result = result * base + value;
+               cp++;
+       }
+       if (endp)
+               *endp = (char *)cp;
+
+       return result;
+}
+
+long simple_strtol(const char *cp, char **endp, unsigned int base)
+{
+       if (*cp == '-')
+               return -simple_strtoull(cp + 1, endp, base);
+
+       return simple_strtoull(cp, endp, base);
+}