-
Notifications
You must be signed in to change notification settings - Fork 521
/
Copy path0005-gpt-consolidate-crc32-computation-code.patch
116 lines (101 loc) · 3.69 KB
/
0005-gpt-consolidate-crc32-computation-code.patch
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
From 39351efcb1aab82bb86052d21f07308429414a83 Mon Sep 17 00:00:00 2001
From: Michael Marineau <michael.marineau@coreos.com>
Date: Sat, 18 Oct 2014 18:18:17 -0700
Subject: [PATCH] gpt: consolidate crc32 computation code
The gcrypt API is overly verbose, wrap it up in a helper function to
keep this rather common operation easy to use.
---
grub-core/lib/gpt.c | 43 ++++++++++++++++++++++++-------------------
1 file changed, 24 insertions(+), 19 deletions(-)
diff --git a/grub-core/lib/gpt.c b/grub-core/lib/gpt.c
index 01df7f3e8..43a150942 100644
--- a/grub-core/lib/gpt.c
+++ b/grub-core/lib/gpt.c
@@ -32,22 +32,17 @@ static grub_uint8_t grub_gpt_magic[] = GRUB_GPT_HEADER_MAGIC;
static grub_err_t
-grub_gpt_header_crc32 (struct grub_gpt_header *gpt, grub_uint32_t *crc)
+grub_gpt_lecrc32 (void *data, grub_size_t len, grub_uint32_t *crc)
{
grub_uint8_t *crc32_context;
- grub_uint32_t old;
crc32_context = grub_zalloc (GRUB_MD_CRC32->contextsize);
if (!crc32_context)
return grub_errno;
- /* crc32 must be computed with the field cleared. */
- old = gpt->crc32;
- gpt->crc32 = 0;
GRUB_MD_CRC32->init (crc32_context);
- GRUB_MD_CRC32->write (crc32_context, gpt, sizeof (*gpt));
+ GRUB_MD_CRC32->write (crc32_context, data, len);
GRUB_MD_CRC32->final (crc32_context);
- gpt->crc32 = old;
/* GRUB_MD_CRC32 always uses big endian, gpt is always little. */
*crc = grub_swap_bytes32 (*(grub_uint32_t *)
@@ -58,6 +53,25 @@ grub_gpt_header_crc32 (struct grub_gpt_header *gpt, grub_uint32_t *crc)
return GRUB_ERR_NONE;
}
+static grub_err_t
+grub_gpt_header_lecrc32 (struct grub_gpt_header *header, grub_uint32_t *crc)
+{
+ grub_uint32_t old, new;
+ grub_err_t err;
+
+ /* crc32 must be computed with the field cleared. */
+ old = header->crc32;
+ header->crc32 = 0;
+ err = grub_gpt_lecrc32 (header, sizeof (*header), &new);
+ header->crc32 = old;
+
+ if (err)
+ return err;
+
+ *crc = new;
+ return GRUB_ERR_NONE;
+}
+
/* Make sure the MBR is a protective MBR and not a normal MBR. */
grub_err_t
grub_gpt_pmbr_check (struct grub_msdos_partition_mbr *mbr)
@@ -87,7 +101,7 @@ grub_gpt_header_check (struct grub_gpt_header *gpt,
if (gpt->version != GRUB_GPT_HEADER_VERSION)
return grub_error (GRUB_ERR_BAD_PART_TABLE, "unknown GPT version");
- if (grub_gpt_header_crc32 (gpt, &crc))
+ if (grub_gpt_header_lecrc32 (gpt, &crc))
return grub_errno;
if (gpt->crc32 != crc)
@@ -158,7 +172,6 @@ grub_gpt_read_entries (grub_disk_t disk, grub_gpt_t gpt,
struct grub_gpt_header *header)
{
struct grub_gpt_partentry *entries = NULL;
- grub_uint8_t *crc32_context = NULL;
grub_uint32_t count, size, crc;
grub_disk_addr_t addr;
grub_size_t entries_size;
@@ -185,22 +198,15 @@ grub_gpt_read_entries (grub_disk_t disk, grub_gpt_t gpt,
if (grub_disk_read (disk, addr, 0, entries_size, entries))
goto fail;
- crc32_context = grub_zalloc (GRUB_MD_CRC32->contextsize);
- if (!crc32_context)
+ if (grub_gpt_lecrc32 (entries, entries_size, &crc))
goto fail;
- GRUB_MD_CRC32->init (crc32_context);
- GRUB_MD_CRC32->write (crc32_context, entries, entries_size);
- GRUB_MD_CRC32->final (crc32_context);
-
- crc = *(grub_uint32_t *) GRUB_MD_CRC32->read (crc32_context);
- if (grub_swap_bytes32 (crc) != header->partentry_crc32)
+ if (crc != header->partentry_crc32)
{
grub_error (GRUB_ERR_BAD_PART_TABLE, "invalid GPT entry crc32");
goto fail;
}
- grub_free (crc32_context);
grub_free (gpt->entries);
gpt->entries = entries;
gpt->entries_size = entries_size;
@@ -208,7 +214,6 @@ grub_gpt_read_entries (grub_disk_t disk, grub_gpt_t gpt,
fail:
grub_free (entries);
- grub_free (crc32_context);
return grub_errno;
}