1ff636
From 210ec6353d3cab2029e1eb160671fea918c97814 Mon Sep 17 00:00:00 2001
1ff636
From: "Jason S. McMullan" <jason.mcmullan@gmail.com>
1ff636
Date: Fri, 22 May 2015 20:30:01 +0200
1ff636
Subject: [PATCH] udev/net_id: Only read the first 64 bytes of PCI config space
1ff636
1ff636
The original code used fread(), which on some libc implementions
1ff636
(ie glibc 2.17) would pre-read a full 4K (PAGE_SIZE) of the
1ff636
PCI config space, when only 64 bytes were requested.
1ff636
1ff636
I have recently come across PCIe hardware which responds with
1ff636
Completion Timeouts when accesses above 256 bytes are attempted.
1ff636
1ff636
This can cause server systems with GHES/AEPI support to cause
1ff636
and immediate kernel panic due to the failed PCI transaction.
1ff636
1ff636
This change replaces the buffered fread() with an explict
1ff636
unbuffered read() of 64 bytes, which corrects this issue by
1ff636
only reading the guaranteed first 64 bytes of PCIe config space.
1ff636
1ff636
(cherry picked from commit 0454229c100a2113ba82df55703436d6cb2c492b)
1ff636
1ff636
Cherry-picked from: 0454229
1ff636
Resolves: #1222517
1ff636
---
1ff636
 src/udev/udev-builtin-net_id.c | 9 +++++----
1ff636
 1 file changed, 5 insertions(+), 4 deletions(-)
1ff636
1ff636
diff --git a/src/udev/udev-builtin-net_id.c b/src/udev/udev-builtin-net_id.c
181b3f
index 66474f772..dd2886caf 100644
1ff636
--- a/src/udev/udev-builtin-net_id.c
1ff636
+++ b/src/udev/udev-builtin-net_id.c
1ff636
@@ -91,6 +91,7 @@
1ff636
 #include <stdlib.h>
1ff636
 #include <stdarg.h>
1ff636
 #include <unistd.h>
1ff636
+#include <fcntl.h>
1ff636
 #include <string.h>
1ff636
 #include <errno.h>
1ff636
 #include <net/if.h>
1ff636
@@ -166,15 +167,15 @@ static int dev_pci_onboard(struct udev_device *dev, struct netnames *names) {
1ff636
 
1ff636
 /* read the 256 bytes PCI configuration space to check the multi-function bit */
1ff636
 static bool is_pci_multifunction(struct udev_device *dev) {
1ff636
-        _cleanup_fclose_ FILE *f = NULL;
1ff636
+        _cleanup_close_ int fd = -1;
1ff636
         const char *filename;
1ff636
         uint8_t config[64];
1ff636
 
1ff636
         filename = strjoina(udev_device_get_syspath(dev), "/config");
1ff636
-        f = fopen(filename, "re");
1ff636
-        if (!f)
1ff636
+        fd = open(filename, O_RDONLY | O_CLOEXEC);
1ff636
+        if (fd < 0)
1ff636
                 return false;
1ff636
-        if (fread(&config, sizeof(config), 1, f) != 1)
1ff636
+        if (read(fd, &config, sizeof(config)) != sizeof(config))
1ff636
                 return false;
1ff636
 
1ff636
         /* bit 0-6 header type, bit 7 multi/single function device */