summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGustavo André dos Santos Lopes <cataphract@php.net>2012-02-05 14:57:57 +0000
committerGustavo André dos Santos Lopes <cataphract@php.net>2012-03-19 16:36:21 +0000
commitcfdd6c5788afc6fb907f6f518dceab4fd82c922e (patch)
tree360bb8a46905aed3ae576f6bca93854309c49dca
parent9a460497da3cc2b755f4628350756427fc0a1051 (diff)
downloadphp-git-cfdd6c5788afc6fb907f6f518dceab4fd82c922e.tar.gz
MFH: 7dcada1 for 5.4
- Fixed possible unsigned int wrap around in html.c. Note that 5.3 has the same (potential) problem; even though the code is substantially different, the variable name and the fashion it was incremented was kept.
-rw-r--r--ext/standard/html.c18
1 files changed, 13 insertions, 5 deletions
diff --git a/ext/standard/html.c b/ext/standard/html.c
index 5b47a83788..65e63f41cc 100644
--- a/ext/standard/html.c
+++ b/ext/standard/html.c
@@ -1258,9 +1258,13 @@ PHPAPI char *php_escape_html_entities_ex(unsigned char *old, size_t oldlen, size
maxlen = 128;
} else {
maxlen = 2 * oldlen;
+ if (maxlen < oldlen) {
+ zend_error_noreturn(E_ERROR, "Input string is too long");
+ return NULL;
+ }
}
- replaced = emalloc(maxlen + 1);
+ replaced = emalloc(maxlen + 1); /* adding 1 is safe: maxlen is even */
len = 0;
cursor = 0;
while (cursor < oldlen) {
@@ -1272,8 +1276,9 @@ PHPAPI char *php_escape_html_entities_ex(unsigned char *old, size_t oldlen, size
/* guarantee we have at least 40 bytes to write.
* In HTML5, entities may take up to 33 bytes */
- if (len + 40 > maxlen) {
- replaced = erealloc(replaced, (maxlen += 128) + 1);
+ if (len > maxlen - 40) { /* maxlen can never be smaller than 128 */
+ replaced = safe_erealloc(replaced, maxlen , 1, 128 + 1);
+ maxlen += 128;
}
if (status == FAILURE) {
@@ -1402,8 +1407,11 @@ encode_amp:
}
/* checks passed; copy entity to result */
/* entity size is unbounded, we may need more memory */
- if (maxlen < len + ent_len + 2 /* & and ; */) {
- replaced = erealloc(replaced, (maxlen += ent_len + 128) + 1);
+ /* at this point maxlen - len >= 40 */
+ if (maxlen - len < ent_len + 2 /* & and ; */) {
+ /* ent_len < oldlen, which is certainly <= SIZE_MAX/2 */
+ replaced = safe_erealloc(replaced, maxlen, 1, ent_len + 128 + 1);
+ maxlen += ent_len + 128;
}
replaced[len++] = '&';
memcpy(&replaced[len], &old[cursor], ent_len);