improve message parsing performance by about 25%
authorMichael Wallner <mike@php.net>
Mon, 28 Jul 2014 12:19:57 +0000 (14:19 +0200)
committerMichael Wallner <mike@php.net>
Mon, 28 Jul 2014 12:19:57 +0000 (14:19 +0200)
php_http_misc.c
php_http_misc.h

index 79c37893398bc09d860fff560de33083c21c79db..7afa006053299cb1215069e98c49a9a47c765e1d 100644 (file)
@@ -87,16 +87,16 @@ int php_http_match(const char *haystack_str, const char *needle_str, int flags)
        return result;
 }
 
-char *php_http_pretty_key(char *key, size_t key_len, zend_bool uctitle, zend_bool xhyphen)
+char *php_http_pretty_key(register char *key, size_t key_len, zend_bool uctitle, zend_bool xhyphen)
 {
-       size_t i;
+       size_t i = 1;
        int wasalpha;
 
        if (key && key_len) {
                if ((wasalpha = PHP_HTTP_IS_CTYPE(alpha, key[0]))) {
                        key[0] = (char) (uctitle ? PHP_HTTP_TO_CTYPE(upper, key[0]) : PHP_HTTP_TO_CTYPE(lower, key[0]));
                }
-               for (i = 1; i < key_len; i++) {
+               PHP_HTTP_DUFF(1, key_len,
                        if (PHP_HTTP_IS_CTYPE(alpha, key[i])) {
                                key[i] = (char) (((!wasalpha) && uctitle) ? PHP_HTTP_TO_CTYPE(upper, key[i]) : PHP_HTTP_TO_CTYPE(lower, key[i]));
                                wasalpha = 1;
@@ -106,7 +106,8 @@ char *php_http_pretty_key(char *key, size_t key_len, zend_bool uctitle, zend_boo
                                }
                                wasalpha = 0;
                        }
-               }
+                       ++i;
+               );
        }
        return key;
 }
index b8c54faba2b51da8839436187f65c423308f98cf..d47477c91ccaf87dc6256ae84bbef31d6c5fe14a 100644 (file)
@@ -60,25 +60,51 @@ PHP_HTTP_API void php_http_sleep(double s);
 #define PHP_HTTP_MATCH_STRICT  (PHP_HTTP_MATCH_CASE|PHP_HTTP_MATCH_FULL)
 
 int php_http_match(const char *haystack, const char *needle, int flags);
-char *php_http_pretty_key(char *key, size_t key_len, zend_bool uctitle, zend_bool xhyphen);
+char *php_http_pretty_key(register char *key, size_t key_len, zend_bool uctitle, zend_bool xhyphen);
 size_t php_http_boundary(char *buf, size_t len TSRMLS_DC);
 int php_http_select_str(const char *cmp, int argc, ...);
 
-static inline const char *php_http_locate_str(const char *h, size_t h_len, const char *n, size_t n_len)
+/* See "A Reusable Duff Device" By Ralf Holly, August 01, 2005 */
+#define PHP_HTTP_DUFF_BREAK(i) do { \
+       times_##i = 1; \
+} while (0)
+
+#define PHP_HTTP_DUFF(i, c, a) do { \
+               size_t count_##i = (c); \
+               size_t times_##i = (count_##i + 7) >> 3; \
+               switch (count_##i & 7){ \
+               case 0: do { a; \
+               case 7: a; \
+               case 6: a; \
+               case 5: a; \
+               case 4: a; \
+               case 3: a; \
+               case 2: a; \
+               case 1: a; \
+               } while (--times_##i > 0); \
+       } \
+} while (0)
+
+
+static inline const char *php_http_locate_str(register const char *h, size_t h_len, const char *n, size_t n_len)
 {
-       const char *p, *e;
+       register const char *p1, *p2;
 
-       if (n_len && h_len) {
-               e = h + h_len;
-               do {
+       if (n_len && h_len && h_len >= n_len) {
+               PHP_HTTP_DUFF(1, h_len - n_len + 1,
                        if (*h == *n) {
-                               for (p = n; *p == h[p-n]; ++p) {
-                                       if (p == n+n_len-1) {
+                               p1 = h;
+                               p2 = n;
+                               PHP_HTTP_DUFF(2, n_len,
+                                       if (*p1++ != *p2++) {
+                                               PHP_HTTP_DUFF_BREAK(2);
+                                       } else if (p2 == n + n_len - 1) {
                                                return h;
                                        }
-                               }
+                               );
                        }
-               } while (h++ != e);
+                       ++h;
+               );
        }
 
        return NULL;
@@ -96,17 +122,19 @@ static inline const char *php_http_locate_eol(const char *line, int *eol_len)
 
 static inline const char *php_http_locate_bin_eol(const char *bin, size_t len, int *eol_len)
 {
-       const char *eol;
+       register const char *eol = bin;
 
-       for (eol = bin; eol - bin < len; ++eol) {
-               if (*eol == '\r' || *eol == '\n') {
-                       if (eol_len) {
-                               *eol_len = ((eol[0] == '\r' && eol[1] == '\n') ? 2 : 1);
+       if (len > 0) {
+               PHP_HTTP_DUFF(1, len,
+                       if (*eol == '\r' || *eol == '\n') {
+                               if (eol_len) {
+                                       *eol_len = ((eol[0] == '\r' && eol[1] == '\n') ? 2 : 1);
+                               }
+                               return eol;
                        }
-                       return eol;
-               }
+                       ++eol;
+               );
        }
-
        return NULL;
 }