d2399a
From 2fefae47716d501aec41c1102f3fd4531f070b05 Mon Sep 17 00:00:00 2001
d2399a
From: Remi Collet <remi@php.net>
d2399a
Date: Tue, 19 Aug 2014 08:33:49 +0200
d2399a
Subject: [PATCH] Fixed Sec Bug #67717 segfault in dns_get_record CVE-2014-3597
d2399a
d2399a
Incomplete fix for CVE-2014-4049
d2399a
d2399a
Check possible buffer overflow
d2399a
- pass real buffer end to dn_expand calls
d2399a
- check buffer len before each read
d2399a
---
d2399a
 ext/standard/dns.c | 84 ++++++++++++++++++++++++++++++++++++++----------------
d2399a
 1 file changed, 60 insertions(+), 24 deletions(-)
d2399a
d2399a
diff --git a/ext/standard/dns.c b/ext/standard/dns.c
d2399a
index 214a7dc..0b5e69c 100644
d2399a
--- a/ext/standard/dns.c
d2399a
+++ b/ext/standard/dns.c
d2399a
@@ -412,8 +412,14 @@ PHP_FUNCTION(dns_check_record)
d2399a
 
d2399a
 #if HAVE_FULL_DNS_FUNCS
d2399a
 
d2399a
+#define CHECKCP(n) do { \
d2399a
+	if (cp + n > end) { \
d2399a
+		return NULL; \
d2399a
+	} \
d2399a
+} while (0)
d2399a
+
d2399a
 /* {{{ php_parserr */
d2399a
-static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int store, int raw, zval **subarray)
d2399a
+static u_char *php_parserr(u_char *cp, u_char *end, querybuf *answer, int type_to_fetch, int store, int raw, zval **subarray)
d2399a
 {
d2399a
 	u_short type, class, dlen;
d2399a
 	u_long ttl;
d2399a
@@ -425,16 +431,18 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 
d2399a
 	*subarray = NULL;
d2399a
 
d2399a
-	n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, sizeof(name) - 2);
d2399a
+	n = dn_expand(answer->qb2, end, cp, name, sizeof(name) - 2);
d2399a
 	if (n < 0) {
d2399a
 		return NULL;
d2399a
 	}
d2399a
 	cp += n;
d2399a
 
d2399a
+	CHECKCP(10);
d2399a
 	GETSHORT(type, cp);
d2399a
 	GETSHORT(class, cp);
d2399a
 	GETLONG(ttl, cp);
d2399a
 	GETSHORT(dlen, cp);
