Blame SOURCES/0037-Implement-efivar-export-foo.var.patch

a43681
From 488976e782a7307cc2df4e1a8aaea1f6dfa757dd Mon Sep 17 00:00:00 2001
a43681
From: Peter Jones <pjones@redhat.com>
a43681
Date: Mon, 17 Jun 2019 15:04:11 -0400
a43681
Subject: [PATCH 37/63] Implement 'efivar --export=foo.var'
a43681
a43681
Signed-off-by: Peter Jones <pjones@redhat.com>
a43681
---
a43681
 .gitignore                  |  17 +-
a43681
 src/Makefile                |   2 +-
a43681
 src/crc32.h                 |  19 ++
a43681
 src/efivar.c                | 362 ++++++++++++++++++++------
a43681
 src/export.c                | 502 +++++++++++++++++++++++++++++++++---
a43681
 src/gpt.c                   |  18 --
a43681
 src/include/efivar/efivar.h |   3 +
a43681
 7 files changed, 783 insertions(+), 140 deletions(-)
a43681
a43681
diff --git a/.gitignore b/.gitignore
a43681
index 5c3fd0e3f52..947d88eec42 100644
a43681
--- a/.gitignore
a43681
+++ b/.gitignore
a43681
@@ -1,23 +1,26 @@
a43681
+.*.c.P
a43681
+.*.h.P
a43681
+.*.d
a43681
 .*.sw?
a43681
 *~
a43681
 *.a
a43681
+*.env
a43681
 *.E
a43681
 *.o
a43681
+*.map
a43681
 *.pc
a43681
 *.S
a43681
 !src/guids.S
a43681
 *.so
a43681
 *.so.*
a43681
+*.spec
a43681
 *.tar.*
a43681
-.*.c.P
a43681
-.*.h.P
a43681
-.*.d
a43681
+*.var
a43681
 core.*
a43681
-*.spec
a43681
+cov-int
a43681
+vgcore.*
a43681
+scan-results/
a43681
 src/efivar
a43681
 src/efivar-static
a43681
 src/makeguids
a43681
 src/guid-symbols.c
a43681
-*.map
a43681
-cov-int
a43681
-scan-results/
a43681
diff --git a/src/Makefile b/src/Makefile
a43681
index ecbbc02e1f7..addfaa03c85 100644
a43681
--- a/src/Makefile
a43681
+++ b/src/Makefile
a43681
@@ -17,7 +17,7 @@ STATICTARGETS=$(STATICLIBTARGETS) $(STATICBINTARGETS)
a43681
 LIBEFIBOOT_SOURCES = crc32.c creator.c disk.c gpt.c loadopt.c path-helpers.c \
a43681
 		     linux.c $(wildcard linux-*.c)
a43681
 LIBEFIBOOT_OBJECTS = $(patsubst %.c,%.o,$(LIBEFIBOOT_SOURCES))
a43681
-LIBEFIVAR_SOURCES = dp.c dp-acpi.c dp-hw.c dp-media.c dp-message.c \
a43681
+LIBEFIVAR_SOURCES = crc32.c dp.c dp-acpi.c dp-hw.c dp-media.c dp-message.c \
a43681
 	efivarfs.c error.c export.c guid.c guids.S guid-symbols.c \
a43681
 	lib.c vars.c
a43681
 LIBEFIVAR_OBJECTS = $(patsubst %.S,%.o,$(patsubst %.c,%.o,$(LIBEFIVAR_SOURCES)))
a43681
diff --git a/src/crc32.h b/src/crc32.h
a43681
index b5b975a5768..4e833aafac0 100644
a43681
--- a/src/crc32.h
a43681
+++ b/src/crc32.h
a43681
@@ -29,6 +29,25 @@
a43681
 
a43681
 extern uint32_t crc32 (const void *buf, unsigned long len, uint32_t seed);
a43681
 
a43681
+/**
a43681
+ * efi_crc32() - EFI version of crc32 function
a43681
+ * @buf: buffer to calculate crc32 of
a43681
+ * @len - length of buf
a43681
+ *
a43681
+ * Description: Returns EFI-style CRC32 value for @buf
a43681
+ *
a43681
+ * This function uses the little endian Ethernet polynomial
a43681
+ * but seeds the function with ~0, and xor's with ~0 at the end.
a43681
+ * Note, the EFI Specification, v1.02, has a reference to
a43681
+ * Dr. Dobbs Journal, May 1994 (actually it's in May 1992).
a43681
+ */
a43681
+static inline uint32_t
a43681
+efi_crc32(const void *buf, unsigned long len)
a43681
+{
a43681
+	return (crc32(buf, len, ~0L) ^ ~0L);
a43681
+}
a43681
+
a43681
+
a43681
 #endif /* _CRC32_H */
a43681
 
a43681
 // vim:fenc=utf-8:tw=75:noet
a43681
diff --git a/src/efivar.c b/src/efivar.c
a43681
index 7f16ab15bab..885a9af864b 100644
a43681
--- a/src/efivar.c
a43681
+++ b/src/efivar.c
a43681
@@ -46,6 +46,8 @@ extern int optind, opterr, optopt;
a43681
 #define ACTION_LIST_GUIDS	0x08
a43681
 #define ACTION_WRITE		0x10
a43681
 #define ACTION_PRINT_DEC	0x20
a43681
+#define ACTION_IMPORT		0x40
a43681
+#define ACTION_EXPORT		0x80
a43681
 
a43681
 #define EDIT_APPEND	0
a43681
 #define EDIT_WRITE	1
a43681
@@ -173,31 +175,10 @@ bad_name:
a43681
 }
a43681
 
a43681
 static void
