mirror of
https://github.com/php/php-src.git
synced 2024-12-14 04:16:30 +08:00
902 lines
28 KiB
C
902 lines
28 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| Zend OPcache |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 1998-2018 The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| 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. |
|
|
+----------------------------------------------------------------------+
|
|
| Authors: Andi Gutmans <andi@zend.com> |
|
|
| Zeev Suraski <zeev@zend.com> |
|
|
| Stanislav Malyshev <stas@zend.com> |
|
|
| Dmitry Stogov <dmitry@zend.com> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include "zend_API.h"
|
|
#include "zend_constants.h"
|
|
#include "zend_accelerator_util_funcs.h"
|
|
#include "zend_persist.h"
|
|
#include "zend_shared_alloc.h"
|
|
|
|
#if SIZEOF_SIZE_T <= SIZEOF_ZEND_LONG
|
|
/* If sizeof(void*) == sizeof(ulong) we can use zend_hash index functions */
|
|
# define accel_xlat_set(old, new) zend_hash_index_add_new_ptr(&ZCG(bind_hash), (zend_ulong)(zend_uintptr_t)(old), (new))
|
|
# define accel_xlat_get(old) zend_hash_index_find_ptr(&ZCG(bind_hash), (zend_ulong)(zend_uintptr_t)(old))
|
|
#else
|
|
# define accel_xlat_set(old, new) zend_hash_str_add_new_ptr(&ZCG(bind_hash), (char*)&(old), sizeof(void*), (new))
|
|
# define accel_xlat_get(old) zend_hash_str_find_ptr(&ZCG(bind_hash), (char*)&(old), sizeof(void*))
|
|
#endif
|
|
|
|
#define ARENA_REALLOC(ptr) \
|
|
(void*)(((char*)(ptr)) + ((char*)ZCG(arena_mem) - (char*)ZCG(current_persistent_script)->arena_mem))
|
|
|
|
typedef int (*id_function_t)(void *, void *);
|
|
typedef void (*unique_copy_ctor_func_t)(void *pElement);
|
|
|
|
zend_persistent_script* create_persistent_script(void)
|
|
{
|
|
zend_persistent_script *persistent_script = (zend_persistent_script *) emalloc(sizeof(zend_persistent_script));
|
|
memset(persistent_script, 0, sizeof(zend_persistent_script));
|
|
|
|
zend_hash_init(&persistent_script->script.function_table, 128, NULL, ZEND_FUNCTION_DTOR, 0);
|
|
/* class_table is usually destroyed by free_persistent_script() that
|
|
* overrides destructor. ZEND_CLASS_DTOR may be used by standard
|
|
* PHP compiler
|
|
*/
|
|
zend_hash_init(&persistent_script->script.class_table, 16, NULL, ZEND_CLASS_DTOR, 0);
|
|
|
|
return persistent_script;
|
|
}
|
|
|
|
void free_persistent_script(zend_persistent_script *persistent_script, int destroy_elements)
|
|
{
|
|
if (!destroy_elements) {
|
|
persistent_script->script.function_table.pDestructor = NULL;
|
|
persistent_script->script.class_table.pDestructor = NULL;
|
|
}
|
|
|
|
zend_hash_destroy(&persistent_script->script.function_table);
|
|
zend_hash_destroy(&persistent_script->script.class_table);
|
|
|
|
if (persistent_script->script.filename) {
|
|
zend_string_release_ex(persistent_script->script.filename, 0);
|
|
}
|
|
|
|
efree(persistent_script);
|
|
}
|
|
|
|
void zend_accel_move_user_functions(HashTable *src, zend_script *script)
|
|
{
|
|
Bucket *p;
|
|
HashTable *dst = &script->function_table;
|
|
zend_string *filename = script->main_op_array.filename;
|
|
dtor_func_t orig_dtor = src->pDestructor;
|
|
|
|
src->pDestructor = NULL;
|
|
zend_hash_extend(dst, dst->nNumUsed + src->nNumUsed, 0);
|
|
ZEND_HASH_REVERSE_FOREACH_BUCKET(src, p) {
|
|
zend_function *function = Z_PTR(p->val);
|
|
|
|
if (EXPECTED(function->type == ZEND_USER_FUNCTION)
|
|
&& EXPECTED(function->op_array.filename == filename)) {
|
|
_zend_hash_append_ptr(dst, p->key, function);
|
|
zend_hash_del_bucket(src, p);
|
|
} else {
|
|
break;
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
src->pDestructor = orig_dtor;
|
|
}
|
|
|
|
static inline void zend_clone_zval(zval *src)
|
|
{
|
|
void *ptr;
|
|
|
|
if (Z_TYPE_P(src) == IS_REFERENCE) {
|
|
ptr = accel_xlat_get(Z_REF_P(src));
|
|
if (ptr != NULL) {
|
|
Z_REF_P(src) = ptr;
|
|
return;
|
|
} else {
|
|
zend_reference *old = Z_REF_P(src);
|
|
ZVAL_NEW_REF(src, &old->val);
|
|
Z_REF_P(src)->gc = old->gc;
|
|
accel_xlat_set(old, Z_REF_P(src));
|
|
src = Z_REFVAL_P(src);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void zend_hash_clone_constants(HashTable *ht, HashTable *source)
|
|
{
|
|
Bucket *p, *q, *end;
|
|
zend_ulong nIndex;
|
|
zend_class_constant *c;
|
|
|
|
ht->nTableSize = source->nTableSize;
|
|
ht->nTableMask = source->nTableMask;
|
|
ht->nNumUsed = 0;
|
|
ht->nNumOfElements = source->nNumOfElements;
|
|
ht->nNextFreeElement = source->nNextFreeElement;
|
|
ht->pDestructor = NULL;
|
|
HT_FLAGS(ht) = (HT_FLAGS(source) & (HASH_FLAG_INITIALIZED | HASH_FLAG_STATIC_KEYS));
|
|
ht->nInternalPointer = 0;
|
|
|
|
if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
|
|
ht->arData = source->arData;
|
|
return;
|
|
}
|
|
|
|
ZEND_ASSERT((HT_FLAGS(source) & HASH_FLAG_PACKED) == 0);
|
|
HT_SET_DATA_ADDR(ht, emalloc(HT_SIZE(ht)));
|
|
HT_HASH_RESET(ht);
|
|
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
|
|
nIndex = p->h | ht->nTableMask;
|
|
|
|
/* Insert into hash collision list */
|
|
q = ht->arData + ht->nNumUsed;
|
|
Z_NEXT(q->val) = HT_HASH(ht, nIndex);
|
|
HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(ht->nNumUsed++);
|
|
|
|
/* Initialize key */
|
|
q->h = p->h;
|
|
q->key = p->key;
|
|
|
|
/* Copy data */
|
|
c = ARENA_REALLOC(Z_PTR(p->val));
|
|
ZVAL_PTR(&q->val, c);
|
|
|
|
zend_clone_zval(&c->value);
|
|
if ((void*)c->ce >= ZCG(current_persistent_script)->arena_mem &&
|
|
(void*)c->ce < (void*)((char*)ZCG(current_persistent_script)->arena_mem + ZCG(current_persistent_script)->arena_size)) {
|
|
c->ce = ARENA_REALLOC(c->ce);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void zend_hash_clone_methods(HashTable *ht, HashTable *source, zend_class_entry *old_ce, zend_class_entry *ce)
|
|
{
|
|
Bucket *p, *q, *end;
|
|
zend_ulong nIndex;
|
|
zend_op_array *new_entry;
|
|
|
|
ht->nTableSize = source->nTableSize;
|
|
ht->nTableMask = source->nTableMask;
|
|
ht->nNumUsed = 0;
|
|
ht->nNumOfElements = source->nNumOfElements;
|
|
ht->nNextFreeElement = source->nNextFreeElement;
|
|
ht->pDestructor = ZEND_FUNCTION_DTOR;
|
|
HT_FLAGS(ht) = (HT_FLAGS(source) & (HASH_FLAG_INITIALIZED | HASH_FLAG_STATIC_KEYS));
|
|
ht->nInternalPointer = 0;
|
|
|
|
if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
|
|
ht->arData = source->arData;
|
|
return;
|
|
}
|
|
|
|
ZEND_ASSERT(!(HT_FLAGS(source) & HASH_FLAG_PACKED));
|
|
HT_SET_DATA_ADDR(ht, emalloc(HT_SIZE(ht)));
|
|
HT_HASH_RESET(ht);
|
|
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
|
|
|
|
nIndex = p->h | ht->nTableMask;
|
|
|
|
/* Insert into hash collision list */
|
|
q = ht->arData + ht->nNumUsed;
|
|
Z_NEXT(q->val) = HT_HASH(ht, nIndex);
|
|
HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(ht->nNumUsed++);
|
|
|
|
/* Initialize key */
|
|
q->h = p->h;
|
|
ZEND_ASSERT(p->key != NULL);
|
|
q->key = p->key;
|
|
|
|
/* Copy data */
|
|
ZVAL_PTR(&q->val, ARENA_REALLOC(Z_PTR(p->val)));
|
|
new_entry = (zend_op_array*)Z_PTR(q->val);
|
|
|
|
if ((void*)new_entry->scope >= ZCG(current_persistent_script)->arena_mem &&
|
|
(void*)new_entry->scope < (void*)((char*)ZCG(current_persistent_script)->arena_mem + ZCG(current_persistent_script)->arena_size)) {
|
|
|
|
new_entry->scope = ARENA_REALLOC(new_entry->scope);
|
|
|
|
/* update prototype */
|
|
if (new_entry->prototype) {
|
|
new_entry->prototype = ARENA_REALLOC(new_entry->prototype);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void zend_hash_clone_prop_info(HashTable *ht, HashTable *source, zend_class_entry *old_ce)
|
|
{
|
|
Bucket *p, *q, *end;
|
|
zend_ulong nIndex;
|
|
zend_property_info *prop_info;
|
|
|
|
ht->nTableSize = source->nTableSize;
|
|
ht->nTableMask = source->nTableMask;
|
|
ht->nNumUsed = 0;
|
|
ht->nNumOfElements = source->nNumOfElements;
|
|
ht->nNextFreeElement = source->nNextFreeElement;
|
|
ht->pDestructor = NULL;
|
|
HT_FLAGS(ht) = (HT_FLAGS(source) & (HASH_FLAG_INITIALIZED | HASH_FLAG_STATIC_KEYS));
|
|
ht->nInternalPointer = 0;
|
|
|
|
if (!(HT_FLAGS(ht) & HASH_FLAG_INITIALIZED)) {
|
|
ht->arData = source->arData;
|
|
return;
|
|
}
|
|
|
|
ZEND_ASSERT(!(HT_FLAGS(source) & HASH_FLAG_PACKED));
|
|
HT_SET_DATA_ADDR(ht, emalloc(HT_SIZE(ht)));
|
|
HT_HASH_RESET(ht);
|
|
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
|
|
|
|
nIndex = p->h | ht->nTableMask;
|
|
|
|
/* Insert into hash collision list */
|
|
q = ht->arData + ht->nNumUsed;
|
|
Z_NEXT(q->val) = HT_HASH(ht, nIndex);
|
|
HT_HASH(ht, nIndex) = HT_IDX_TO_HASH(ht->nNumUsed++);
|
|
|
|
/* Initialize key */
|
|
q->h = p->h;
|
|
ZEND_ASSERT(p->key != NULL);
|
|
q->key = p->key;
|
|
|
|
/* Copy data */
|
|
prop_info = ARENA_REALLOC(Z_PTR(p->val));
|
|
ZVAL_PTR(&q->val, prop_info);
|
|
|
|
if ((void*)prop_info->ce >= ZCG(current_persistent_script)->arena_mem &&
|
|
(void*)prop_info->ce < (void*)((char*)ZCG(current_persistent_script)->arena_mem + ZCG(current_persistent_script)->arena_size)) {
|
|
prop_info->ce = ARENA_REALLOC(prop_info->ce);
|
|
}
|
|
}
|
|
}
|
|
|
|
#define zend_update_inherited_handler(handler) \
|
|
{ \
|
|
if (ce->handler != NULL) { \
|
|
ce->handler = ARENA_REALLOC(ce->handler); \
|
|
} \
|
|
}
|
|
|
|
/* Protects class' refcount, copies default properties, functions and class name */
|
|
static void zend_class_copy_ctor(zend_class_entry **pce)
|
|
{
|
|
zend_class_entry *ce = *pce;
|
|
zend_class_entry *old_ce = ce;
|
|
zval *src, *dst, *end;
|
|
|
|
*pce = ce = ARENA_REALLOC(old_ce);
|
|
ce->refcount = 1;
|
|
|
|
if (ce->parent && !(ce->ce_flags & ZEND_ACC_UNRESOLVED_PARENT)) {
|
|
ce->parent = ARENA_REALLOC(ce->parent);
|
|
}
|
|
|
|
if (old_ce->default_properties_table) {
|
|
ce->default_properties_table = emalloc(sizeof(zval) * old_ce->default_properties_count);
|
|
src = old_ce->default_properties_table;
|
|
end = src + old_ce->default_properties_count;
|
|
dst = ce->default_properties_table;
|
|
for (; src != end; src++, dst++) {
|
|
ZVAL_COPY_VALUE(dst, src);
|
|
zend_clone_zval(dst);
|
|
}
|
|
}
|
|
|
|
zend_hash_clone_methods(&ce->function_table, &old_ce->function_table, old_ce, ce);
|
|
|
|
/* static members */
|
|
if (old_ce->default_static_members_table) {
|
|
int i, end;
|
|
zend_class_entry *parent = (ce->ce_flags & ZEND_ACC_UNRESOLVED_PARENT) ? NULL : ce->parent;
|
|
|
|
ce->default_static_members_table = emalloc(sizeof(zval) * old_ce->default_static_members_count);
|
|
i = ce->default_static_members_count - 1;
|
|
|
|
/* Copy static properties in this class */
|
|
end = parent ? parent->default_static_members_count : 0;
|
|
for (; i >= end; i--) {
|
|
zval *p = &ce->default_static_members_table[i];
|
|
ZVAL_COPY_VALUE(p, &old_ce->default_static_members_table[i]);
|
|
zend_clone_zval(p);
|
|
}
|
|
|
|
/* Create indirections to static properties from parent classes */
|
|
while (parent && parent->default_static_members_table) {
|
|
end = parent->parent ? parent->parent->default_static_members_count : 0;
|
|
for (; i >= end; i--) {
|
|
zval *p = &ce->default_static_members_table[i];
|
|
ZVAL_INDIRECT(p, &parent->default_static_members_table[i]);
|
|
}
|
|
|
|
parent = parent->parent;
|
|
}
|
|
}
|
|
ce->static_members_table = ce->default_static_members_table;
|
|
|
|
/* properties_info */
|
|
zend_hash_clone_prop_info(&ce->properties_info, &old_ce->properties_info, old_ce);
|
|
|
|
/* constants table */
|
|
zend_hash_clone_constants(&ce->constants_table, &old_ce->constants_table);
|
|
|
|
if (ce->num_interfaces) {
|
|
zend_class_name *interface_names;
|
|
|
|
ZEND_ASSERT(ce->ce_flags & ZEND_ACC_UNRESOLVED_INTERFACES);
|
|
interface_names = emalloc(sizeof(zend_class_name) * ce->num_interfaces);
|
|
memcpy(interface_names, ce->interface_names, sizeof(zend_class_name) * ce->num_interfaces);
|
|
ce->interface_names = interface_names;
|
|
}
|
|
|
|
zend_update_inherited_handler(constructor);
|
|
zend_update_inherited_handler(destructor);
|
|
zend_update_inherited_handler(clone);
|
|
zend_update_inherited_handler(__get);
|
|
zend_update_inherited_handler(__set);
|
|
zend_update_inherited_handler(__call);
|
|
/* 5.1 stuff */
|
|
zend_update_inherited_handler(serialize_func);
|
|
zend_update_inherited_handler(unserialize_func);
|
|
zend_update_inherited_handler(__isset);
|
|
zend_update_inherited_handler(__unset);
|
|
/* 5.2 stuff */
|
|
zend_update_inherited_handler(__tostring);
|
|
|
|
/* 5.3 stuff */
|
|
zend_update_inherited_handler(__callstatic);
|
|
zend_update_inherited_handler(__debugInfo);
|
|
|
|
/* 5.4 traits */
|
|
if (ce->num_traits) {
|
|
zend_class_name *trait_names = emalloc(sizeof(zend_class_name) * ce->num_traits);
|
|
|
|
memcpy(trait_names, ce->trait_names, sizeof(zend_class_name) * ce->num_traits);
|
|
ce->trait_names = trait_names;
|
|
|
|
if (ce->trait_aliases) {
|
|
zend_trait_alias **trait_aliases;
|
|
int i = 0;
|
|
|
|
while (ce->trait_aliases[i]) {
|
|
i++;
|
|
}
|
|
trait_aliases = emalloc(sizeof(zend_trait_alias*) * (i + 1));
|
|
i = 0;
|
|
while (ce->trait_aliases[i]) {
|
|
trait_aliases[i] = emalloc(sizeof(zend_trait_alias));
|
|
memcpy(trait_aliases[i], ce->trait_aliases[i], sizeof(zend_trait_alias));
|
|
i++;
|
|
}
|
|
trait_aliases[i] = NULL;
|
|
ce->trait_aliases = trait_aliases;
|
|
}
|
|
|
|
if (ce->trait_precedences) {
|
|
zend_trait_precedence **trait_precedences;
|
|
int i = 0;
|
|
|
|
while (ce->trait_precedences[i]) {
|
|
i++;
|
|
}
|
|
trait_precedences = emalloc(sizeof(zend_trait_precedence*) * (i + 1));
|
|
i = 0;
|
|
while (ce->trait_precedences[i]) {
|
|
trait_precedences[i] = emalloc(sizeof(zend_trait_precedence) + (ce->trait_precedences[i]->num_excludes - 1) * sizeof(zend_string*));
|
|
memcpy(trait_precedences[i], ce->trait_precedences[i], sizeof(zend_trait_precedence) + (ce->trait_precedences[i]->num_excludes - 1) * sizeof(zend_string*));
|
|
i++;
|
|
}
|
|
trait_precedences[i] = NULL;
|
|
ce->trait_precedences = trait_precedences;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void zend_accel_function_hash_copy(HashTable *target, HashTable *source)
|
|
{
|
|
zend_function *function1, *function2;
|
|
Bucket *p, *end;
|
|
zval *t;
|
|
|
|
zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
|
|
ZEND_ASSERT(p->key);
|
|
t = zend_hash_find_ex(target, p->key, 1);
|
|
if (UNEXPECTED(t != NULL)) {
|
|
if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
|
|
/* Mangled key */
|
|
t = zend_hash_update(target, p->key, &p->val);
|
|
} else {
|
|
goto failure;
|
|
}
|
|
} else {
|
|
_zend_hash_append_ptr(target, p->key, Z_PTR(p->val));
|
|
}
|
|
}
|
|
target->nInternalPointer = 0;
|
|
return;
|
|
|
|
failure:
|
|
function1 = Z_PTR(p->val);
|
|
function2 = Z_PTR_P(t);
|
|
CG(in_compilation) = 1;
|
|
zend_set_compiled_filename(function1->op_array.filename);
|
|
CG(zend_lineno) = function1->op_array.opcodes[0].lineno;
|
|
if (function2->type == ZEND_USER_FUNCTION
|
|
&& function2->op_array.last > 0) {
|
|
zend_error(E_ERROR, "Cannot redeclare %s() (previously declared in %s:%d)",
|
|
ZSTR_VAL(function1->common.function_name),
|
|
ZSTR_VAL(function2->op_array.filename),
|
|
(int)function2->op_array.opcodes[0].lineno);
|
|
} else {
|
|
zend_error(E_ERROR, "Cannot redeclare %s()", ZSTR_VAL(function1->common.function_name));
|
|
}
|
|
}
|
|
|
|
static void zend_accel_function_hash_copy_from_shm(HashTable *target, HashTable *source)
|
|
{
|
|
zend_function *function1, *function2;
|
|
Bucket *p, *end;
|
|
zval *t;
|
|
|
|
zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
|
|
ZEND_ASSERT(p->key);
|
|
t = zend_hash_find_ex(target, p->key, 1);
|
|
if (UNEXPECTED(t != NULL)) {
|
|
if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
|
|
/* Mangled key */
|
|
zend_hash_update_ptr(target, p->key, Z_PTR(p->val));
|
|
} else {
|
|
goto failure;
|
|
}
|
|
} else {
|
|
_zend_hash_append_ptr_ex(target, p->key, Z_PTR(p->val), 1);
|
|
}
|
|
}
|
|
target->nInternalPointer = 0;
|
|
return;
|
|
|
|
failure:
|
|
function1 = Z_PTR(p->val);
|
|
function2 = Z_PTR_P(t);
|
|
CG(in_compilation) = 1;
|
|
zend_set_compiled_filename(function1->op_array.filename);
|
|
CG(zend_lineno) = function1->op_array.opcodes[0].lineno;
|
|
if (function2->type == ZEND_USER_FUNCTION
|
|
&& function2->op_array.last > 0) {
|
|
zend_error(E_ERROR, "Cannot redeclare %s() (previously declared in %s:%d)",
|
|
ZSTR_VAL(function1->common.function_name),
|
|
ZSTR_VAL(function2->op_array.filename),
|
|
(int)function2->op_array.opcodes[0].lineno);
|
|
} else {
|
|
zend_error(E_ERROR, "Cannot redeclare %s()", ZSTR_VAL(function1->common.function_name));
|
|
}
|
|
}
|
|
|
|
static void zend_accel_class_hash_copy(HashTable *target, HashTable *source)
|
|
{
|
|
Bucket *p, *end;
|
|
zval *t;
|
|
|
|
zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
if (UNEXPECTED(Z_TYPE(p->val) == IS_UNDEF)) continue;
|
|
ZEND_ASSERT(p->key);
|
|
t = zend_hash_find_ex(target, p->key, 1);
|
|
if (UNEXPECTED(t != NULL)) {
|
|
if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
|
|
/* Mangled key - ignore and wait for runtime */
|
|
continue;
|
|
} else if (UNEXPECTED(!ZCG(accel_directives).ignore_dups)) {
|
|
zend_class_entry *ce1 = Z_PTR(p->val);
|
|
if (!(ce1->ce_flags & ZEND_ACC_ANON_CLASS)) {
|
|
CG(in_compilation) = 1;
|
|
zend_set_compiled_filename(ce1->info.user.filename);
|
|
CG(zend_lineno) = ce1->info.user.line_start;
|
|
zend_error(E_ERROR,
|
|
"Cannot declare %s %s, because the name is already in use",
|
|
zend_get_object_type(ce1), ZSTR_VAL(ce1->name));
|
|
return;
|
|
}
|
|
continue;
|
|
}
|
|
} else {
|
|
t = _zend_hash_append_ptr(target, p->key, Z_PTR(p->val));
|
|
}
|
|
}
|
|
target->nInternalPointer = 0;
|
|
return;
|
|
}
|
|
|
|
static void zend_accel_class_hash_copy_from_shm(HashTable *target, HashTable *source)
|
|
{
|
|
Bucket *p, *end;
|
|
zval *t;
|
|
|
|
zend_hash_extend(target, target->nNumUsed + source->nNumUsed, 0);
|
|
p = source->arData;
|
|
end = p + source->nNumUsed;
|
|
for (; p != end; p++) {
|
|
ZEND_ASSERT(Z_TYPE(p->val) != IS_UNDEF);
|
|
ZEND_ASSERT(p->key);
|
|
t = zend_hash_find_ex(target, p->key, 1);
|
|
if (UNEXPECTED(t != NULL)) {
|
|
if (EXPECTED(ZSTR_LEN(p->key) > 0) && EXPECTED(ZSTR_VAL(p->key)[0] == 0)) {
|
|
/* Mangled key - ignore and wait for runtime */
|
|
continue;
|
|
} else if (UNEXPECTED(!ZCG(accel_directives).ignore_dups)) {
|
|
zend_class_entry *ce1 = Z_PTR(p->val);
|
|
if (!(ce1->ce_flags & ZEND_ACC_ANON_CLASS)) {
|
|
CG(in_compilation) = 1;
|
|
zend_set_compiled_filename(ce1->info.user.filename);
|
|
CG(zend_lineno) = ce1->info.user.line_start;
|
|
zend_error(E_ERROR,
|
|
"Cannot declare %s %s, because the name is already in use",
|
|
zend_get_object_type(ce1), ZSTR_VAL(ce1->name));
|
|
return;
|
|
}
|
|
continue;
|
|
}
|
|
} else {
|
|
t = _zend_hash_append_ptr_ex(target, p->key, Z_PTR(p->val), 1);
|
|
zend_class_copy_ctor((zend_class_entry**)&Z_PTR_P(t));
|
|
}
|
|
}
|
|
target->nInternalPointer = 0;
|
|
return;
|
|
}
|
|
|
|
#if defined(__AVX__)
|
|
# include <nmmintrin.h>
|
|
# if defined(__GNUC__) && defined(__i386__)
|
|
static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
|
|
{
|
|
size_t delta = (char*)dest - (char*)src;
|
|
|
|
__asm__ volatile (
|
|
".align 16\n\t"
|
|
".LL0%=:\n\t"
|
|
"prefetchnta 0x40(%1)\n\t"
|
|
"vmovaps (%1), %%ymm0\n\t"
|
|
"vmovaps 0x20(%1), %%ymm1\n\t"
|
|
"vmovaps %%ymm0, (%1,%2)\n\t"
|
|
"vmovaps %%ymm1, 0x20(%1,%2)\n\t"
|
|
"addl $0x40, %1\n\t"
|
|
"subl $0x40, %0\n\t"
|
|
"ja .LL0%="
|
|
: "+r"(size),
|
|
"+r"(src)
|
|
: "r"(delta)
|
|
: "cc", "memory", "%ymm0", "%ymm1");
|
|
}
|
|
# elif defined(__GNUC__) && defined(__x86_64__)
|
|
static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
|
|
{
|
|
size_t delta = (char*)dest - (char*)src;
|
|
|
|
__asm__ volatile (
|
|
".align 16\n\t"
|
|
".LL0%=:\n\t"
|
|
"prefetchnta 0x40(%1)\n\t"
|
|
"vmovaps (%1), %%ymm0\n\t"
|
|
"vmovaps 0x20(%1), %%ymm1\n\t"
|
|
"vmovaps %%ymm0, (%1,%2)\n\t"
|
|
"vmovaps %%ymm1, 0x20(%1,%2)\n\t"
|
|
"addq $0x40, %1\n\t"
|
|
"subq $0x40, %0\n\t"
|
|
"ja .LL0%="
|
|
: "+r"(size),
|
|
"+r"(src)
|
|
: "r"(delta)
|
|
: "cc", "memory", "%ymm0", "%ymm1");
|
|
}
|
|
# else
|
|
static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
|
|
{
|
|
__m256 *dqdest = (__m256*)dest;
|
|
const __m256 *dqsrc = (const __m256*)src;
|
|
const __m256 *end = (const __m256*)((const char*)src + size);
|
|
|
|
do {
|
|
#ifdef PHP_WIN32
|
|
_mm_prefetch((const char *)(dqsrc + 2), _MM_HINT_NTA);
|
|
#else
|
|
_mm_prefetch(dqsrc + 2, _MM_HINT_NTA);
|
|
#endif
|
|
|
|
__m256 ymm0 = _mm256_load_ps((const float *)(dqsrc + 0));
|
|
__m256 ymm1 = _mm256_load_ps((const float *)(dqsrc + 1));
|
|
dqsrc += 2;
|
|
_mm256_store_ps((float *)(dqdest + 0), ymm0);
|
|
_mm256_store_ps((float *)(dqdest + 1), ymm1);
|
|
dqdest += 2;
|
|
} while (dqsrc != end);
|
|
}
|
|
# endif
|
|
#elif defined(__SSE2__)
|
|
# include <emmintrin.h>
|
|
# if defined(__GNUC__) && defined(__i386__)
|
|
static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
|
|
{
|
|
size_t delta = (char*)dest - (char*)src;
|
|
|
|
__asm__ volatile (
|
|
".align 16\n\t"
|
|
".LL0%=:\n\t"
|
|
"prefetchnta 0x40(%1)\n\t"
|
|
"movdqa (%1), %%xmm0\n\t"
|
|
"movdqa 0x10(%1), %%xmm1\n\t"
|
|
"movdqa 0x20(%1), %%xmm2\n\t"
|
|
"movdqa 0x30(%1), %%xmm3\n\t"
|
|
"movdqa %%xmm0, (%1,%2)\n\t"
|
|
"movdqa %%xmm1, 0x10(%1,%2)\n\t"
|
|
"movdqa %%xmm2, 0x20(%1,%2)\n\t"
|
|
"movdqa %%xmm3, 0x30(%1,%2)\n\t"
|
|
"addl $0x40, %1\n\t"
|
|
"subl $0x40, %0\n\t"
|
|
"ja .LL0%="
|
|
: "+r"(size),
|
|
"+r"(src)
|
|
: "r"(delta)
|
|
: "cc", "memory", "%xmm0", "%xmm1", "%xmm1", "%xmm2");
|
|
}
|
|
# elif defined(__GNUC__) && defined(__x86_64__)
|
|
static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
|
|
{
|
|
size_t delta = (char*)dest - (char*)src;
|
|
|
|
__asm__ volatile (
|
|
".align 16\n\t"
|
|
".LL0%=:\n\t"
|
|
"prefetchnta 0x40(%1)\n\t"
|
|
"movdqa (%1), %%xmm0\n\t"
|
|
"movdqa 0x10(%1), %%xmm1\n\t"
|
|
"movdqa 0x20(%1), %%xmm2\n\t"
|
|
"movdqa 0x30(%1), %%xmm3\n\t"
|
|
"movdqa %%xmm0, (%1,%2)\n\t"
|
|
"movdqa %%xmm1, 0x10(%1,%2)\n\t"
|
|
"movdqa %%xmm2, 0x20(%1,%2)\n\t"
|
|
"movdqa %%xmm3, 0x30(%1,%2)\n\t"
|
|
"addq $0x40, %1\n\t"
|
|
"subq $0x40, %0\n\t"
|
|
"ja .LL0%="
|
|
: "+r"(size),
|
|
"+r"(src)
|
|
: "r"(delta)
|
|
: "cc", "memory", "%xmm0", "%xmm1", "%xmm1", "%xmm2");
|
|
}
|
|
# else
|
|
static zend_always_inline void fast_memcpy(void *dest, const void *src, size_t size)
|
|
{
|
|
__m128i *dqdest = (__m128i*)dest;
|
|
const __m128i *dqsrc = (const __m128i*)src;
|
|
const __m128i *end = (const __m128i*)((const char*)src + size);
|
|
|
|
do {
|
|
#ifdef PHP_WIN32
|
|
_mm_prefetch((const char *)(dqsrc + 4), _MM_HINT_NTA);
|
|
#else
|
|
_mm_prefetch(dqsrc + 4, _MM_HINT_NTA);
|
|
#endif
|
|
|
|
__m128i xmm0 = _mm_load_si128(dqsrc + 0);
|
|
__m128i xmm1 = _mm_load_si128(dqsrc + 1);
|
|
__m128i xmm2 = _mm_load_si128(dqsrc + 2);
|
|
__m128i xmm3 = _mm_load_si128(dqsrc + 3);
|
|
dqsrc += 4;
|
|
_mm_store_si128(dqdest + 0, xmm0);
|
|
_mm_store_si128(dqdest + 1, xmm1);
|
|
_mm_store_si128(dqdest + 2, xmm2);
|
|
_mm_store_si128(dqdest + 3, xmm3);
|
|
dqdest += 4;
|
|
} while (dqsrc != end);
|
|
}
|
|
# endif
|
|
#endif
|
|
|
|
zend_op_array* zend_accel_load_script(zend_persistent_script *persistent_script, int from_shared_memory)
|
|
{
|
|
zend_op_array *op_array;
|
|
|
|
op_array = (zend_op_array *) emalloc(sizeof(zend_op_array));
|
|
*op_array = persistent_script->script.main_op_array;
|
|
|
|
if (EXPECTED(from_shared_memory)) {
|
|
zend_hash_init(&ZCG(bind_hash), 10, NULL, NULL, 0);
|
|
|
|
ZCG(current_persistent_script) = persistent_script;
|
|
ZCG(arena_mem) = NULL;
|
|
if (EXPECTED(persistent_script->arena_size)) {
|
|
#if defined(__AVX__) || defined(__SSE2__)
|
|
/* Target address must be aligned to 64-byte boundary */
|
|
_mm_prefetch(persistent_script->arena_mem, _MM_HINT_NTA);
|
|
ZCG(arena_mem) = zend_arena_alloc(&CG(arena), persistent_script->arena_size + 64);
|
|
ZCG(arena_mem) = (void*)(((zend_uintptr_t)ZCG(arena_mem) + 63L) & ~63L);
|
|
fast_memcpy(ZCG(arena_mem), persistent_script->arena_mem, persistent_script->arena_size);
|
|
#else
|
|
ZCG(arena_mem) = zend_arena_alloc(&CG(arena), persistent_script->arena_size);
|
|
memcpy(ZCG(arena_mem), persistent_script->arena_mem, persistent_script->arena_size);
|
|
#endif
|
|
}
|
|
|
|
/* Copy all the necessary stuff from shared memory to regular memory, and protect the shared script */
|
|
if (zend_hash_num_elements(&persistent_script->script.class_table) > 0) {
|
|
zend_accel_class_hash_copy_from_shm(CG(class_table), &persistent_script->script.class_table);
|
|
}
|
|
/* we must first to copy all classes and then prepare functions, since functions may try to bind
|
|
classes - which depend on pre-bind class entries existent in the class table */
|
|
if (zend_hash_num_elements(&persistent_script->script.function_table) > 0) {
|
|
zend_accel_function_hash_copy_from_shm(CG(function_table), &persistent_script->script.function_table);
|
|
}
|
|
|
|
/* Register __COMPILER_HALT_OFFSET__ constant */
|
|
if (persistent_script->compiler_halt_offset != 0 &&
|
|
persistent_script->script.filename) {
|
|
zend_string *name;
|
|
static const char haltoff[] = "__COMPILER_HALT_OFFSET__";
|
|
|
|
name = zend_mangle_property_name(haltoff, sizeof(haltoff) - 1, ZSTR_VAL(persistent_script->script.filename), ZSTR_LEN(persistent_script->script.filename), 0);
|
|
if (!zend_hash_exists(EG(zend_constants), name)) {
|
|
zend_register_long_constant(ZSTR_VAL(name), ZSTR_LEN(name), persistent_script->compiler_halt_offset, CONST_CS, 0);
|
|
}
|
|
zend_string_release_ex(name, 0);
|
|
}
|
|
|
|
zend_hash_destroy(&ZCG(bind_hash));
|
|
ZCG(current_persistent_script) = NULL;
|
|
} else /* if (!from_shared_memory) */ {
|
|
if (zend_hash_num_elements(&persistent_script->script.function_table) > 0) {
|
|
zend_accel_function_hash_copy(CG(function_table), &persistent_script->script.function_table);
|
|
}
|
|
if (zend_hash_num_elements(&persistent_script->script.class_table) > 0) {
|
|
zend_accel_class_hash_copy(CG(class_table), &persistent_script->script.class_table);
|
|
}
|
|
}
|
|
|
|
if (persistent_script->script.first_early_binding_opline != (uint32_t)-1) {
|
|
zend_string *orig_compiled_filename = CG(compiled_filename);
|
|
CG(compiled_filename) = persistent_script->script.filename;
|
|
zend_do_delayed_early_binding(op_array, persistent_script->script.first_early_binding_opline);
|
|
CG(compiled_filename) = orig_compiled_filename;
|
|
}
|
|
|
|
if (UNEXPECTED(!from_shared_memory)) {
|
|
free_persistent_script(persistent_script, 0); /* free only hashes */
|
|
}
|
|
|
|
return op_array;
|
|
}
|
|
|
|
/*
|
|
* zend_adler32() is based on zlib implementation
|
|
* Computes the Adler-32 checksum of a data stream
|
|
*
|
|
* Copyright (C) 1995-2005 Mark Adler
|
|
* For conditions of distribution and use, see copyright notice in zlib.h
|
|
*
|
|
* Copyright (C) 1995-2005 Jean-loup Gailly and Mark Adler
|
|
*
|
|
* This software is provided 'as-is', without any express or implied
|
|
* warranty. In no event will the authors be held liable for any damages
|
|
* arising from the use of this software.
|
|
*
|
|
* Permission is granted to anyone to use this software for any purpose,
|
|
* including commercial applications, and to alter it and redistribute it
|
|
* freely, subject to the following restrictions:
|
|
*
|
|
* 1. The origin of this software must not be misrepresented; you must not
|
|
* claim that you wrote the original software. If you use this software
|
|
* in a product, an acknowledgment in the product documentation would be
|
|
* appreciated but is not required.
|
|
* 2. Altered source versions must be plainly marked as such, and must not be
|
|
* misrepresented as being the original software.
|
|
* 3. This notice may not be removed or altered from any source distribution.
|
|
*
|
|
*/
|
|
|
|
#define ADLER32_BASE 65521 /* largest prime smaller than 65536 */
|
|
#define ADLER32_NMAX 5552
|
|
/* NMAX is the largest n such that 255n(n+1)/2 + (n+1)(BASE-1) <= 2^32-1 */
|
|
|
|
#define ADLER32_DO1(buf) {s1 += *(buf); s2 += s1;}
|
|
#define ADLER32_DO2(buf, i) ADLER32_DO1(buf + i); ADLER32_DO1(buf + i + 1);
|
|
#define ADLER32_DO4(buf, i) ADLER32_DO2(buf, i); ADLER32_DO2(buf, i + 2);
|
|
#define ADLER32_DO8(buf, i) ADLER32_DO4(buf, i); ADLER32_DO4(buf, i + 4);
|
|
#define ADLER32_DO16(buf) ADLER32_DO8(buf, 0); ADLER32_DO8(buf, 8);
|
|
|
|
unsigned int zend_adler32(unsigned int checksum, signed char *buf, uint32_t len)
|
|
{
|
|
unsigned int s1 = checksum & 0xffff;
|
|
unsigned int s2 = (checksum >> 16) & 0xffff;
|
|
signed char *end;
|
|
|
|
while (len >= ADLER32_NMAX) {
|
|
len -= ADLER32_NMAX;
|
|
end = buf + ADLER32_NMAX;
|
|
do {
|
|
ADLER32_DO16(buf);
|
|
buf += 16;
|
|
} while (buf != end);
|
|
s1 %= ADLER32_BASE;
|
|
s2 %= ADLER32_BASE;
|
|
}
|
|
|
|
if (len) {
|
|
if (len >= 16) {
|
|
end = buf + (len & 0xfff0);
|
|
len &= 0xf;
|
|
do {
|
|
ADLER32_DO16(buf);
|
|
buf += 16;
|
|
} while (buf != end);
|
|
}
|
|
if (len) {
|
|
end = buf + len;
|
|
do {
|
|
ADLER32_DO1(buf);
|
|
buf++;
|
|
} while (buf != end);
|
|
}
|
|
s1 %= ADLER32_BASE;
|
|
s2 %= ADLER32_BASE;
|
|
}
|
|
|
|
return (s2 << 16) | s1;
|
|
}
|
|
|
|
unsigned int zend_accel_script_checksum(zend_persistent_script *persistent_script)
|
|
{
|
|
signed char *mem = (signed char*)persistent_script->mem;
|
|
size_t size = persistent_script->size;
|
|
size_t persistent_script_check_block_size = ((char *)&(persistent_script->dynamic_members)) - (char *)persistent_script;
|
|
unsigned int checksum = ADLER32_INIT;
|
|
|
|
if (mem < (signed char*)persistent_script) {
|
|
checksum = zend_adler32(checksum, mem, (signed char*)persistent_script - mem);
|
|
size -= (signed char*)persistent_script - mem;
|
|
mem += (signed char*)persistent_script - mem;
|
|
}
|
|
|
|
zend_adler32(checksum, mem, persistent_script_check_block_size);
|
|
mem += sizeof(*persistent_script);
|
|
size -= sizeof(*persistent_script);
|
|
|
|
if (size > 0) {
|
|
checksum = zend_adler32(checksum, mem, size);
|
|
}
|
|
return checksum;
|
|
}
|