2013-02-13 20:26:47 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
2013-03-19 14:32:24 +08:00
|
|
|
| Zend OPcache |
|
2013-02-13 20:26:47 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2019-02-05 15:33:28 +08:00
|
|
|
| Copyright (c) The PHP Group |
|
2013-02-13 20:26:47 +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. |
|
|
|
|
+----------------------------------------------------------------------+
|
2018-11-02 00:30:28 +08:00
|
|
|
| Authors: Andi Gutmans <andi@php.net> |
|
|
|
|
| Zeev Suraski <zeev@php.net> |
|
2013-02-13 20:26:47 +08:00
|
|
|
| Stanislav Malyshev <stas@zend.com> |
|
2018-11-02 00:30:28 +08:00
|
|
|
| Dmitry Stogov <dmitry@php.net> |
|
2013-02-13 20:26:47 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ZendAccelerator.h"
|
|
|
|
#include "zend_accelerator_hash.h"
|
|
|
|
#include "zend_hash.h"
|
|
|
|
#include "zend_shared_alloc.h"
|
|
|
|
|
|
|
|
/* Generated on an Octa-ALPHA 300MHz CPU & 2.5GB RAM monster */
|
2016-11-26 22:18:42 +08:00
|
|
|
static uint32_t prime_numbers[] =
|
2013-02-13 20:26:47 +08:00
|
|
|
{5, 11, 19, 53, 107, 223, 463, 983, 1979, 3907, 7963, 16229, 32531, 65407, 130987, 262237, 524521, 1048793 };
|
2016-11-26 22:18:42 +08:00
|
|
|
static uint32_t num_prime_numbers = sizeof(prime_numbers) / sizeof(uint32_t);
|
2013-02-13 20:26:47 +08:00
|
|
|
|
|
|
|
void zend_accel_hash_clean(zend_accel_hash *accel_hash)
|
|
|
|
{
|
|
|
|
accel_hash->num_entries = 0;
|
|
|
|
accel_hash->num_direct_entries = 0;
|
|
|
|
memset(accel_hash->hash_table, 0, sizeof(zend_accel_hash_entry *)*accel_hash->max_num_entries);
|
|
|
|
}
|
|
|
|
|
2014-08-26 01:28:33 +08:00
|
|
|
void zend_accel_hash_init(zend_accel_hash *accel_hash, uint32_t hash_size)
|
2013-02-13 20:26:47 +08:00
|
|
|
{
|
2016-11-26 22:18:42 +08:00
|
|
|
uint32_t i;
|
2013-02-13 20:26:47 +08:00
|
|
|
|
|
|
|
for (i=0; i<num_prime_numbers; i++) {
|
|
|
|
if (hash_size <= prime_numbers[i]) {
|
|
|
|
hash_size = prime_numbers[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
accel_hash->num_entries = 0;
|
|
|
|
accel_hash->num_direct_entries = 0;
|
|
|
|
accel_hash->max_num_entries = hash_size;
|
|
|
|
|
|
|
|
/* set up hash pointers table */
|
|
|
|
accel_hash->hash_table = zend_shared_alloc(sizeof(zend_accel_hash_entry *)*accel_hash->max_num_entries);
|
|
|
|
if (!accel_hash->hash_table) {
|
|
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Insufficient shared memory!");
|
2013-02-18 17:13:25 +08:00
|
|
|
return;
|
2013-02-13 20:26:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* set up hash values table */
|
|
|
|
accel_hash->hash_entries = zend_shared_alloc(sizeof(zend_accel_hash_entry)*accel_hash->max_num_entries);
|
|
|
|
if (!accel_hash->hash_entries) {
|
|
|
|
zend_accel_error(ACCEL_LOG_FATAL, "Insufficient shared memory!");
|
2013-02-18 17:13:25 +08:00
|
|
|
return;
|
2013-02-13 20:26:47 +08:00
|
|
|
}
|
|
|
|
memset(accel_hash->hash_table, 0, sizeof(zend_accel_hash_entry *)*accel_hash->max_num_entries);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns NULL if hash is full
|
|
|
|
* Returns pointer the actual hash entry on success
|
|
|
|
* key needs to be already allocated as it is not copied
|
|
|
|
*/
|
2017-10-19 17:32:42 +08:00
|
|
|
zend_accel_hash_entry* zend_accel_hash_update(zend_accel_hash *accel_hash, const char *key, uint32_t key_length, zend_bool indirect, void *data)
|
2013-02-13 20:26:47 +08:00
|
|
|
{
|
2014-08-26 01:24:55 +08:00
|
|
|
zend_ulong hash_value;
|
|
|
|
zend_ulong index;
|
2013-02-13 20:26:47 +08:00
|
|
|
zend_accel_hash_entry *entry;
|
|
|
|
zend_accel_hash_entry *indirect_bucket = NULL;
|
|
|
|
|
|
|
|
if (indirect) {
|
|
|
|
indirect_bucket = (zend_accel_hash_entry*)data;
|
|
|
|
while (indirect_bucket->indirect) {
|
|
|
|
indirect_bucket = (zend_accel_hash_entry*)indirect_bucket->data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
hash_value = zend_inline_hash_func(key, key_length);
|
2016-11-16 17:43:10 +08:00
|
|
|
#ifndef ZEND_WIN32
|
|
|
|
hash_value ^= ZCG(root_hash);
|
|
|
|
#endif
|
2013-02-13 20:26:47 +08:00
|
|
|
index = hash_value % accel_hash->max_num_entries;
|
|
|
|
|
|
|
|
/* try to see if the element already exists in the hash */
|
|
|
|
entry = accel_hash->hash_table[index];
|
|
|
|
while (entry) {
|
|
|
|
if (entry->hash_value == hash_value
|
|
|
|
&& entry->key_length == key_length
|
|
|
|
&& !memcmp(entry->key, key, key_length)) {
|
|
|
|
|
|
|
|
if (entry->indirect) {
|
|
|
|
if (indirect_bucket) {
|
|
|
|
entry->data = indirect_bucket;
|
|
|
|
} else {
|
|
|
|
((zend_accel_hash_entry*)entry->data)->data = data;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (indirect_bucket) {
|
|
|
|
accel_hash->num_direct_entries--;
|
|
|
|
entry->data = indirect_bucket;
|
|
|
|
entry->indirect = 1;
|
|
|
|
} else {
|
|
|
|
entry->data = data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
entry = entry->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Does not exist, add a new entry */
|
|
|
|
if (accel_hash->num_entries == accel_hash->max_num_entries) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
entry = &accel_hash->hash_entries[accel_hash->num_entries++];
|
|
|
|
if (indirect) {
|
|
|
|
entry->data = indirect_bucket;
|
|
|
|
entry->indirect = 1;
|
|
|
|
} else {
|
|
|
|
accel_hash->num_direct_entries++;
|
|
|
|
entry->data = data;
|
|
|
|
entry->indirect = 0;
|
|
|
|
}
|
|
|
|
entry->hash_value = hash_value;
|
|
|
|
entry->key = key;
|
|
|
|
entry->key_length = key_length;
|
|
|
|
entry->next = accel_hash->hash_table[index];
|
|
|
|
accel_hash->hash_table[index] = entry;
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2017-10-19 17:32:42 +08:00
|
|
|
static zend_always_inline void* zend_accel_hash_find_ex(zend_accel_hash *accel_hash, const char *key, uint32_t key_length, zend_ulong hash_value, int data)
|
2013-02-13 20:26:47 +08:00
|
|
|
{
|
|
|
|
zend_ulong index;
|
|
|
|
zend_accel_hash_entry *entry;
|
|
|
|
|
2016-11-16 17:43:10 +08:00
|
|
|
#ifndef ZEND_WIN32
|
|
|
|
hash_value ^= ZCG(root_hash);
|
|
|
|
#endif
|
2013-02-13 20:26:47 +08:00
|
|
|
index = hash_value % accel_hash->max_num_entries;
|
|
|
|
|
|
|
|
entry = accel_hash->hash_table[index];
|
|
|
|
while (entry) {
|
|
|
|
if (entry->hash_value == hash_value
|
|
|
|
&& entry->key_length == key_length
|
|
|
|
&& !memcmp(entry->key, key, key_length)) {
|
|
|
|
if (entry->indirect) {
|
2015-03-06 21:26:40 +08:00
|
|
|
if (data) {
|
|
|
|
return ((zend_accel_hash_entry*)entry->data)->data;
|
|
|
|
} else {
|
|
|
|
return entry->data;
|
|
|
|
}
|
2013-02-13 20:26:47 +08:00
|
|
|
} else {
|
2015-03-06 21:26:40 +08:00
|
|
|
if (data) {
|
|
|
|
return entry->data;
|
|
|
|
} else {
|
|
|
|
return entry;
|
|
|
|
}
|
2013-02-13 20:26:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
entry = entry->next;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-03-06 21:26:40 +08:00
|
|
|
/* Returns the data associated with key on success
|
|
|
|
* Returns NULL if data doesn't exist
|
|
|
|
*/
|
|
|
|
void* zend_accel_hash_find(zend_accel_hash *accel_hash, zend_string *key)
|
|
|
|
{
|
|
|
|
return zend_accel_hash_find_ex(
|
|
|
|
accel_hash,
|
2015-06-30 09:05:24 +08:00
|
|
|
ZSTR_VAL(key),
|
|
|
|
ZSTR_LEN(key),
|
2015-03-06 21:26:40 +08:00
|
|
|
zend_string_hash_val(key),
|
|
|
|
1);
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:26:47 +08:00
|
|
|
/* Returns the hash entry associated with key on success
|
|
|
|
* Returns NULL if it doesn't exist
|
|
|
|
*/
|
2015-03-06 21:26:40 +08:00
|
|
|
zend_accel_hash_entry* zend_accel_hash_find_entry(zend_accel_hash *accel_hash, zend_string *key)
|
2013-02-13 20:26:47 +08:00
|
|
|
{
|
2015-03-06 21:26:40 +08:00
|
|
|
return (zend_accel_hash_entry *)zend_accel_hash_find_ex(
|
|
|
|
accel_hash,
|
2015-06-30 09:05:24 +08:00
|
|
|
ZSTR_VAL(key),
|
|
|
|
ZSTR_LEN(key),
|
2015-03-06 21:26:40 +08:00
|
|
|
zend_string_hash_val(key),
|
|
|
|
0);
|
|
|
|
}
|
2013-02-13 20:26:47 +08:00
|
|
|
|
2015-03-06 21:26:40 +08:00
|
|
|
/* Returns the data associated with key on success
|
|
|
|
* Returns NULL if data doesn't exist
|
|
|
|
*/
|
2017-10-19 17:32:42 +08:00
|
|
|
void* zend_accel_hash_str_find(zend_accel_hash *accel_hash, const char *key, uint32_t key_length)
|
2015-03-06 21:26:40 +08:00
|
|
|
{
|
|
|
|
return zend_accel_hash_find_ex(
|
|
|
|
accel_hash,
|
|
|
|
key,
|
|
|
|
key_length,
|
|
|
|
zend_inline_hash_func(key, key_length),
|
|
|
|
1);
|
|
|
|
}
|
2013-02-13 20:26:47 +08:00
|
|
|
|
2015-03-06 21:26:40 +08:00
|
|
|
/* Returns the hash entry associated with key on success
|
|
|
|
* Returns NULL if it doesn't exist
|
|
|
|
*/
|
2017-10-19 17:32:42 +08:00
|
|
|
zend_accel_hash_entry* zend_accel_hash_str_find_entry(zend_accel_hash *accel_hash, const char *key, uint32_t key_length)
|
2015-03-06 21:26:40 +08:00
|
|
|
{
|
|
|
|
return (zend_accel_hash_entry *)zend_accel_hash_find_ex(
|
|
|
|
accel_hash,
|
|
|
|
key,
|
|
|
|
key_length,
|
|
|
|
zend_inline_hash_func(key, key_length),
|
|
|
|
0);
|
2013-02-13 20:26:47 +08:00
|
|
|
}
|
|
|
|
|
2017-10-19 17:32:42 +08:00
|
|
|
int zend_accel_hash_unlink(zend_accel_hash *accel_hash, const char *key, uint32_t key_length)
|
2013-02-13 20:26:47 +08:00
|
|
|
{
|
2014-08-26 01:24:55 +08:00
|
|
|
zend_ulong hash_value;
|
|
|
|
zend_ulong index;
|
2013-02-13 20:26:47 +08:00
|
|
|
zend_accel_hash_entry *entry, *last_entry=NULL;
|
|
|
|
|
|
|
|
hash_value = zend_inline_hash_func(key, key_length);
|
2016-11-16 17:43:10 +08:00
|
|
|
#ifndef ZEND_WIN32
|
|
|
|
hash_value ^= ZCG(root_hash);
|
|
|
|
#endif
|
2013-02-13 20:26:47 +08:00
|
|
|
index = hash_value % accel_hash->max_num_entries;
|
|
|
|
|
|
|
|
entry = accel_hash->hash_table[index];
|
|
|
|
while (entry) {
|
|
|
|
if (entry->hash_value == hash_value
|
|
|
|
&& entry->key_length == key_length
|
|
|
|
&& !memcmp(entry->key, key, key_length)) {
|
|
|
|
if (!entry->indirect) {
|
|
|
|
accel_hash->num_direct_entries--;
|
|
|
|
}
|
|
|
|
if (last_entry) {
|
|
|
|
last_entry->next = entry->next;
|
|
|
|
} else {
|
|
|
|
accel_hash->hash_table[index] = entry->next;
|
|
|
|
}
|
|
|
|
return SUCCESS;
|
|
|
|
}
|
|
|
|
last_entry = entry;
|
|
|
|
entry = entry->next;
|
|
|
|
}
|
|
|
|
return FAILURE;
|
|
|
|
}
|