a43681
-show_variable(char *guid_name, int display_type)
a43681
+show_variable_data(efi_guid_t guid, const char *name, uint32_t attributes,
a43681
+		   uint8_t *data, size_t data_size,
a43681
+		   int display_type)
a43681
 {
a43681
-	efi_guid_t guid = efi_guid_empty;
a43681
-	char *name = NULL;
a43681
-	int rc;
a43681
-
a43681
-	uint8_t *data = NULL;
a43681
-	size_t data_size = 0;
a43681
-	uint32_t attributes;
a43681
-
a43681
-	parse_name(guid_name, &name, &guid);
a43681
-	if (!name || efi_guid_is_empty(&guid)) {
a43681
-		fprintf(stderr, "efivar: could not parse variable name.\n");
a43681
-		show_errors();
a43681
-		exit(1);
a43681
-	}
a43681
-
a43681
-	errno = 0;
a43681
-	rc = efi_get_variable(guid, name, &data, &data_size, &attributes);
a43681
-	if (rc < 0) {
a43681
-		fprintf(stderr, "efivar: show variable: %m\n");
a43681
-		show_errors();
a43681
-		exit(1);
a43681
-	}
a43681
-
a43681
 	if (display_type == SHOW_VERBOSE) {
a43681
 		printf("GUID: "GUID_FORMAT "\n",
a43681
 		       guid.a, guid.b, guid.c, bswap_16(guid.d),
a43681
@@ -257,6 +238,117 @@ show_variable(char *guid_name, int display_type)
a43681
 		}
a43681
 		printf("\n");
a43681
 	}
a43681
+}
a43681
+
a43681
+static void
a43681
+show_variable(char *guid_name, int display_type)
a43681
+{
a43681
+	efi_guid_t guid = efi_guid_empty;
a43681
+	char *name = NULL;
a43681
+	int rc;
a43681
+
a43681
+	uint8_t *data = NULL;
a43681
+	size_t data_size = 0;
a43681
+	uint32_t attributes;
a43681
+
a43681
+	parse_name(guid_name, &name, &guid);
a43681
+	if (!name || efi_guid_is_empty(&guid)) {
a43681
+		fprintf(stderr, "efivar: could not parse variable name.\n");
a43681
+		show_errors();
a43681
+		exit(1);
a43681
+	}
a43681
+
a43681
+	errno = 0;
a43681
+	rc = efi_get_variable(guid, name, &data, &data_size, &attributes);
a43681
+	if (rc < 0) {
a43681
+		fprintf(stderr, "efivar: show variable: %m\n");
a43681
+		show_errors();
a43681
+		exit(1);
a43681
+	}
a43681
+
a43681
+	show_variable_data(guid, name, attributes,
a43681
+			   data, data_size, display_type);
a43681
+
a43681
+	free(name);
a43681
+	if (data)
a43681
+		free(data);
a43681
+}
a43681
+
a43681
+static void
a43681
+save_variable_data(efi_variable_t *var, char *outfile, bool dmpstore)
a43681
+{
a43681
+	FILE *out = NULL;
a43681
+	ssize_t sz;
a43681
+	uint8_t *data = NULL;
a43681
+	size_t datasz = 0;
a43681
+	ssize_t (*export)(efi_variable_t *var, uint8_t *data, size_t size) =
a43681
+		dmpstore ? efi_variable_export_dmpstore : efi_variable_export;
a43681
+
a43681
+	out = fopen(outfile, "w");
a43681
+	if (!out)
a43681
+		err(1, "Could not open \"%s\" for writing", outfile);
a43681
+
a43681
+	sz = export(var, data, datasz);
a43681
+	data = calloc(sz, 1);
a43681
+	if (!data)
a43681
+		err(1, "Could not allocate memory");
a43681
+	datasz = sz;
a43681
+
a43681
+	sz = export(var, data, datasz);
a43681
+	if (sz < 0)
a43681
+		err(1, "Could not format data");
a43681
+	datasz = sz;
a43681
+
a43681
+	sz = fwrite(data, 1, datasz, out);
a43681
+	if (sz < (ssize_t)datasz)
a43681
+		err(1, "Could not write to \"%s\"", outfile);
a43681
+
a43681
+	fflush(out);
a43681
+	fclose(out);
a43681
+}
a43681
+
a43681
+static void
a43681
+save_variable(char *guid_name, char *outfile, bool dmpstore)
a43681
+{
a43681
+	efi_guid_t guid = efi_guid_empty;
a43681
+	char *name = NULL;
a43681
+	int rc;
a43681
+
a43681
+	uint8_t *data = NULL;
a43681
+	size_t data_size = 0;
a43681
+	uint32_t attributes = 7;
a43681
+	efi_variable_t *var;
a43681
+
a43681
+	parse_name(guid_name, &name, &guid);
a43681
+	if (!name || efi_guid_is_empty(&guid)) {
a43681
+		fprintf(stderr, "efivar: could not parse variable name.\n");
a43681
+		show_errors();
a43681
+		exit(1);
a43681
+	}
a43681
+
a43681
+	errno = 0;
a43681
+	rc = efi_get_variable(guid, name, &data, &data_size, &attributes);
a43681
+	if (rc < 0) {
a43681
+		fprintf(stderr, "efivar: show variable: %m\n");
a43681
+		show_errors();
a43681
+		exit(1);
a43681
+	}
a43681
+
a43681
+	var = efi_variable_alloc();
a43681
+	if (!var) {
a43681
+		fprintf(stderr, "efivar: could not allocate variable storage.\n");
a43681
+		show_errors();
a43681
+		exit(1);
a43681
+	}
a43681
+
a43681
+	efi_variable_set_name(var, (unsigned char *)name);
a43681
+	efi_variable_set_guid(var, &guid);
a43681
+	efi_variable_set_attributes(var, attributes);
a43681
+	efi_variable_set_data(var, data, data_size);
a43681
+
a43681
+	save_variable_data(var, outfile, dmpstore);
a43681
+
a43681
+	efi_variable_free(var, false);
a43681
 
a43681
 	free(name);
a43681
 	if (data)
a43681
@@ -372,16 +464,18 @@ usage(int ret)
a43681
 	FILE *out = ret == 0 ? stdout : stderr;
a43681
 	fprintf(out,
a43681
 		"Usage: %s [OPTION...]\n"
a43681
+		"  -t, --attributes=<attributes>     attributes to use on append\n"
a43681
 		"  -l, --list                        list current variables\n"
a43681
 		"  -p, --print                       print variable specified by --name\n"
a43681
+		"  -D, --dmpstore                    use DMPSTORE format when exporting\n"
a43681
 		"  -d, --print-decimal               print variable in decimal values specified\n"
a43681
 		"                                    by --name\n"
a43681
 		"  -n, --name=<guid-name>            variable to manipulate, in the form\n"
a43681
 		"                                    8be4df61-93ca-11d2-aa0d-00e098032b8c-Boot0000\n"
a43681
 		"  -a, --append                      append to variable specified by --name\n"
a43681
+		"  -f, --datafile=<file>             load or save variable contents from <file>\n"
a43681
 		"  -e, --export=<file>               export variable to <file>\n"
a43681
-		"  -f, --fromfile=<file>             use data from <file>\n"
a43681
-		"  -t, --attributes=<attributes>     attributes to use on append\n"
a43681
+		"  -i, --import=<file>               import variable from 
a43681
 		"  -L, --list-guids                  show internal guid list\n"
a43681
 		"  -w, --write                       write to variable specified by --name\n\n"
a43681
 		"Help options:\n"
a43681
@@ -398,55 +492,79 @@ int main(int argc, char *argv[])
a43681
 	int action = 0;
a43681
 	uint8_t *data = NULL;
a43681
 	size_t data_size = 0;
a43681
-	char *name = NULL;
a43681
-	char *file = NULL;
a43681
-	uint32_t attributes = 0;
a43681
-	char *sopts = "lpdn:af:t:Lw?";
a43681
-	struct option lopts[] =
a43681
-		{ {"list", no_argument, 0, 'l'},
a43681
-		  {"print", no_argument, 0, 'p'},
a43681
-		  {"print-decimal", no_argument, 0, 'd'},
a43681
-		  {"name", required_argument, 0, 'n'},
a43681
-		  {"append", no_argument, 0, 'a'},
a43681
-		  {"fromfile", required_argument, 0, 'f'},
a43681
-		  {"attributes", required_argument, 0, 't'},
a43681
-		  {"list-guids", no_argument, 0, 'L'},
a43681
-		  {"write", no_argument, 0, 'w'},
a43681
-		  {"help", no_argument, 0, '?'},
a43681
-		  {"usage", no_argument, 0, 0},
a43681
-		  {0, 0, 0, 0}
a43681
-		};
a43681
+	char *guid_name = NULL;
a43681
+	char *infile = NULL;
a43681
+	char *outfile = NULL;
a43681
+	char *datafile = NULL;
a43681
+	bool dmpstore = false;
a43681
+	int verbose = 0;
a43681
+	uint32_t attributes = EFI_VARIABLE_NON_VOLATILE
a43681
+			      | EFI_VARIABLE_BOOTSERVICE_ACCESS
a43681
+			      | EFI_VARIABLE_RUNTIME_ACCESS;
a43681
+	char *sopts = "aA:Dde:f:i:Llpn:vw?";
a43681
+	struct option lopts[] = {
a43681
+		{"append", no_argument, 0, 'a'},
a43681
+		{"attributes", required_argument, 0, 'A'},
a43681
+		{"datafile", required_argument, 0, 'f'},
a43681
+		{"dmpstore", no_argument, 0, 'D'},
a43681
+		{"export", required_argument, 0, 'e'},
a43681
+		{"help", no_argument, 0, '?'},
a43681
+		{"import", required_argument, 0, 'i'},
a43681
+		{"list", no_argument, 0, 'l'},
a43681
+		{"list-guids", no_argument, 0, 'L'},
a43681
+		{"name", required_argument, 0, 'n'},
a43681
+		{"print", no_argument, 0, 'p'},
a43681
+		{"print-decimal", no_argument, 0, 'd'},
a43681
+		{"usage", no_argument, 0, 0},
a43681
+		{"verbose", no_argument, 0, 'v'},
a43681
+		{"write", no_argument, 0, 'w'},
a43681
+		{0, 0, 0, 0}
a43681
+	};
a43681
 
a43681
 	while ((c = getopt_long(argc, argv, sopts, lopts, &i)) != -1) {
a43681
 		switch (c) {
a43681
-			case 'l':
a43681
-				action |= ACTION_LIST;
a43681
+			case 'A':
a43681
+				attributes = strtoul(optarg, NULL, 10);
a43681
+				if (errno == ERANGE || errno == EINVAL)
a43681
+					err(1,
a43681
+					    "invalid argument for -t: %s: %m\n",
a43681
+					    optarg);
a43681
 				break;
a43681
-			case 'p':
a43681
-				action |= ACTION_PRINT;
a43681
+			case 'a':
a43681
+				action |= ACTION_APPEND;
a43681
+				break;
a43681
+			case 'D':
a43681
+				dmpstore = true;
a43681
 				break;
a43681
 			case 'd':
a43681
 				action |= ACTION_PRINT_DEC;
a43681
 				break;
a43681
-			case 'n':
a43681
-				name = optarg;
a43681
-				break;
a43681
-			case 'a':
a43681
-				action |= ACTION_APPEND;
a43681
+			case 'e':
a43681
+				action |= ACTION_EXPORT;
a43681
+				outfile = optarg;
a43681
 				break;
a43681
 			case 'f':
a43681
-				file = optarg;
a43681
+				datafile = optarg;
a43681
 				break;
a43681
-			case 't':
a43681
-				attributes = strtoul(optarg, NULL, 10);
a43681
-				if (errno == ERANGE || errno == EINVAL)
a43681
-					err(1,
a43681
-					    "invalid argument for -t: %s: %m\n",
a43681
-					    optarg);
a43681
+			case 'i':
a43681
+				action |= ACTION_IMPORT;
a43681
+				infile = optarg;
a43681
 				break;
a43681
 			case 'L':
a43681
 				action |= ACTION_LIST_GUIDS;
a43681
 				break;
a43681
+			case 'l':
a43681
+				action |= ACTION_LIST;
a43681
+				break;
a43681
+			case 'n':
a43681
+				guid_name = optarg;
a43681
+				break;
a43681
+			case 'p':
a43681
+				action |= ACTION_PRINT;
a43681
+				break;
a43681
+			case 'v':
a43681
+				verbose += 1;
a43681
+				break;
a43681
 			case 'w':
a43681
 				action |= ACTION_WRITE;
a43681
 				break;
a43681
@@ -460,7 +578,9 @@ int main(int argc, char *argv[])
a43681
 		}
a43681
 	}
a43681
 
a43681
-	if (name)
a43681
+	efi_set_verbose(verbose, stderr);
a43681
+
a43681
+	if (guid_name && !outfile)
a43681
 		action |= ACTION_PRINT;
a43681
 
a43681
 	switch (action) {
a43681
@@ -468,23 +588,23 @@ int main(int argc, char *argv[])
a43681
 			list_all_variables();
a43681
 			break;
a43681
 		case ACTION_PRINT:
a43681
-			validate_name(name);
a43681
-			show_variable(name, SHOW_VERBOSE);
a43681
+			validate_name(guid_name);
a43681
+			show_variable(guid_name, SHOW_VERBOSE);
a43681
 			break;
a43681
 		case ACTION_PRINT_DEC | ACTION_PRINT:
a43681
-			validate_name(name);
a43681
-			show_variable(name, SHOW_DECIMAL);
a43681
+			validate_name(guid_name);
a43681
+			show_variable(guid_name, SHOW_DECIMAL);
a43681
 			break;
a43681
 		case ACTION_APPEND | ACTION_PRINT:
a43681
-			validate_name(name);
a43681
-			prepare_data(file, &data, &data_size);
a43681
-			edit_variable(name, data, data_size, attributes,
a43681
+			validate_name(guid_name);
a43681
+			prepare_data(infile, &data, &data_size);
a43681
+			edit_variable(guid_name, data, data_size, attributes,
a43681
 				      EDIT_APPEND);
a43681
 			break;
a43681
 		case ACTION_WRITE | ACTION_PRINT:
a43681
-			validate_name(name);
a43681
-			prepare_data(file, &data, &data_size);
a43681
-			edit_variable(name, data, data_size, attributes,
a43681
+			validate_name(guid_name);
a43681
+			prepare_data(infile, &data, &data_size);
a43681
+			edit_variable(guid_name, data, data_size, attributes,
a43681
 				      EDIT_WRITE);
a43681
 			break;
a43681
 		case ACTION_LIST_GUIDS: {
a43681
@@ -509,10 +629,108 @@ int main(int argc, char *argv[])
a43681
 					guid[i].symbol + strlen("efi_guid_"),
a43681
 					guid[i].symbol, guid[i].name);
a43681
 			}
a43681
+			break;
a43681
+					}
a43681
+		case ACTION_EXPORT:
a43681
+			if (datafile) {
a43681
+				char *name = NULL;
a43681
+				efi_guid_t guid = efi_guid_zero;
a43681
+				efi_variable_t *var;
a43681
+
a43681
+				parse_name(guid_name, &name, &guid);
a43681
+				prepare_data(datafile, &data, &data_size);
a43681
+
a43681
+				var = efi_variable_alloc();
a43681
+				if (!var)
a43681
+					err(1, "Could not allocate memory");
a43681
+
a43681
+				efi_variable_set_name(var, (unsigned char *)name);
a43681
+				efi_variable_set_guid(var, &guid);
a43681
+				efi_variable_set_attributes(var, attributes);
a43681
+				efi_variable_set_data(var, data, data_size);
a43681
+
a43681
+				save_variable_data(var, outfile, dmpstore);
a43681
+
a43681
+				efi_variable_free(var, false);
a43681
+			} else {
a43681
+				validate_name(guid_name);
a43681
+				save_variable(guid_name, outfile, dmpstore);
a43681
+			}
a43681
+			break;
a43681
+		case ACTION_IMPORT:
a43681
+		case ACTION_IMPORT | ACTION_PRINT:
a43681
+		case ACTION_IMPORT | ACTION_PRINT | ACTION_PRINT_DEC:
a43681
+			{
a43681
+				ssize_t sz;
a43681
+				efi_variable_t *var = NULL;
a43681
+				char *name;
a43681
+				efi_guid_t *guid;
a43681
+				uint64_t attributes;
a43681
+				int display_type = (action & ACTION_PRINT_DEC)
a43681
+					? SHOW_VERBOSE|SHOW_DECIMAL
a43681
+					: SHOW_VERBOSE;
a43681
+
a43681
+
a43681
+				prepare_data(infile, &data, &data_size);
a43681
+				sz = efi_variable_import(data, data_size, &var);
a43681
+				if (sz < 0)
a43681
+					err(1, "Could not import data from \"%s\"", infile);
a43681
+
a43681
+				free(data);
a43681
+				data = NULL;
a43681
+				data_size = 0;
a43681
+
a43681
+				name = (char *)efi_variable_get_name(var);
a43681
+				efi_variable_get_guid(var, &guid);
a43681
+				efi_variable_get_attributes(var, &attributes);
a43681
+				efi_variable_get_data(var, &data, &data_size);
a43681
+
a43681
+				if (datafile) {
a43681
+					FILE *out;
a43681
+					int rc;
a43681
+
a43681
+					out = fopen(datafile, "w");
a43681
+					if (!out)
a43681
+						err(1, "Could not open \"%s\" for writing",
a43681
+						    datafile);
a43681
+
a43681
+					rc = fwrite(data, data_size, 1, out);
a43681
+					if (rc < (long)data_size)
a43681
+						err(1, "Could not write to \"%s\"",
a43681
+						    datafile);
a43681
+
a43681
+					fclose(out);
a43681
+					free(guid_name);
a43681
+				}
a43681
+				if (action & ACTION_PRINT)
a43681
+					show_variable_data(*guid, name,
a43681
+						((uint32_t)(attributes & 0xffffffff)),
a43681
+						 data, data_size, display_type);
a43681
+
a43681
+				efi_variable_free(var, false);
a43681
+				break;
a43681
+			}
a43681
+		case ACTION_IMPORT | ACTION_EXPORT:
a43681
+			{
a43681
+				efi_variable_t *var = NULL;
a43681
+				ssize_t sz;
a43681
+
a43681
+				if (datafile)
a43681
+					errx(1, "--datafile cannot be used with --import and --export");
a43681
+
a43681
+				prepare_data(infile, &data, &data_size);
a43681
+				sz = efi_variable_import(data, data_size, &var);
a43681
+				if (sz < 0)
a43681
+					err(1, "Could not import data from \"%s\"", infile);
a43681
+
a43681
+				save_variable_data(var, outfile, dmpstore);
a43681
+
a43681
+				efi_variable_free(var, false);
a43681
+				break;
a43681
+			}
a43681
 		case ACTION_USAGE:
a43681
 		default:
a43681
 			usage(EXIT_FAILURE);
a43681
-		}
a43681
 	};
a43681
 
a43681
 	return 0;
a43681
diff --git a/src/export.c b/src/export.c
a43681
index 6b78412cce1..cfb021525ff 100644
a43681
--- a/src/export.c
a43681
+++ b/src/export.c
a43681
@@ -27,7 +27,7 @@
a43681
 
a43681
 #include "efivar.h"
a43681
 
a43681
-#define EFIVAR_MAGIC 0xf3df1597
a43681
+#define EFIVAR_MAGIC 0xf3df1597u
a43681
 
a43681
 #define ATTRS_UNSET 0xa5a5a5a5a5a5a5a5
a43681
 #define ATTRS_MASK 0xffffffff
a43681
@@ -50,12 +50,166 @@ struct efi_variable {
a43681
  *	uint32_t data_len;
a43681
  *	char16_t name[];
a43681
  *	uint8_t data[];
a43681
- *	uint32_t magic;
a43681
+ *	uint32_t crc32;
a43681
+ * }
a43681
+ *
a43681
+ * Unfortunately the exported structure from dmpstore is:
a43681
+ * struct {
a43681
+ *	uint32_t name_size; // in bytes
a43681
+ *	uint32_t data_size; // in bytes
a43681
+ *	char16_t name[];
a43681
+ *	efi_guid_t guid;
a43681
+ *	uint32_t attr;
a43681
+ *	unit8_t data[];
a43681
+ *	uint32_t crc32;
a43681
  * }
a43681
  */
a43681
 
a43681
-ssize_t NONNULL(1, 3) PUBLIC
a43681
-efi_variable_import(uint8_t *data, size_t size, efi_variable_t **var_out)
a43681
+#ifdef EFIVAR_BUILD_ENVIRONMENT
a43681
+#error wtf
a43681
+#endif
a43681
+
a43681
+ssize_t NONNULL(1, 3)
a43681
+efi_variable_import_dmpstore(uint8_t *data, size_t size,
a43681
+			     efi_variable_t **var_out)
a43681
+{
a43681
+	efi_variable_t var;
a43681
+	uint32_t namesz;
a43681
+	uint32_t datasz;
a43681
+	size_t min = sizeof (uint32_t)		/* name size */
a43681
+		   + sizeof (uint32_t)		/* data size */
a43681
+		   + sizeof (char16_t)		/* two bytes of name */
a43681
+		   + sizeof (efi_guid_t)	/* guid */
a43681
+		   + sizeof (uint32_t)		/* attr */
a43681
+		   + 1				/* one byte of data */
a43681
+		   + sizeof (uint32_t);		/* crc32 */
a43681
+	size_t sz = sizeof (uint32_t)		/* name size */
a43681
+		  + sizeof (uint32_t)		/* data size */
a43681
+		  + sizeof (efi_guid_t)		/* guid */
a43681
+		  + sizeof (uint32_t)		/* attr */
a43681
+		  + sizeof (uint32_t);		/* crc32 */
a43681
+	uint8_t *ptr = data;
a43681
+	uint32_t crc;
a43681
+	int saved_errno;
a43681
+
a43681
+	if (size <= min) {
a43681
+etoosmall:
a43681
+		errno = EINVAL;
a43681
+		efi_error("data size is too small for dmpstore variable (%zu < %zu)",
a43681
+			  size, min);
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	memset(&var, 0, sizeof(var));
a43681
+
a43681
+	namesz = *(uint32_t *)ptr;
a43681
+	debug("namesz:%"PRIu32, namesz);
a43681
+	ptr += sizeof(uint32_t);
a43681
+
a43681
+	if (namesz <= 2) {
a43681
+		errno = EINVAL;
a43681
+		debug("name size (%"PRIu32") must be greater than 2", namesz);
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	if (namesz % 2 != 0) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("name size (%"PRIu32") cannot be odd", namesz);
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	datasz = *(uint32_t *)ptr;
a43681
+	ptr += sizeof(uint32_t);
a43681
+	debug("datasz:%"PRIu32, datasz);
a43681
+
a43681
+	if (datasz == 0) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("data size (%"PRIu32") must be nonzero", datasz);
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	if (add(sz, namesz, &sz)) {
a43681
+overflow:
a43681
+		errno = EOVERFLOW;
a43681
+		efi_error("arithmetic overflow computing allocation size");
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	if (add(sz, datasz, &min))
a43681
+		goto overflow;
a43681
+
a43681
+	if (size < min)
a43681
+		goto etoosmall;
a43681
+	size = min;
a43681
+
a43681
+	if (!(ptr[namesz - 1] == 0 && ptr[namesz -2] == 0)) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("variable name is not properly terminated.");
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	crc = efi_crc32(data, size - sizeof(uint32_t));
a43681
+	debug("efi_crc32(%p, %lu) -> 0x%"PRIx32", expected 0x%"PRIx32,
a43681
+	      data, size - sizeof(uint32_t), crc,
a43681
+	      *(uint32_t*)(data + size - sizeof(uint32_t)));
a43681
+
a43681
+	if (memcmp(data + size - sizeof(uint32_t),
a43681
+		    &crc, sizeof(uint32_t))) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("crc32 did not match");
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	var.name = ucs2_to_utf8(ptr, -1);
a43681
+	if (!var.name)
a43681
+		goto oom;
a43681
+	ptr += namesz;
a43681
+
a43681
+	var.guid = malloc(sizeof (efi_guid_t));
a43681
+	if (!var.guid)
a43681
+		goto oom;
a43681
+	memcpy(var.guid, ptr, sizeof (efi_guid_t));
a43681
+	ptr += sizeof (efi_guid_t);
a43681
+
a43681
+	var.attrs = *(uint32_t *)ptr;
a43681
+	ptr += sizeof(uint32_t);
a43681
+
a43681
+	var.data_size = datasz;
a43681
+	var.data = malloc(datasz);
a43681
+	if (!var.data) {
a43681
+		efi_error("Could not allocate %"PRIu32" bytes", datasz);
a43681
+		goto oom;
a43681
+	}
a43681
+	memcpy(var.data, ptr, datasz);
a43681
+
a43681
+	if (!*var_out) {
a43681
+		*var_out =malloc(sizeof (var));
a43681
+		if (!*var_out)
a43681
+			goto oom;
a43681
+		memcpy(*var_out, &var, sizeof (var));
a43681
+	} else {
a43681
+		return -1;
a43681
+	}
a43681
+	return size;
a43681
+oom:
a43681
+	saved_errno = errno;
a43681
+
a43681
+	if (var.guid)
a43681
+		free(var.guid);
a43681
+
a43681
+	if (var.name)
a43681
+		free(var.name);
a43681
+
a43681
+	if (var.data)
a43681
+		free(var.data);
a43681
+
a43681
+	errno = saved_errno;
a43681
+	efi_error("Could not allocate memory");
a43681
+	return -1;
a43681
+}
a43681
+
a43681
+ssize_t NONNULL(1, 3)
a43681
+efi_variable_import_efivar(uint8_t *data, size_t datasz, efi_variable_t **var_out)
a43681
 {
a43681
 	efi_variable_t var;
a43681
 	size_t min = sizeof (uint32_t) * 2	/* magic */
a43681
@@ -63,47 +217,83 @@ efi_variable_import(uint8_t *data, size_t size, efi_variable_t **var_out)
a43681
 		   + sizeof (uint64_t)		/* attr */
a43681
 		   + sizeof (efi_guid_t)	/* guid */
a43681
 		   + sizeof (uint32_t) * 2	/* name_len and data_len */
a43681
-		   + sizeof (char16_t)	/* two bytes of name */
a43681
-		   + 1;				/* one byte of data */
a43681
+		   + sizeof (char16_t)		/* two bytes of name */
a43681
+		   + 1				/* one byte of data */
a43681
+		   + 4;				/* crc32 */
a43681
+	uint32_t crc;
a43681
+	uint8_t *ptr = data;
a43681
+	uint32_t magic = EFIVAR_MAGIC;
a43681
+	int test;
a43681
+
a43681
 	errno = EINVAL;
a43681
-	if (size <= min)
a43681
+	if (datasz <= min)
a43681
 		return -1;
a43681
 
a43681
-	uint8_t *ptr = data;
a43681
-	uint32_t magic = EFIVAR_MAGIC;
a43681
-	if (memcmp(data, &magic, sizeof (uint32_t)) ||
a43681
-			memcmp(data + size - sizeof (uint32_t), &magic,
a43681
-				sizeof (uint32_t)))
a43681
+	test = memcmp(data, &magic, sizeof (uint32_t));
a43681
+	debug("test magic 0: cmp(0x%04x,0x%04x)->%d", *(uint32_t *)data, magic, test);
a43681
+	if (test) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("MAGIC for file format did not match.");
a43681
 		return -1;
a43681
-	size -= sizeof (uint32_t);
a43681
+	}
a43681
+
a43681
 	ptr += sizeof (uint32_t);
a43681
 
a43681
+	debug("test version");
a43681
 	if (*(uint32_t *)ptr == 1) {
a43681
 		ptr += sizeof (uint32_t);
a43681
+		debug("version 1");
a43681
+
a43681
 		var.attrs = *(uint64_t *)ptr;
a43681
-		ptr += sizeof (uint32_t);
a43681
+		ptr += sizeof (uint64_t);
a43681
+		debug("var.attrs:0x%08"PRIx64, var.attrs);
a43681
 
a43681
 		var.guid = malloc(sizeof (efi_guid_t));
a43681
 		if (!var.guid)
a43681
 			return -1;
a43681
 		*var.guid = *(efi_guid_t *)ptr;
a43681
 		ptr += sizeof (efi_guid_t);
a43681
+		debug("var.guid:"GUID_FORMAT,
a43681
+		      var.guid->a, var.guid->b, var.guid->c,
a43681
+		      bswap_16(var.guid->d),
a43681
+		      var.guid->e[0], var.guid->e[1], var.guid->e[2],
a43681
+		      var.guid->e[3], var.guid->e[4], var.guid->e[5]);
a43681
 
a43681
 		uint32_t name_len = *(uint32_t *)ptr;
a43681
 		ptr += sizeof (uint32_t);
a43681
+		debug("name_len:%"PRIu32, name_len);
a43681
+
a43681
 		uint32_t data_len = *(uint32_t *)ptr;
a43681
 		ptr += sizeof (uint32_t);
a43681
+		debug("data_len:%"PRIu32, data_len);
a43681
+
a43681
+		min -= 3;
a43681
+		min += name_len;
a43681
+		min += data_len;
a43681
 
a43681
-		if (name_len < 1 ||
a43681
-		    name_len != ((data + size) - ptr - data_len) ||
a43681
+		if (name_len < 2 ||
a43681
+		    name_len > (datasz - data_len) ||
a43681
 		    data_len < 1 ||
a43681
-		    data_len != ((data + size) - ptr - name_len)) {
a43681
+		    data_len > (datasz - name_len)) {
a43681
 			int saved_errno = errno;
a43681
 			free(var.guid);
a43681
 			errno = saved_errno;
a43681
 			return -1;
a43681
 		}
a43681
 
a43681
+		crc = efi_crc32(data, datasz - sizeof(uint32_t));
a43681
+		debug("efi_crc32(%p, %lu) -> 0x%"PRIx32", expected 0x%"PRIx32,
a43681
+		      data, datasz - sizeof(uint32_t), crc,
a43681
+		      *(uint32_t*)(data + datasz - sizeof(uint32_t)));
a43681
+
a43681
+		if (memcmp(data + datasz - sizeof (uint32_t), &crc,
a43681
+			   sizeof (uint32_t))) {
a43681
+			free(var.guid);
a43681
+			errno = EINVAL;
a43681
+			efi_error("crc32 did not match");
a43681
+			return -1;
a43681
+		}
a43681
+
a43681
 		var.name = calloc(1, name_len + 1);
a43681
 		if (!var.name) {
a43681
 			int saved_errno = errno;
a43681
@@ -115,7 +305,8 @@ efi_variable_import(uint8_t *data, size_t size, efi_variable_t **var_out)
a43681
 		char16_t *wname = (char16_t *)ptr;
a43681
 		for (uint32_t i = 0; i < name_len; i++)
a43681
 			var.name[i] = wname[i] & 0xff;
a43681
-		ptr += name_len * 2;
a43681
+		ptr += name_len;
a43681
+		debug("name:%s", var.name);
a43681
 
a43681
 		var.data_size = data_len;
a43681
 		var.data = malloc(data_len);
a43681
@@ -143,31 +334,228 @@ efi_variable_import(uint8_t *data, size_t size, efi_variable_t **var_out)
a43681
 	} else {
a43681
 		return -1;
a43681
 	}
a43681
-	return size;
a43681
+	return min;
a43681
+}
a43681
+
a43681
+ssize_t NONNULL(1, 3) PUBLIC
a43681
+efi_variable_import(uint8_t *data, size_t size, efi_variable_t **var_out)
a43681
+{
a43681
+	ssize_t rc;
a43681
+
a43681
+	rc = efi_variable_import_efivar(data, size, var_out);
a43681
+	if (rc >= 0)
a43681
+		return rc;
a43681
+
a43681
+	rc = efi_variable_import_dmpstore(data, size, var_out);
a43681
+	return rc;
a43681
 }
a43681
 
a43681
 ssize_t NONNULL(1) PUBLIC
a43681
-efi_variable_export(efi_variable_t *var, uint8_t *data, size_t size)
a43681
+efi_variable_export_dmpstore(efi_variable_t *var, uint8_t *data, size_t datasz)
a43681
 {
a43681
-	size_t name_len = strlen((char *)var->name);
a43681
-
a43681
-	size_t needed = sizeof (uint32_t)		/* magic */
a43681
-		      + sizeof (uint32_t)		/* version */
a43681
-		      + sizeof (uint64_t)		/* attr */
a43681
-		      + sizeof (efi_guid_t)		/* guid */
a43681
-		      + sizeof (uint32_t)		/* name_len */
a43681
-		      + sizeof (uint32_t)		/* data_len */
a43681
-		      + sizeof (char16_t) * name_len	/* name */
a43681
-		      + var->data_size			/* data */
a43681
-		      + sizeof (uint32_t);		/* magic again */
a43681
-
a43681
-	if (!data || size == 0) {
a43681
+	uint32_t tmpu32;
a43681
+	ssize_t tmpssz;
a43681
+	uint32_t namesz;
a43681
+	uint32_t needed = sizeof (uint32_t)		/* name_size */
a43681
+			+ sizeof (uint32_t)		/* data_size */
a43681
+			+ 2				/* name */
a43681
+			+ sizeof (efi_guid_t)		/* guid */
a43681
+			+ sizeof (uint32_t)		/* attrs */
a43681
+			+ 1				/* data */
a43681
+			+ 4;				/* crc32 */
a43681
+	uint8_t *ptr;
a43681
+	uint32_t crc;
a43681
+
a43681
+	if (!var) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("var cannot be NULL");
a43681
+		return -1;
a43681
+	}
a43681
+	if (!var->name) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("var->name cannot be NULL");
a43681
+		return -1;
a43681
+	}
a43681
+	if (!var->data) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("var->data cannot be NULL");
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	debug("data: %p datasz: %zu", data, datasz);
a43681
+
a43681
+	namesz = utf8size(var->name, -1);
a43681
+	debug("sizeof(uint16_t):%zd * namesz:%"PRIu32, sizeof(uint16_t), namesz);
a43681
+	if (mul(sizeof (uint16_t), namesz, &namesz)) {
a43681
+overflow:
a43681
+		errno = EOVERFLOW;
a43681
+		efi_error("arithmetic overflow computing name size");
a43681
+		return -1;
a43681
+	}
a43681
+	debug("namesz -> %"PRIu32, namesz);
a43681
+
a43681
+	/*
a43681
+	 * Remove our stand-ins for name size and data size before we add
a43681
+	 * them back in.
a43681
+	 */
a43681
+	needed -= 3;
a43681
+
a43681
+	debug("needed:%"PRIu32" + namesz:%"PRIu32, needed, namesz);
a43681
+	if (add(needed, namesz, &needed))
a43681
+		goto overflow;
a43681
+	debug("needed -> %"PRIu32, needed);
a43681
+
a43681
+	debug("needed:%"PRIu32" + var->data_size:%zd", needed, var->data_size);
a43681
+	if (add(needed, var->data_size, &needed))
a43681
+		goto overflow;
a43681
+	debug("needed -> %"PRIu32, needed);
a43681
+
a43681
+	if (!data || datasz == 0) {
a43681
+		debug("data: %p datasz: %zd -> returning needed size %"PRIu32,
a43681
+		      data, datasz, needed);
a43681
 		return needed;
a43681
-	} else if (size < needed) {
a43681
-		return needed - size;
a43681
 	}
a43681
 
a43681
-	uint8_t *ptr = data;
a43681
+	debug("datasz:%zu needed: %"PRIu32, datasz, needed);
a43681
+	if (datasz < needed) {
a43681
+		efi_error("needed: %"PRIu32" datasz: %zd -> returning needed datasz %zu",
a43681
+			  needed, datasz, needed - datasz);
a43681
+		return needed - datasz;
a43681
+	}
a43681
+
a43681
+	ptr = data;
a43681
+
a43681
+	tmpssz = utf8_to_ucs2(ptr + 8, datasz - 8, true, var->name);
a43681
+	if (tmpssz < 0) {
a43681
+		efi_error("UTF-8 to UCS-2 conversion failed");
a43681
+		return -1;
a43681
+	}
a43681
+	tmpu32 = tmpssz;
a43681
+	tmpu32 *= sizeof(uint16_t);
a43681
+
a43681
+	debug("namesz:%"PRIu32" - tmpu32:%"PRIu32, namesz, tmpu32);
a43681
+	if (sub(namesz, tmpu32, &tmpu32))
a43681
+		goto overflow;
a43681
+	debug("tmpu32 -> %"PRIu32, tmpu32);
a43681
+
a43681
+	debug("namesz:%"PRIu32" - tmpu32:%"PRIu32, namesz, tmpu32);
a43681
+	if (sub(namesz, tmpu32, &namesz))
a43681
+		goto overflow;
a43681
+	debug("namesz -> %"PRIu32, namesz);
a43681
+
a43681
+	debug("needed:%"PRIu32" - tmpu32:%"PRIu32, needed, tmpu32);
a43681
+	if (sub(needed, tmpu32, &needed))
a43681
+		goto overflow;
a43681
+	debug("needed -> %"PRIu32, needed);
a43681
+
a43681
+	debug("datasz:%zu needed: %"PRIu32, datasz, needed);
a43681
+	if (datasz < needed) {
a43681
+		debug("needed: %"PRIu32" datasz: %zd -> returning needed datasz %"PRIu32,
a43681
+			  needed, datasz, needed);
a43681
+		return needed;
a43681
+	}
a43681
+
a43681
+	*(uint32_t *)ptr = namesz;
a43681
+	ptr += sizeof (uint32_t);
a43681
+
a43681
+	*(uint32_t *)ptr = var->data_size;
a43681
+	ptr += sizeof (uint32_t);
a43681
+
a43681
+	ptr += namesz;
a43681
+
a43681
+	memcpy(ptr, var->guid, sizeof (efi_guid_t));
a43681
+	ptr += sizeof(efi_guid_t);
a43681
+
a43681
+	*(uint32_t *)ptr = var->attrs;
a43681
+	ptr += sizeof (uint32_t);
a43681
+
a43681
+	memcpy(ptr, var->data, var->data_size);
a43681
+	ptr += var->data_size;
a43681
+
a43681
+	crc = efi_crc32(data, needed - sizeof(uint32_t));
a43681
+	debug("efi_crc32(%p, %lu) -> 0x%"PRIx32,
a43681
+	      data, needed - sizeof(uint32_t), crc);
a43681
+	*(uint32_t *)ptr = crc;
a43681
+
a43681
+	return needed;
a43681
+}
a43681
+
a43681
+ssize_t NONNULL(1) PUBLIC
a43681
+efi_variable_export(efi_variable_t *var, uint8_t *data, size_t datasz)
a43681
+{
a43681
+	uint32_t tmpu32;
a43681
+	ssize_t tmpssz;
a43681
+	uint32_t namesz;
a43681
+	uint32_t needed = sizeof (uint32_t)		/* magic */
a43681
+			+ sizeof (uint32_t)		/* version */
a43681
+			+ sizeof (uint64_t)		/* attr */
a43681
+			+ sizeof (efi_guid_t)		/* guid */
a43681
+			+ sizeof (uint32_t)		/* name_len */
a43681
+			+ sizeof (uint32_t)		/* data_len */
a43681
+			+ 2				/* name */
a43681
+			+ 1				/* data */
a43681
+			+ 4;				/* crc32 */
a43681
+	uint8_t *ptr;
a43681
+	uint32_t crc;
a43681
+
a43681
+	if (!var) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("var cannot be NULL");
a43681
+		return -1;
a43681
+	}
a43681
+	if (!var->name) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("var->name cannot be NULL");
a43681
+		return -1;
a43681
+	}
a43681
+	if (!var->data) {
a43681
+		errno = EINVAL;
a43681
+		efi_error("var->data cannot be NULL");
a43681
+		return -1;
a43681
+	}
a43681
+
a43681
+	debug("data: %p datasz: %zu", data, datasz);
a43681
+
a43681
+	namesz = utf8size(var->name, -1);
a43681
+	debug("sizeof(uint16_t):%zd * namesz:%"PRIu32, sizeof(uint16_t), namesz);
a43681
+	if (mul(sizeof (uint16_t), namesz, &namesz)) {
a43681
+overflow:
a43681
+		errno = EOVERFLOW;
a43681
+		efi_error("arithmetic overflow computing name size");
a43681
+		return -1;
a43681
+	}
a43681
+	debug("namesz -> %"PRIu32, namesz);
a43681
+
a43681
+	/*
a43681
+	 * Remove our stand-ins for name size and data size before we add
a43681
+	 * them back in.
a43681
+	 */
a43681
+	needed -= 3;
a43681
+
a43681
+	debug("needed:%"PRIu32" + namesz:%"PRIu32, needed, namesz);
a43681
+	if (add(needed, namesz, &needed))
a43681
+		goto overflow;
a43681
+	debug("needed -> %"PRIu32, needed);
a43681
+
a43681
+	debug("needed:%"PRIu32" + var->data_size:%zd", needed, var->data_size);
a43681
+	if (add(needed, var->data_size, &needed))
a43681
+		goto overflow;
a43681
+	debug("needed -> %"PRIu32, needed);
a43681
+
a43681
+	if (!data || datasz == 0) {
a43681
+		debug("data: %p datasz: %zd -> returning needed datasz %"PRIu32,
a43681
+		      data, datasz, needed);
a43681
+		return needed;
a43681
+	}
a43681
+
a43681
+	debug("datasz:%zu needed: %"PRIu32, datasz, needed);
a43681
+	if (datasz < needed) {
a43681
+		efi_error("needed: %"PRIu32" datasz: %zd -> returning needed datasz %zd",
a43681
+			  needed, datasz, needed - datasz);
a43681
+		return needed - datasz;
a43681
+	}
a43681
+
a43681
+	ptr = data;
a43681
 
a43681
 	*(uint32_t *)ptr = EFIVAR_MAGIC;
a43681
 	ptr += sizeof (uint32_t);
a43681
@@ -181,21 +569,51 @@ efi_variable_export(efi_variable_t *var, uint8_t *data, size_t size)
a43681
 	memcpy(ptr, var->guid, sizeof (efi_guid_t));
a43681
 	ptr += sizeof (efi_guid_t);
a43681
 
a43681
-	*(uint32_t *)ptr = (uint32_t)(sizeof (char16_t) * name_len);
a43681
+	tmpssz = utf8_to_ucs2(ptr + 8, datasz - 8, true, var->name);
a43681
+	if (tmpssz < 0) {
a43681
+		efi_error("UTF-8 to UCS-2 conversion failed");
a43681
+		return -1;
a43681
+	}
a43681
+	tmpu32 = tmpssz;
a43681
+	tmpu32 *= sizeof(uint16_t);
a43681
+
a43681
+	debug("namesz:%"PRIu32" - tmpu32:%"PRIu32, namesz, tmpu32);
a43681
+	if (sub(namesz, tmpu32, &tmpu32))
a43681
+		goto overflow;
a43681
+	debug("tmpu32 -> %"PRIu32, tmpu32);
a43681
+
a43681
+	debug("needed:%"PRIu32" - tmpu32:%"PRIu32, needed, tmpu32);
a43681
+	if (sub(needed, tmpu32, &needed))
a43681
+		goto overflow;
a43681
+	debug("needed -> %"PRIu32, needed);
a43681
+
a43681
+	debug("namesz:%"PRIu32" - tmpu32:%"PRIu32, namesz, tmpu32);
a43681
+	if (sub(namesz, tmpu32, &namesz))
a43681
+		goto overflow;
a43681
+	debug("namesz -> %"PRIu32, namesz);
a43681
+
a43681
+	debug("datasz:%zu needed: %"PRIu32, datasz, needed);
a43681
+	if (datasz < needed) {
a43681
+		efi_error("needed: %"PRIu32" datasz: %zd -> returning needed datasz %zd",
a43681
+			  needed, datasz, needed - datasz);
a43681
+		return needed - datasz;
a43681
+	}
a43681
+
a43681
+	*(uint32_t *)ptr = namesz;
a43681
 	ptr += sizeof (uint32_t);
a43681
 
a43681
 	*(uint32_t *)ptr = var->data_size;
a43681
 	ptr += sizeof (uint32_t);
a43681
 
a43681
-	for (uint32_t i = 0; i < name_len; i++) {
a43681
-		*(char16_t *)ptr = var->name[i];
a43681
-		ptr += sizeof (char16_t);
a43681
-	}
a43681
+	ptr += namesz;
a43681
 
a43681
 	memcpy(ptr, var->data, var->data_size);
a43681
 	ptr += var->data_size;
a43681
 
a43681
-	*(uint32_t *)ptr = EFIVAR_MAGIC;
a43681
+	crc = efi_crc32(data, needed - sizeof(uint32_t));
a43681
+	debug("efi_crc32(%p, %lu) -> 0x%"PRIx32,
a43681
+	      data, needed - sizeof(uint32_t), crc);
a43681
+	*(uint32_t *)ptr = crc;
a43681
 
a43681
 	return needed;
a43681
 }
a43681
diff --git a/src/gpt.c b/src/gpt.c
a43681
index aa4055b9812..8babafeb588 100644
a43681
--- a/src/gpt.c
a43681
+++ b/src/gpt.c
a43681
@@ -48,24 +48,6 @@ struct blkdev_ioctl_param {
a43681
 	char * block_contents;
a43681
 };
a43681
 
a43681
-/**
a43681
- * efi_crc32() - EFI version of crc32 function
a43681
- * @buf: buffer to calculate crc32 of
a43681
- * @len - length of buf
a43681
- *
a43681
- * Description: Returns EFI-style CRC32 value for @buf
a43681
- *
a43681
- * This function uses the little endian Ethernet polynomial
a43681
- * but seeds the function with ~0, and xor's with ~0 at the end.
a43681
- * Note, the EFI Specification, v1.02, has a reference to
a43681
- * Dr. Dobbs Journal, May 1994 (actually it's in May 1992).
a43681
- */
a43681
-static inline uint32_t
a43681
-efi_crc32(const void *buf, unsigned long len)
a43681
-{
a43681
-	return (crc32(buf, len, ~0L) ^ ~0L);
a43681
-}
a43681
-
a43681
 /**
a43681
  * is_pmbr_valid(): test Protective MBR for validity
a43681
  * @mbr: pointer to a legacy mbr structure
a43681
diff --git a/src/include/efivar/efivar.h b/src/include/efivar/efivar.h
a43681
index 729b6fe80f7..8ad14b9be57 100644
a43681
--- a/src/include/efivar/efivar.h
a43681
+++ b/src/include/efivar/efivar.h
a43681
@@ -139,6 +139,9 @@ extern ssize_t efi_variable_import(uint8_t *data, size_t size,
a43681
 extern ssize_t efi_variable_export(efi_variable_t *var, uint8_t *data,
a43681
 				size_t size)
a43681
 			__attribute__((__nonnull__ (1)));
a43681
+extern ssize_t efi_variable_export_dmpstore(efi_variable_t *var, uint8_t *data,
a43681
+				size_t size)
a43681
+			__attribute__((__nonnull__ (1)));
a43681
 
a43681
 extern efi_variable_t *efi_variable_alloc(void)
a43681
 			__attribute__((__visibility__ ("default")));
a43681
-- 
a43681
2.26.2
a43681