2010-04-20 19:05:54 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Zend Engine |
|
|
|
|
+----------------------------------------------------------------------+
|
2014-01-03 11:08:10 +08:00
|
|
|
| Copyright (c) 1998-2014 Zend Technologies Ltd. (http://www.zend.com) |
|
2010-04-20 19:05:54 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| This source file is subject to version 2.00 of the Zend license, |
|
2015-01-03 17:22:58 +08:00
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
2010-04-20 19:05:54 +08:00
|
|
|
| available through the world-wide-web at the following url: |
|
|
|
|
| http://www.zend.com/license/2_00.txt. |
|
|
|
|
| If you did not receive a copy of the Zend license and are unable to |
|
|
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
|
|
| license@zend.com so we can mail you a copy immediately. |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Authors: Dmitry Stogov <dmitry@zend.com> |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* $Id: $ */
|
|
|
|
|
|
|
|
#include "zend.h"
|
|
|
|
#include "zend_globals.h"
|
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
ZEND_API zend_string *(*zend_new_interned_string)(zend_string *str);
|
|
|
|
ZEND_API void (*zend_interned_strings_snapshot)(void);
|
|
|
|
ZEND_API void (*zend_interned_strings_restore)(void);
|
2010-05-25 17:00:20 +08:00
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
static zend_string *zend_new_interned_string_int(zend_string *str);
|
|
|
|
static void zend_interned_strings_snapshot_int(void);
|
|
|
|
static void zend_interned_strings_restore_int(void);
|
2010-04-20 19:05:54 +08:00
|
|
|
|
2014-08-26 01:24:55 +08:00
|
|
|
ZEND_API zend_ulong zend_hash_func(const char *str, size_t len)
|
2014-02-10 14:04:30 +08:00
|
|
|
{
|
|
|
|
return zend_inline_hash_func(str, len);
|
|
|
|
}
|
|
|
|
|
2014-08-26 05:05:05 +08:00
|
|
|
#ifndef ZTS
|
2014-02-17 21:59:18 +08:00
|
|
|
static void _str_dtor(zval *zv)
|
|
|
|
{
|
|
|
|
zend_string *str = Z_STR_P(zv);
|
2014-04-02 18:34:44 +08:00
|
|
|
GC_FLAGS(str) &= ~IS_STR_INTERNED;
|
|
|
|
GC_REFCOUNT(str) = 1;
|
2014-02-17 21:59:18 +08:00
|
|
|
}
|
2014-08-26 05:05:05 +08:00
|
|
|
#endif
|
2014-02-17 21:59:18 +08:00
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
void zend_interned_strings_init(void)
|
2010-04-20 19:05:54 +08:00
|
|
|
{
|
2014-10-06 21:44:43 +08:00
|
|
|
#ifndef ZTS
|
2014-02-10 14:04:30 +08:00
|
|
|
zend_string *str;
|
2010-04-20 19:05:54 +08:00
|
|
|
|
2014-04-21 22:25:34 +08:00
|
|
|
zend_hash_init(&CG(interned_strings), 1024, NULL, _str_dtor, 1);
|
2015-01-03 17:22:58 +08:00
|
|
|
|
2010-04-20 19:05:54 +08:00
|
|
|
CG(interned_strings).nTableMask = CG(interned_strings).nTableSize - 1;
|
2014-04-21 22:25:34 +08:00
|
|
|
CG(interned_strings).arData = (Bucket*) pecalloc(CG(interned_strings).nTableSize, sizeof(Bucket), 1);
|
2014-08-26 01:28:33 +08:00
|
|
|
CG(interned_strings).arHash = (uint32_t*) pecalloc(CG(interned_strings).nTableSize, sizeof(uint32_t), 1);
|
|
|
|
memset(CG(interned_strings).arHash, INVALID_IDX, CG(interned_strings).nTableSize * sizeof(uint32_t));
|
2014-02-10 14:04:30 +08:00
|
|
|
|
|
|
|
/* interned empty string */
|
2014-08-26 01:24:55 +08:00
|
|
|
str = zend_string_alloc(sizeof("")-1, 1);
|
2014-02-10 14:04:30 +08:00
|
|
|
str->val[0] = '\000';
|
2014-12-14 06:06:14 +08:00
|
|
|
CG(empty_string) = zend_new_interned_string_int(str);
|
2010-04-20 19:05:54 +08:00
|
|
|
#endif
|
|
|
|
|
2014-05-06 03:16:49 +08:00
|
|
|
/* one char strings (the actual interned strings are going to be created by ext/opcache) */
|
|
|
|
memset(CG(one_char_string), 0, sizeof(CG(one_char_string)));
|
|
|
|
|
2010-05-25 17:00:20 +08:00
|
|
|
zend_new_interned_string = zend_new_interned_string_int;
|
|
|
|
zend_interned_strings_snapshot = zend_interned_strings_snapshot_int;
|
|
|
|
zend_interned_strings_restore = zend_interned_strings_restore_int;
|
2010-04-20 19:05:54 +08:00
|
|
|
}
|
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
void zend_interned_strings_dtor(void)
|
2010-04-20 19:05:54 +08:00
|
|
|
{
|
|
|
|
#ifndef ZTS
|
2014-02-17 21:59:18 +08:00
|
|
|
zend_hash_destroy(&CG(interned_strings));
|
2010-04-20 19:05:54 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
static zend_string *zend_new_interned_string_int(zend_string *str)
|
2010-04-20 19:05:54 +08:00
|
|
|
{
|
|
|
|
#ifndef ZTS
|
2014-08-26 01:24:55 +08:00
|
|
|
zend_ulong h;
|
2010-04-20 19:05:54 +08:00
|
|
|
uint nIndex;
|
2014-02-10 14:04:30 +08:00
|
|
|
uint idx;
|
2010-04-20 19:05:54 +08:00
|
|
|
Bucket *p;
|
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
if (IS_INTERNED(str)) {
|
|
|
|
return str;
|
2010-04-20 19:05:54 +08:00
|
|
|
}
|
|
|
|
|
2014-08-26 01:24:55 +08:00
|
|
|
h = zend_string_hash_val(str);
|
2010-04-20 19:05:54 +08:00
|
|
|
nIndex = h & CG(interned_strings).nTableMask;
|
2014-02-10 14:04:30 +08:00
|
|
|
idx = CG(interned_strings).arHash[nIndex];
|
|
|
|
while (idx != INVALID_IDX) {
|
|
|
|
p = CG(interned_strings).arData + idx;
|
|
|
|
if ((p->h == h) && (p->key->len == str->len)) {
|
|
|
|
if (!memcmp(p->key->val, str->val, str->len)) {
|
2014-08-26 01:24:55 +08:00
|
|
|
zend_string_release(str);
|
2014-02-10 14:04:30 +08:00
|
|
|
return p->key;
|
2010-04-20 19:05:54 +08:00
|
|
|
}
|
|
|
|
}
|
2014-04-02 18:34:44 +08:00
|
|
|
idx = Z_NEXT(p->val);
|
2010-04-20 19:05:54 +08:00
|
|
|
}
|
2015-01-03 17:22:58 +08:00
|
|
|
|
2014-04-02 18:34:44 +08:00
|
|
|
GC_REFCOUNT(str) = 1;
|
|
|
|
GC_FLAGS(str) |= IS_STR_INTERNED;
|
2014-02-10 14:04:30 +08:00
|
|
|
|
|
|
|
if (CG(interned_strings).nNumUsed >= CG(interned_strings).nTableSize) {
|
2010-04-20 19:05:54 +08:00
|
|
|
if ((CG(interned_strings).nTableSize << 1) > 0) { /* Let's double the table size */
|
2014-04-21 22:25:34 +08:00
|
|
|
Bucket *d = (Bucket *) perealloc_recoverable(CG(interned_strings).arData, (CG(interned_strings).nTableSize << 1) * sizeof(Bucket), 1);
|
2014-08-26 01:28:33 +08:00
|
|
|
uint32_t *h = (uint32_t *) perealloc_recoverable(CG(interned_strings).arHash, (CG(interned_strings).nTableSize << 1) * sizeof(uint32_t), 1);
|
2010-04-20 19:05:54 +08:00
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
if (d && h) {
|
2010-04-20 19:05:54 +08:00
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
2014-02-10 14:04:30 +08:00
|
|
|
CG(interned_strings).arData = d;
|
|
|
|
CG(interned_strings).arHash = h;
|
2010-04-20 19:05:54 +08:00
|
|
|
CG(interned_strings).nTableSize = (CG(interned_strings).nTableSize << 1);
|
|
|
|
CG(interned_strings).nTableMask = CG(interned_strings).nTableSize - 1;
|
|
|
|
zend_hash_rehash(&CG(interned_strings));
|
|
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
2015-01-03 17:22:58 +08:00
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
idx = CG(interned_strings).nNumUsed++;
|
|
|
|
CG(interned_strings).nNumOfElements++;
|
|
|
|
p = CG(interned_strings).arData + idx;
|
|
|
|
p->h = h;
|
|
|
|
p->key = str;
|
|
|
|
Z_STR(p->val) = str;
|
2014-04-03 19:26:23 +08:00
|
|
|
Z_TYPE_INFO(p->val) = IS_INTERNED_STRING_EX;
|
2014-02-10 14:04:30 +08:00
|
|
|
nIndex = h & CG(interned_strings).nTableMask;
|
2014-04-02 18:34:44 +08:00
|
|
|
Z_NEXT(p->val) = CG(interned_strings).arHash[nIndex];
|
2014-02-10 14:04:30 +08:00
|
|
|
CG(interned_strings).arHash[nIndex] = idx;
|
2015-01-03 17:22:58 +08:00
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
2010-04-20 19:05:54 +08:00
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
return str;
|
2010-04-20 19:05:54 +08:00
|
|
|
#else
|
2014-02-10 14:04:30 +08:00
|
|
|
return str;
|
2010-04-20 19:05:54 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
static void zend_interned_strings_snapshot_int(void)
|
2010-04-20 19:05:54 +08:00
|
|
|
{
|
2014-02-10 14:04:30 +08:00
|
|
|
#ifndef ZTS
|
|
|
|
uint idx;
|
|
|
|
Bucket *p;
|
|
|
|
|
|
|
|
idx = CG(interned_strings).nNumUsed;
|
|
|
|
while (idx > 0) {
|
|
|
|
idx--;
|
|
|
|
p = CG(interned_strings).arData + idx;
|
2014-04-02 18:34:44 +08:00
|
|
|
ZEND_ASSERT(GC_FLAGS(p->key) & IS_STR_PERSISTENT);
|
|
|
|
GC_FLAGS(p->key) |= IS_STR_PERMANENT;
|
2014-02-10 14:04:30 +08:00
|
|
|
}
|
|
|
|
#endif
|
2010-04-20 19:05:54 +08:00
|
|
|
}
|
|
|
|
|
2014-12-14 06:06:14 +08:00
|
|
|
static void zend_interned_strings_restore_int(void)
|
2010-04-20 19:05:54 +08:00
|
|
|
{
|
|
|
|
#ifndef ZTS
|
2014-02-10 14:04:30 +08:00
|
|
|
uint nIndex;
|
|
|
|
uint idx;
|
2010-04-20 19:05:54 +08:00
|
|
|
Bucket *p;
|
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
idx = CG(interned_strings).nNumUsed;
|
|
|
|
while (idx > 0) {
|
|
|
|
idx--;
|
|
|
|
p = CG(interned_strings).arData + idx;
|
2014-04-02 18:34:44 +08:00
|
|
|
if (GC_FLAGS(p->key) & IS_STR_PERMANENT) break;
|
2014-02-10 14:04:30 +08:00
|
|
|
CG(interned_strings).nNumUsed--;
|
|
|
|
CG(interned_strings).nNumOfElements--;
|
|
|
|
|
2014-04-02 18:34:44 +08:00
|
|
|
GC_FLAGS(p->key) &= ~IS_STR_INTERNED;
|
|
|
|
GC_REFCOUNT(p->key) = 1;
|
2014-08-26 01:24:55 +08:00
|
|
|
zend_string_free(p->key);
|
2014-02-18 02:30:35 +08:00
|
|
|
|
2014-02-10 14:04:30 +08:00
|
|
|
nIndex = p->h & CG(interned_strings).nTableMask;
|
|
|
|
if (CG(interned_strings).arHash[nIndex] == idx) {
|
2014-04-02 18:34:44 +08:00
|
|
|
CG(interned_strings).arHash[nIndex] = Z_NEXT(p->val);
|
2014-02-10 14:04:30 +08:00
|
|
|
} else {
|
|
|
|
uint prev = CG(interned_strings).arHash[nIndex];
|
2014-04-02 18:34:44 +08:00
|
|
|
while (Z_NEXT(CG(interned_strings).arData[prev].val) != idx) {
|
|
|
|
prev = Z_NEXT(CG(interned_strings).arData[prev].val);
|
2014-02-10 14:04:30 +08:00
|
|
|
}
|
2014-04-02 18:34:44 +08:00
|
|
|
Z_NEXT(CG(interned_strings).arData[prev].val) = Z_NEXT(p->val);
|
2014-02-10 14:04:30 +08:00
|
|
|
}
|
|
|
|
}
|
2010-04-20 19:05:54 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Local variables:
|
|
|
|
* tab-width: 4
|
|
|
|
* c-basic-offset: 4
|
|
|
|
* indent-tabs-mode: t
|
|
|
|
* End:
|
|
|
|
*/
|