fd304b
From c081fa410e7c466df4b3b257e7b974b71fb7f250 Mon Sep 17 00:00:00 2001
fd304b
From: Jean Delvare <jdelvare@suse.de>
fd304b
Date: Wed, 14 Oct 2015 14:37:04 +0200
fd304b
Subject: [PATCH 2/9] Avoid SIGBUS on mmap failure
fd304b
fd304b
mmap will fail with SIGBUS if trying to map a non-existent portion of
fd304b
a file. While this should never happen with /dev/mem, it can happen if
fd304b
passing a regular file with option -d. While people should no longer
fd304b
do that, failure gracefully seems better than crashing. So check for
fd304b
the file size before calling mmap.
fd304b
fd304b
This closes bug #46066:
fd304b
http://savannah.nongnu.org/bugs/?46066
fd304b
---
fd304b
 CHANGELOG |  6 ++++++
fd304b
 util.c    | 21 +++++++++++++++++++++
fd304b
 2 files changed, 27 insertions(+)
fd304b
fd304b
diff --git a/CHANGELOG b/CHANGELOG
fd304b
index 42d815c..aa1c28f 100644
fd304b
--- a/CHANGELOG
fd304b
+++ b/CHANGELOG
fd304b
@@ -1,3 +1,9 @@
fd304b
+2015-10-14  Jean Delvare  <jdelvare@suse.de>
fd304b
+
fd304b
+	* util.c: Avoid SIGBUS on mmap failure.
fd304b
+	  This fixes Savannah bug #46066:
fd304b
+	  https://savannah.nongnu.org/bugs/?46066
fd304b
+
fd304b
 2015-10-01  Roy Franz  <roy.franz@linaro.org>
fd304b
 
fd304b
 	* dmiopt.c: Add "--no-sysfs" option description to -h output.
fd304b
diff --git a/util.c b/util.c
fd304b
index 8cafe5c..5795d02 100644
fd304b
--- a/util.c
fd304b
+++ b/util.c
fd304b
@@ -152,6 +152,7 @@ void *mem_chunk(off_t base, size_t len, const char *devmem)
fd304b
 	void *p;
fd304b
 	int fd;
fd304b
 #ifdef USE_MMAP
fd304b
+	struct stat statbuf;
fd304b
 	off_t mmoffset;
fd304b
 	void *mmp;
fd304b
 #endif
fd304b
@@ -169,6 +170,26 @@ void *mem_chunk(off_t base, size_t len, const char *devmem)
fd304b
 	}
fd304b
 
fd304b
 #ifdef USE_MMAP
fd304b
+	if (fstat(fd, &statbuf) == -1)
fd304b
+	{
fd304b
+		fprintf(stderr, "%s: ", devmem);
fd304b
+		perror("stat");
fd304b
+		free(p);
fd304b
+		return NULL;
fd304b
+	}
fd304b
+
fd304b
+	/*
fd304b
+	 * mmap() will fail with SIGBUS if trying to map beyond the end of
fd304b
+	 * the file.
fd304b
+	 */
fd304b
+	if (S_ISREG(statbuf.st_mode) && base + (off_t)len > statbuf.st_size)
fd304b
+	{
fd304b
+		fprintf(stderr, "mmap: Can't map beyond end of file %s\n",
fd304b
+			devmem);
fd304b
+		free(p);
fd304b
+		return NULL;
fd304b
+	}
fd304b
+
fd304b
 #ifdef _SC_PAGESIZE
fd304b
 	mmoffset = base % sysconf(_SC_PAGESIZE);
fd304b
 #else
fd304b
-- 
fd304b
2.5.0
fd304b