d2399a
+	CHECKCP(dlen);
d2399a
 	if (type_to_fetch != T_ANY && type != type_to_fetch) {
d2399a
 		cp += dlen;
d2399a
 		return cp;
d2399a
@@ -461,12 +469,14 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 
d2399a
 	switch (type) {
d2399a
 		case DNS_T_A:
d2399a
+			CHECKCP(4);
d2399a
 			add_assoc_string(*subarray, "type", "A", 1);
d2399a
 			snprintf(name, sizeof(name), "%d.%d.%d.%d", cp[0], cp[1], cp[2], cp[3]);
d2399a
 			add_assoc_string(*subarray, "ip", name, 1);
d2399a
 			cp += dlen;
d2399a
 			break;
d2399a
 		case DNS_T_MX:
d2399a
+			CHECKCP(2);
d2399a
 			add_assoc_string(*subarray, "type", "MX", 1);
d2399a
 			GETSHORT(n, cp);
d2399a
 			add_assoc_long(*subarray, "pri", n);
d2399a
@@ -485,7 +495,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 			if (type == DNS_T_PTR) {
d2399a
 				add_assoc_string(*subarray, "type", "PTR", 1);
d2399a
 			}
d2399a
-			n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, (sizeof name) - 2);
d2399a
+			n = dn_expand(answer->qb2, end, cp, name, (sizeof name) - 2);
d2399a
 			if (n < 0) {
d2399a
 				return NULL;
d2399a
 			}
d2399a
@@ -495,18 +505,22 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 		case DNS_T_HINFO:
d2399a
 			/* See RFC 1010 for values */
d2399a
 			add_assoc_string(*subarray, "type", "HINFO", 1);
d2399a
+			CHECKCP(1);
d2399a
 			n = *cp & 0xFF;
d2399a
 			cp++;
d2399a
+			CHECKCP(n);
d2399a
 			add_assoc_stringl(*subarray, "cpu", (char*)cp, n, 1);
d2399a
 			cp += n;
d2399a
+			CHECKCP(1);
d2399a
 			n = *cp & 0xFF;
d2399a
 			cp++;
d2399a
+			CHECKCP(n);
d2399a
 			add_assoc_stringl(*subarray, "os", (char*)cp, n, 1);
d2399a
 			cp += n;
d2399a
 			break;
d2399a
 		case DNS_T_TXT:
d2399a
 			{
d2399a
-				int ll = 0;
d2399a
+				int l1 = 0, l2 = 0;
d2399a
 				zval *entries = NULL;
d2399a
 
d2399a
 				add_assoc_string(*subarray, "type", "TXT", 1);
d2399a
@@ -515,37 +529,41 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 				MAKE_STD_ZVAL(entries);
d2399a
 				array_init(entries);
d2399a
 				
d2399a
-				while (ll < dlen) {
d2399a
-					n = cp[ll];
d2399a
-					if ((ll + n) >= dlen) {
d2399a
+				while (l1 < dlen) {
d2399a
+					n = cp[l1];
d2399a
+					if ((l1 + n) >= dlen) {
d2399a
 						// Invalid chunk length, truncate
d2399a
-						n = dlen - (ll + 1);
d2399a
+						n = dlen - (l1 + 1);
d2399a
+					}
d2399a
+					if (n) {
d2399a
+						memcpy(tp + l2 , cp + l1 + 1, n);
d2399a
+						add_next_index_stringl(entries, cp + l1 + 1, n, 1);
d2399a
 					}
d2399a
-					memcpy(tp + ll , cp + ll + 1, n);
d2399a
-					add_next_index_stringl(entries, cp + ll + 1, n, 1);
d2399a
-					ll = ll + n + 1;
d2399a
+					l1 = l1 + n + 1;
d2399a
+					l2 = l2 + n;
d2399a
 				}
d2399a
-				tp[dlen] = '\0';
d2399a
+				tp[l2] = '\0';
d2399a
 				cp += dlen;
d2399a
 
d2399a
-				add_assoc_stringl(*subarray, "txt", tp, (dlen>0)?dlen - 1:0, 0);
d2399a
+				add_assoc_stringl(*subarray, "txt", tp, l2, 0);
d2399a
 				add_assoc_zval(*subarray, "entries", entries);
d2399a
 			}
d2399a
 			break;
d2399a
 		case DNS_T_SOA:
d2399a
 			add_assoc_string(*subarray, "type", "SOA", 1);
d2399a
-			n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, (sizeof name) -2);
d2399a
+			n = dn_expand(answer->qb2, end, cp, name, (sizeof name) -2);
d2399a
 			if (n < 0) {
d2399a
 				return NULL;
d2399a
 			}
d2399a
 			cp += n;
d2399a
 			add_assoc_string(*subarray, "mname", name, 1);
d2399a
-			n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, (sizeof name) -2);
d2399a
+			n = dn_expand(answer->qb2, end, cp, name, (sizeof name) -2);
d2399a
 			if (n < 0) {
d2399a
 				return NULL;
d2399a
 			}
d2399a
 			cp += n;
d2399a
 			add_assoc_string(*subarray, "rname", name, 1);
d2399a
+			CHECKCP(5*4);
d2399a
 			GETLONG(n, cp);
d2399a
 			add_assoc_long(*subarray, "serial", n);
d2399a
 			GETLONG(n, cp);
d2399a
@@ -559,6 +577,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 			break;
d2399a
 		case DNS_T_AAAA:
d2399a
 			tp = (u_char*)name;
d2399a
+			CHECKCP(8*2);
d2399a
 			for(i=0; i < 8; i++) {
d2399a
 				GETSHORT(s, cp);
d2399a
 				if (s != 0) {
d2399a
@@ -593,6 +612,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 		case DNS_T_A6:
d2399a
 			p = cp;
d2399a
 			add_assoc_string(*subarray, "type", "A6", 1);
d2399a
+			CHECKCP(1);
d2399a
 			n = ((int)cp[0]) & 0xFF;
d2399a
 			cp++;
d2399a
 			add_assoc_long(*subarray, "masklen", n);
d2399a
@@ -628,6 +648,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 				cp++;
d2399a
 			}
d2399a
 			for (i = (n + 8) / 16; i < 8; i++) {
d2399a
+				CHECKCP(2);
d2399a
 				GETSHORT(s, cp);
d2399a
 				if (s != 0) {
d2399a
 					if (tp > (u_char *)name) {
d2399a
@@ -657,7 +678,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 			tp[0] = '\0';
d2399a
 			add_assoc_string(*subarray, "ipv6", name, 1);
d2399a
 			if (cp < p + dlen) {
d2399a
-				n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, (sizeof name) - 2);
d2399a
+				n = dn_expand(answer->qb2, end, cp, name, (sizeof name) - 2);
d2399a
 				if (n < 0) {
d2399a
 					return NULL;
d2399a
 				}
d2399a
@@ -666,6 +687,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 			}
d2399a
 			break;
d2399a
 		case DNS_T_SRV:
d2399a
+			CHECKCP(3*2);
d2399a
 			add_assoc_string(*subarray, "type", "SRV", 1);
d2399a
 			GETSHORT(n, cp);
d2399a
 			add_assoc_long(*subarray, "pri", n);
d2399a
@@ -673,7 +695,7 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 			add_assoc_long(*subarray, "weight", n);
d2399a
 			GETSHORT(n, cp);
d2399a
 			add_assoc_long(*subarray, "port", n);
d2399a
-			n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, (sizeof name) - 2);
d2399a
+			n = dn_expand(answer->qb2, end, cp, name, (sizeof name) - 2);
d2399a
 			if (n < 0) {
d2399a
 				return NULL;
d2399a
 			}
d2399a
@@ -681,21 +703,35 @@ static u_char *php_parserr(u_char *cp, querybuf *answer, int type_to_fetch, int
d2399a
 			add_assoc_string(*subarray, "target", name, 1);
d2399a
 			break;
d2399a
 		case DNS_T_NAPTR:
d2399a
+			CHECKCP(2*2);
d2399a
 			add_assoc_string(*subarray, "type", "NAPTR", 1);
d2399a
 			GETSHORT(n, cp);
d2399a
 			add_assoc_long(*subarray, "order", n);
d2399a
 			GETSHORT(n, cp);
d2399a
 			add_assoc_long(*subarray, "pref", n);
d2399a
+
d2399a
+			CHECKCP(1);
d2399a
 			n = (cp[0] & 0xFF);
d2399a
-			add_assoc_stringl(*subarray, "flags", (char*)++cp, n, 1);
d2399a
+			cp++;
d2399a
+			CHECKCP(n);
d2399a
+			add_assoc_stringl(*subarray, "flags", (char*)cp, n, 1);
d2399a
 			cp += n;
d2399a
+
d2399a
+			CHECKCP(1);
d2399a
 			n = (cp[0] & 0xFF);
d2399a
-			add_assoc_stringl(*subarray, "services", (char*)++cp, n, 1);
d2399a
+			cp++;
d2399a
+			CHECKCP(n);
d2399a
+			add_assoc_stringl(*subarray, "services", (char*)cp, n, 1);
d2399a
 			cp += n;
d2399a
+
d2399a
+			CHECKCP(1);
d2399a
 			n = (cp[0] & 0xFF);
d2399a
-			add_assoc_stringl(*subarray, "regex", (char*)++cp, n, 1);
d2399a
+			cp++;
d2399a
+			CHECKCP(n);
d2399a
+			add_assoc_stringl(*subarray, "regex", (char*)cp, n, 1);
d2399a
 			cp += n;
d2399a
-			n = dn_expand(answer->qb2, answer->qb2+65536, cp, name, (sizeof name) - 2);
d2399a
+
d2399a
+			n = dn_expand(answer->qb2, end, cp, name, (sizeof name) - 2);
d2399a
 			if (n < 0) {
d2399a
 				return NULL;
d2399a
 			}
d2399a
@@ -888,7 +924,7 @@ PHP_FUNCTION(dns_get_record)
d2399a
 			while (an-- && cp && cp < end) {
d2399a
 				zval *retval;
d2399a
 
d2399a
-				cp = php_parserr(cp, &answer, type_to_fetch, store_results, raw, &retval);
d2399a
+				cp = php_parserr(cp, end, &answer, type_to_fetch, store_results, raw, &retval);
d2399a
 				if (retval != NULL && store_results) {
d2399a
 					add_next_index_zval(return_value, retval);
d2399a
 				}
d2399a
@@ -901,7 +937,7 @@ PHP_FUNCTION(dns_get_record)
d2399a
 				while (ns-- > 0 && cp && cp < end) {
d2399a
 					zval *retval = NULL;
d2399a
 
d2399a
-					cp = php_parserr(cp, &answer, DNS_T_ANY, authns != NULL, raw, &retval);
d2399a
+					cp = php_parserr(cp, end, &answer, DNS_T_ANY, authns != NULL, raw, &retval);
d2399a
 					if (retval != NULL) {
d2399a
 						add_next_index_zval(authns, retval);
d2399a
 					}
d2399a
@@ -913,7 +949,7 @@ PHP_FUNCTION(dns_get_record)
d2399a
 				while (ar-- > 0 && cp && cp < end) {
d2399a
 					zval *retval = NULL;
d2399a
 
d2399a
-					cp = php_parserr(cp, &answer, DNS_T_ANY, 1, raw, &retval);
d2399a
+					cp = php_parserr(cp, end, &answer, DNS_T_ANY, 1, raw, &retval);
d2399a
 					if (retval != NULL) {
d2399a
 						add_next_index_zval(addtl, retval);
d2399a
 					}
d2399a
-- 
d2399a
1.9.2
d2399a