]> granicus.if.org Git - php/commitdiff
Fixed: removed possible integer over-/underflows
authorStefan Esser <sesser@php.net>
Sun, 28 Nov 2004 12:44:28 +0000 (12:44 +0000)
committerStefan Esser <sesser@php.net>
Sun, 28 Nov 2004 12:44:28 +0000 (12:44 +0000)
ext/standard/pack.c
main/php.h

index d45f9817d15278f76e0f46e5ce1cfc36a9cd8939..f41695c56249d4b41f3c9b45ea85a2c6a8482502 100644 (file)
 #include <netinet/in.h>
 #endif
 
+#define INC_OUTPUTPOS(a,b) \
+       if ((a) < 0 || ((INT_MAX - outputpos)/(b)) < (a)) { \
+               php_error_docref(NULL TSRMLS_CC, E_WARNING, "Type %c: integer overflow in format string", code); \
+               RETURN_FALSE; \
+       } \
+       outputpos += (a)*(b);
+
 /* Whether machine is little endian */
 char machine_little_endian;
 
@@ -243,7 +250,7 @@ PHP_FUNCTION(pack)
                switch ((int) code) {
                        case 'h': 
                        case 'H': 
-                               outputpos += (arg + 1) / 2;             /* 4 bit per arg */
+                               INC_OUTPUTPOS((arg + 1) / 2,1)  /* 4 bit per arg */
                                break;
 
                        case 'a': 
@@ -251,34 +258,34 @@ PHP_FUNCTION(pack)
                        case 'c': 
                        case 'C':
                        case 'x':
-                               outputpos += arg;               /* 8 bit per arg */
+                               INC_OUTPUTPOS(arg,1)            /* 8 bit per arg */
                                break;
 
                        case 's': 
                        case 'S': 
                        case 'n': 
                        case 'v':
-                               outputpos += arg * 2;   /* 16 bit per arg */
+                               INC_OUTPUTPOS(arg,2)            /* 16 bit per arg */
                                break;
 
                        case 'i': 
                        case 'I':
-                               outputpos += arg * sizeof(int);
+                               INC_OUTPUTPOS(arg,sizeof(int))
                                break;
 
                        case 'l': 
                        case 'L': 
                        case 'N': 
                        case 'V':
-                               outputpos += arg * 4;   /* 32 bit per arg */
+                               INC_OUTPUTPOS(arg,4)            /* 32 bit per arg */
                                break;
 
                        case 'f':
-                               outputpos += arg * sizeof(float);
+                               INC_OUTPUTPOS(arg,sizeof(float))
                                break;
 
                        case 'd':
-                               outputpos += arg * sizeof(double);
+                               INC_OUTPUTPOS(arg,sizeof(double))
                                break;
 
                        case 'X':
@@ -647,6 +654,11 @@ PHP_FUNCTION(unpack)
                                sprintf(n, "%.*s", namelen, name);
                        }
 
+                       if (size != 0 && size != -1 && INT_MAX - size + 1 < inputpos) {
+                               php_error_docref(NULL TSRMLS_CC, E_WARNING, "Type %c: integer overflow", type);
+                               inputpos = 0;
+                       }
+
                        if ((inputpos + size) <= inputlen) {
                                switch ((int) type) {
                                        case 'a': 
@@ -817,6 +829,10 @@ PHP_FUNCTION(unpack)
                                }
 
                                inputpos += size;
+                               if (inputpos < 0) {
+                                       php_error_docref(NULL TSRMLS_CC, E_WARNING, "Type %c: outside of string", type);
+                                       inputpos = 0;
+                               }
                        } else if (arg < 0) {
                                /* Reached end of input for '*' repeater */
                                break;
index 6c886f2b4d113d148a2b2b5f73f41f4ceb779a63..9fd661905700bcbbaf0ec562ed0518eaec9d93de 100644 (file)
@@ -222,6 +222,14 @@ char *strerror(int);
 #define LONG_MIN (- LONG_MAX - 1)
 #endif
 
+#ifndef INT_MAX
+#define INT_MAX 2147483647
+#endif
+
+#ifndef INT_MIN
+#define INT_MIN (- INT_MAX - 1)
+#endif
+
 #define PHP_GCC_VERSION ZEND_GCC_VERSION
 #define PHP_ATTRIBUTE_MALLOC ZEND_ATTRIBUTE_MALLOC
 #define PHP_ATTRIBUTE_FORMAT ZEND_ATTRIBUTE_FORMAT