2015-04-16 19:45:08 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| PHP Version 7 |
|
|
|
|
+----------------------------------------------------------------------+
|
2016-01-02 17:56:11 +08:00
|
|
|
| Copyright (c) 1997-2016 The PHP Group |
|
2015-04-16 19:45:08 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
|
|
| available through the world-wide-web at the following url: |
|
|
|
|
| http://www.php.net/license/3_01.txt |
|
|
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Author: Dmitry Stogov <dmitry@zend.com> |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <zend.h>
|
2015-11-21 23:14:12 +08:00
|
|
|
#include "zend_smart_str.h"
|
2015-04-16 19:45:08 +08:00
|
|
|
|
2015-06-29 21:44:54 +08:00
|
|
|
#define SMART_STR_OVERHEAD (ZEND_MM_OVERHEAD + _ZSTR_HEADER_SIZE)
|
2015-04-16 19:45:08 +08:00
|
|
|
|
|
|
|
#ifndef SMART_STR_PAGE
|
|
|
|
# define SMART_STR_PAGE 4096
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SMART_STR_START_SIZE
|
|
|
|
# define SMART_STR_START_SIZE (256 - SMART_STR_OVERHEAD - 1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define SMART_STR_NEW_SIZE(len) \
|
|
|
|
(((len + SMART_STR_OVERHEAD + SMART_STR_PAGE) & ~(SMART_STR_PAGE - 1)) - SMART_STR_OVERHEAD - 1)
|
|
|
|
|
|
|
|
ZEND_API void ZEND_FASTCALL smart_str_erealloc(smart_str *str, size_t len)
|
|
|
|
{
|
|
|
|
if (UNEXPECTED(!str->s)) {
|
|
|
|
str->a = len < SMART_STR_START_SIZE
|
|
|
|
? SMART_STR_START_SIZE
|
|
|
|
: SMART_STR_NEW_SIZE(len);
|
|
|
|
str->s = zend_string_alloc(str->a, 0);
|
2015-06-30 18:59:27 +08:00
|
|
|
ZSTR_LEN(str->s) = 0;
|
2015-04-16 19:45:08 +08:00
|
|
|
} else {
|
|
|
|
str->a = SMART_STR_NEW_SIZE(len);
|
2015-06-30 18:59:27 +08:00
|
|
|
str->s = (zend_string *) erealloc2(str->s, _ZSTR_HEADER_SIZE + str->a + 1, _ZSTR_HEADER_SIZE + ZSTR_LEN(str->s) + 1);
|
2015-04-16 19:45:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ZEND_API void ZEND_FASTCALL smart_str_realloc(smart_str *str, size_t len)
|
|
|
|
{
|
|
|
|
if (UNEXPECTED(!str->s)) {
|
|
|
|
str->a = len < SMART_STR_START_SIZE
|
|
|
|
? SMART_STR_START_SIZE
|
|
|
|
: SMART_STR_NEW_SIZE(len);
|
|
|
|
str->s = zend_string_alloc(str->a, 1);
|
2015-06-30 18:59:27 +08:00
|
|
|
ZSTR_LEN(str->s) = 0;
|
2015-04-16 19:45:08 +08:00
|
|
|
} else {
|
|
|
|
str->a = SMART_STR_NEW_SIZE(len);
|
2015-06-29 21:44:54 +08:00
|
|
|
str->s = (zend_string *) realloc(str->s, _ZSTR_HEADER_SIZE + str->a + 1);
|
2015-04-16 19:45:08 +08:00
|
|
|
}
|
|
|
|
}
|
2015-11-21 23:14:12 +08:00
|
|
|
|
|
|
|
/* Windows uses VK_ESCAPE instead of \e */
|
|
|
|
#ifndef VK_ESCAPE
|
|
|
|
#define VK_ESCAPE '\e'
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static size_t zend_compute_escaped_string_len(const char *s, size_t l) {
|
|
|
|
size_t i, len = l;
|
|
|
|
for (i = 0; i < l; ++i) {
|
|
|
|
char c = s[i];
|
|
|
|
if (c == '\n' || c == '\r' || c == '\t' ||
|
|
|
|
c == '\f' || c == '\v' || c == '\\' || c == VK_ESCAPE) {
|
|
|
|
len += 1;
|
|
|
|
} else if (c < 32 || c > 126) {
|
|
|
|
len += 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
ZEND_API void ZEND_FASTCALL smart_str_append_escaped(smart_str *str, const char *s, size_t l) {
|
|
|
|
char *res;
|
|
|
|
size_t i, len = zend_compute_escaped_string_len(s, l);
|
|
|
|
|
|
|
|
smart_str_alloc(str, len, 0);
|
|
|
|
res = &ZSTR_VAL(str->s)[ZSTR_LEN(str->s)];
|
|
|
|
ZSTR_LEN(str->s) += len;
|
|
|
|
|
|
|
|
for (i = 0; i < l; ++i) {
|
|
|
|
unsigned char c = s[i];
|
|
|
|
if (c < 32 || c == '\\' || c > 126) {
|
|
|
|
*res++ = '\\';
|
|
|
|
switch (c) {
|
|
|
|
case '\n': *res++ = 'n'; break;
|
|
|
|
case '\r': *res++ = 'r'; break;
|
|
|
|
case '\t': *res++ = 't'; break;
|
|
|
|
case '\f': *res++ = 'f'; break;
|
|
|
|
case '\v': *res++ = 'v'; break;
|
|
|
|
case '\\': *res++ = '\\'; break;
|
|
|
|
case VK_ESCAPE: *res++ = 'e'; break;
|
|
|
|
default:
|
|
|
|
*res++ = 'x';
|
|
|
|
if ((c >> 4) < 10) {
|
|
|
|
*res++ = (c >> 4) + '0';
|
|
|
|
} else {
|
|
|
|
*res++ = (c >> 4) + 'A' - 10;
|
|
|
|
}
|
|
|
|
if ((c & 0xf) < 10) {
|
|
|
|
*res++ = (c & 0xf) + '0';
|
|
|
|
} else {
|
|
|
|
*res++ = (c & 0xf) + 'A' - 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
*res++ = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|