Skip to content

Commit b67775e

Browse files
committed
Merge tag 'efi-urgent-2020-03-15' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull EFI fixes from Thomas Gleixner: "Two EFI fixes: - Prevent a race and buffer overflow in the sysfs efivars interface which causes kernel memory corruption. - Add the missing NULL pointer checks in efivar_store_raw()" * tag 'efi-urgent-2020-03-15' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: efi: Add a sanity check to efivar_store_raw() efi: Fix a race and a buffer overflow while reading efivars via sysfs
2 parents de28a65 + d6c066f commit b67775e

File tree

1 file changed

+23
-9
lines changed

1 file changed

+23
-9
lines changed

drivers/firmware/efi/efivars.c

Lines changed: 23 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -83,13 +83,16 @@ static ssize_t
8383
efivar_attr_read(struct efivar_entry *entry, char *buf)
8484
{
8585
struct efi_variable *var = &entry->var;
86+
unsigned long size = sizeof(var->Data);
8687
char *str = buf;
88+
int ret;
8789

8890
if (!entry || !buf)
8991
return -EINVAL;
9092

91-
var->DataSize = 1024;
92-
if (efivar_entry_get(entry, &var->Attributes, &var->DataSize, var->Data))
93+
ret = efivar_entry_get(entry, &var->Attributes, &size, var->Data);
94+
var->DataSize = size;
95+
if (ret)
9396
return -EIO;
9497

9598
if (var->Attributes & EFI_VARIABLE_NON_VOLATILE)
@@ -116,13 +119,16 @@ static ssize_t
116119
efivar_size_read(struct efivar_entry *entry, char *buf)
117120
{
118121
struct efi_variable *var = &entry->var;
122+
unsigned long size = sizeof(var->Data);
119123
char *str = buf;
124+
int ret;
120125

121126
if (!entry || !buf)
122127
return -EINVAL;
123128

124-
var->DataSize = 1024;
125-
if (efivar_entry_get(entry, &var->Attributes, &var->DataSize, var->Data))
129+
ret = efivar_entry_get(entry, &var->Attributes, &size, var->Data);
130+
var->DataSize = size;
131+
if (ret)
126132
return -EIO;
127133

128134
str += sprintf(str, "0x%lx\n", var->DataSize);
@@ -133,12 +139,15 @@ static ssize_t
133139
efivar_data_read(struct efivar_entry *entry, char *buf)
134140
{
135141
struct efi_variable *var = &entry->var;
142+
unsigned long size = sizeof(var->Data);
143+
int ret;
136144

137145
if (!entry || !buf)
138146
return -EINVAL;
139147

140-
var->DataSize = 1024;
141-
if (efivar_entry_get(entry, &var->Attributes, &var->DataSize, var->Data))
148+
ret = efivar_entry_get(entry, &var->Attributes, &size, var->Data);
149+
var->DataSize = size;
150+
if (ret)
142151
return -EIO;
143152

144153
memcpy(buf, var->Data, var->DataSize);
@@ -199,6 +208,9 @@ efivar_store_raw(struct efivar_entry *entry, const char *buf, size_t count)
199208
u8 *data;
200209
int err;
201210

211+
if (!entry || !buf)
212+
return -EINVAL;
213+
202214
if (in_compat_syscall()) {
203215
struct compat_efi_variable *compat;
204216

@@ -250,14 +262,16 @@ efivar_show_raw(struct efivar_entry *entry, char *buf)
250262
{
251263
struct efi_variable *var = &entry->var;
252264
struct compat_efi_variable *compat;
265+
unsigned long datasize = sizeof(var->Data);
253266
size_t size;
267+
int ret;
254268

255269
if (!entry || !buf)
256270
return 0;
257271

258-
var->DataSize = 1024;
259-
if (efivar_entry_get(entry, &entry->var.Attributes,
260-
&entry->var.DataSize, entry->var.Data))
272+
ret = efivar_entry_get(entry, &var->Attributes, &datasize, var->Data);
273+
var->DataSize = datasize;
274+
if (ret)
261275
return -EIO;
262276

263277
if (in_compat_syscall()) {

0 commit comments

Comments
 (0)