d2399a
From a33759fd275b32ed0bbe89796fe2953b3cb0b41f Mon Sep 17 00:00:00 2001
d2399a
From: Remi Collet <remi@php.net>
d2399a
Date: Tue, 4 Mar 2014 20:32:52 +0100
d2399a
Subject: [PATCH] Fixed Bug #66820 out-of-bounds memory access in fileinfo
d2399a
d2399a
Upstream fix:
d2399a
https://github.com/glensc/file/commit/447558595a3650db2886cd2f416ad0beba965801
d2399a
d2399a
Notice, test changed, with upstream agreement:
d2399a
-define OFFSET_OOB(n, o, i)	((n) < (o) || (i) >= ((n) - (o)))
d2399a
+define OFFSET_OOB(n, o, i)	((n) < (o) || (i) >  ((n) - (o)))
d2399a
---
d2399a
 ext/fileinfo/libmagic/softmagic.c | 34 ++++++++++++++++++----------------
d2399a
 1 file changed, 18 insertions(+), 16 deletions(-)
d2399a
d2399a
diff --git a/ext/fileinfo/libmagic/softmagic.c b/ext/fileinfo/libmagic/softmagic.c
d2399a
index 82a470a..21fea6b 100644
d2399a
--- a/ext/fileinfo/libmagic/softmagic.c
d2399a
+++ b/ext/fileinfo/libmagic/softmagic.c
d2399a
@@ -67,6 +67,8 @@ private void cvt_16(union VALUETYPE *, const struct magic *);
d2399a
 private void cvt_32(union VALUETYPE *, const struct magic *);
d2399a
 private void cvt_64(union VALUETYPE *, const struct magic *);
d2399a
 
d2399a
+#define OFFSET_OOB(n, o, i)	((n) < (o) || (i) > ((n) - (o)))
d2399a
+
d2399a
 /*
d2399a
  * softmagic - lookup one file in parsed, in-memory copy of database
d2399a
  * Passed the name and FILE * of one file to be typed.
d2399a
@@ -1171,7 +1173,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 		}
d2399a
 		switch (cvt_flip(m->in_type, flip)) {
d2399a
 		case FILE_BYTE:
d2399a
-			if (nbytes < (offset + 1))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 1))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1206,7 +1208,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 				offset = ~offset;
d2399a
 			break;
d2399a
 		case FILE_BESHORT:
d2399a
-			if (nbytes < (offset + 2))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 2))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1258,7 +1260,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 				offset = ~offset;
d2399a
 			break;
d2399a
 		case FILE_LESHORT:
d2399a
-			if (nbytes < (offset + 2))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 2))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1310,7 +1312,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 				offset = ~offset;
d2399a
 			break;
d2399a
 		case FILE_SHORT:
d2399a
-			if (nbytes < (offset + 2))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 2))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1347,7 +1349,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 			break;
d2399a
 		case FILE_BELONG:
d2399a
 		case FILE_BEID3:
d2399a
-			if (nbytes < (offset + 4))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 4))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1418,7 +1420,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 			break;
d2399a
 		case FILE_LELONG:
d2399a
 		case FILE_LEID3:
d2399a
-			if (nbytes < (offset + 4))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 4))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1488,7 +1490,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 				offset = ~offset;
d2399a
 			break;
d2399a
 		case FILE_MELONG:
d2399a
-			if (nbytes < (offset + 4))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 4))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1558,7 +1560,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 				offset = ~offset;
d2399a
 			break;
d2399a
 		case FILE_LONG:
d2399a
-			if (nbytes < (offset + 4))
d2399a
+			if (OFFSET_OOB(nbytes, offset, 4))
d2399a
 				return 0;
d2399a
 			if (off) {
d2399a
 				switch (m->in_op & FILE_OPS_MASK) {
d2399a
@@ -1630,14 +1632,14 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 	/* Verify we have enough data to match magic type */
d2399a
 	switch (m->type) {
d2399a
 	case FILE_BYTE:
d2399a
-		if (nbytes < (offset + 1)) /* should alway be true */
d2399a
+		if (OFFSET_OOB(nbytes, offset, 1))
d2399a
 			return 0;
d2399a
 		break;
d2399a
 
d2399a
 	case FILE_SHORT:
d2399a
 	case FILE_BESHORT:
d2399a
 	case FILE_LESHORT:
d2399a
-		if (nbytes < (offset + 2))
d2399a
+		if (OFFSET_OOB(nbytes, offset, 2))
d2399a
 			return 0;
d2399a
 		break;
d2399a
 
d2399a
@@ -1656,33 +1658,33 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 	case FILE_FLOAT:
d2399a
 	case FILE_BEFLOAT:
d2399a
 	case FILE_LEFLOAT:
d2399a
-		if (nbytes < (offset + 4))
d2399a
+		if (OFFSET_OOB(nbytes, offset, 4))
d2399a
 			return 0;
d2399a
 		break;
d2399a
 
d2399a
 	case FILE_DOUBLE:
d2399a
 	case FILE_BEDOUBLE:
d2399a
 	case FILE_LEDOUBLE:
d2399a
-		if (nbytes < (offset + 8))
d2399a
+		if (OFFSET_OOB(nbytes, offset, 8))
d2399a
 			return 0;
d2399a
 		break;
d2399a
 
d2399a
 	case FILE_STRING:
d2399a
 	case FILE_PSTRING:
d2399a
 	case FILE_SEARCH:
d2399a
-		if (nbytes < (offset + m->vallen))
d2399a
+		if (OFFSET_OOB(nbytes, offset, m->vallen))
d2399a
 			return 0;
d2399a
 		break;
d2399a
 
d2399a
 	case FILE_REGEX:
d2399a
-		if (nbytes < offset)
d2399a
+		if (OFFSET_OOB(nbytes, offset, 0))
d2399a
 			return 0;
d2399a
 		break;
d2399a
 
d2399a
 	case FILE_INDIRECT:
d2399a
 		if (offset == 0)
d2399a
 			return 0;
d2399a
-		if (nbytes < offset)
d2399a
+		if (OFFSET_OOB(nbytes, offset, 0))
d2399a
 			return 0;
d2399a
 		sbuf = ms->o.buf;
d2399a
 		soffset = ms->offset;
d2399a
@@ -1716,7 +1718,7 @@ mget(struct magic_set *ms, const unsigned char *s, struct magic *m,
d2399a
 		return rv;
d2399a
 
d2399a
 	case FILE_USE:
d2399a
-		if (nbytes < offset)
d2399a
+		if (OFFSET_OOB(nbytes, offset, 0))
d2399a
 			return 0;
d2399a
 		sbuf = m->value.s;
d2399a
 		if (*sbuf == '^') {
d2399a
-- 
d2399a
1.8.4.3
d2399a