mirror of
https://github.com/php/php-src.git
synced 2024-12-11 19:04:38 +08:00
8f9f308bb9
* PHP-7.3: Fix wrong datatype
701 lines
17 KiB
C
701 lines
17 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| Zend Engine |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) Zend Technologies Ltd. (http://www.zend.com) |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 2.00 of the Zend 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.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. |
|
|
+----------------------------------------------------------------------+
|
|
| Author: Zeev Suraski <zeev@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include "zend.h"
|
|
#include "zend_sort.h"
|
|
#include "zend_API.h"
|
|
#include "zend_ini.h"
|
|
#include "zend_alloc.h"
|
|
#include "zend_operators.h"
|
|
#include "zend_strtod.h"
|
|
|
|
static HashTable *registered_zend_ini_directives;
|
|
|
|
#define NO_VALUE_PLAINTEXT "no value"
|
|
#define NO_VALUE_HTML "<i>no value</i>"
|
|
|
|
/*
|
|
* hash_apply functions
|
|
*/
|
|
static int zend_remove_ini_entries(zval *el, void *arg) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry = (zend_ini_entry *)Z_PTR_P(el);
|
|
int module_number = *(int *)arg;
|
|
|
|
return ini_entry->module_number == module_number;
|
|
}
|
|
/* }}} */
|
|
|
|
static int zend_restore_ini_entry_cb(zend_ini_entry *ini_entry, int stage) /* {{{ */
|
|
{
|
|
int result = FAILURE;
|
|
|
|
if (ini_entry->modified) {
|
|
if (ini_entry->on_modify) {
|
|
zend_try {
|
|
/* even if on_modify bails out, we have to continue on with restoring,
|
|
since there can be allocated variables that would be freed on MM shutdown
|
|
and would lead to memory corruption later ini entry is modified again */
|
|
result = ini_entry->on_modify(ini_entry, ini_entry->orig_value, ini_entry->mh_arg1, ini_entry->mh_arg2, ini_entry->mh_arg3, stage);
|
|
} zend_end_try();
|
|
}
|
|
if (stage == ZEND_INI_STAGE_RUNTIME && result == FAILURE) {
|
|
/* runtime failure is OK */
|
|
return 1;
|
|
}
|
|
if (ini_entry->value != ini_entry->orig_value) {
|
|
zend_string_release(ini_entry->value);
|
|
}
|
|
ini_entry->value = ini_entry->orig_value;
|
|
ini_entry->modifiable = ini_entry->orig_modifiable;
|
|
ini_entry->modified = 0;
|
|
ini_entry->orig_value = NULL;
|
|
ini_entry->orig_modifiable = 0;
|
|
}
|
|
return 0;
|
|
}
|
|
/* }}} */
|
|
|
|
static void free_ini_entry(zval *zv) /* {{{ */
|
|
{
|
|
zend_ini_entry *entry = (zend_ini_entry*)Z_PTR_P(zv);
|
|
|
|
zend_string_release_ex(entry->name, 1);
|
|
if (entry->value) {
|
|
zend_string_release(entry->value);
|
|
}
|
|
if (entry->orig_value) {
|
|
zend_string_release_ex(entry->orig_value, 1);
|
|
}
|
|
free(entry);
|
|
}
|
|
/* }}} */
|
|
|
|
/*
|
|
* Startup / shutdown
|
|
*/
|
|
ZEND_API int zend_ini_startup(void) /* {{{ */
|
|
{
|
|
registered_zend_ini_directives = (HashTable *) malloc(sizeof(HashTable));
|
|
|
|
EG(ini_directives) = registered_zend_ini_directives;
|
|
EG(modified_ini_directives) = NULL;
|
|
EG(error_reporting_ini_entry) = NULL;
|
|
zend_hash_init_ex(registered_zend_ini_directives, 128, NULL, free_ini_entry, 1, 0);
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_ini_shutdown(void) /* {{{ */
|
|
{
|
|
zend_ini_dtor(EG(ini_directives));
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API void zend_ini_dtor(HashTable *ini_directives) /* {{{ */
|
|
{
|
|
zend_hash_destroy(ini_directives);
|
|
free(ini_directives);
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_ini_global_shutdown(void) /* {{{ */
|
|
{
|
|
zend_hash_destroy(registered_zend_ini_directives);
|
|
free(registered_zend_ini_directives);
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_ini_deactivate(void) /* {{{ */
|
|
{
|
|
if (EG(modified_ini_directives)) {
|
|
zend_ini_entry *ini_entry;
|
|
|
|
ZEND_HASH_FOREACH_PTR(EG(modified_ini_directives), ini_entry) {
|
|
zend_restore_ini_entry_cb(ini_entry, ZEND_INI_STAGE_DEACTIVATE);
|
|
} ZEND_HASH_FOREACH_END();
|
|
zend_hash_destroy(EG(modified_ini_directives));
|
|
FREE_HASHTABLE(EG(modified_ini_directives));
|
|
EG(modified_ini_directives) = NULL;
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
#ifdef ZTS
|
|
static void copy_ini_entry(zval *zv) /* {{{ */
|
|
{
|
|
zend_ini_entry *old_entry = (zend_ini_entry*)Z_PTR_P(zv);
|
|
zend_ini_entry *new_entry = pemalloc(sizeof(zend_ini_entry), 1);
|
|
|
|
Z_PTR_P(zv) = new_entry;
|
|
memcpy(new_entry, old_entry, sizeof(zend_ini_entry));
|
|
if (old_entry->name) {
|
|
new_entry->name = zend_string_dup(old_entry->name, 1);
|
|
}
|
|
if (old_entry->value) {
|
|
new_entry->value = zend_string_dup(old_entry->value, 1);
|
|
}
|
|
if (old_entry->orig_value) {
|
|
new_entry->orig_value = zend_string_dup(old_entry->orig_value, 1);
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_copy_ini_directives(void) /* {{{ */
|
|
{
|
|
EG(modified_ini_directives) = NULL;
|
|
EG(error_reporting_ini_entry) = NULL;
|
|
EG(ini_directives) = (HashTable *) malloc(sizeof(HashTable));
|
|
zend_hash_init_ex(EG(ini_directives), registered_zend_ini_directives->nNumOfElements, NULL, free_ini_entry, 1, 0);
|
|
zend_hash_copy(EG(ini_directives), registered_zend_ini_directives, copy_ini_entry);
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
#endif
|
|
|
|
static int ini_key_compare(const void *a, const void *b) /* {{{ */
|
|
{
|
|
const Bucket *f;
|
|
const Bucket *s;
|
|
|
|
f = (const Bucket *) a;
|
|
s = (const Bucket *) b;
|
|
|
|
if (!f->key && !s->key) { /* both numeric */
|
|
if (f->h > s->h) {
|
|
return -1;
|
|
} else if (f->h < s->h) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
} else if (!f->key) { /* f is numeric, s is not */
|
|
return -1;
|
|
} else if (!s->key) { /* s is numeric, f is not */
|
|
return 1;
|
|
} else { /* both strings */
|
|
return zend_binary_strcasecmp(ZSTR_VAL(f->key), ZSTR_LEN(f->key), ZSTR_VAL(s->key), ZSTR_LEN(s->key));
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API void zend_ini_sort_entries(void) /* {{{ */
|
|
{
|
|
zend_hash_sort(EG(ini_directives), ini_key_compare, 0);
|
|
}
|
|
/* }}} */
|
|
|
|
/*
|
|
* Registration / unregistration
|
|
*/
|
|
ZEND_API int zend_register_ini_entries(const zend_ini_entry_def *ini_entry, int module_number) /* {{{ */
|
|
{
|
|
zend_ini_entry *p;
|
|
zval *default_value;
|
|
HashTable *directives = registered_zend_ini_directives;
|
|
|
|
#ifdef ZTS
|
|
/* if we are called during the request, eg: from dl(),
|
|
* then we should not touch the global directives table,
|
|
* and should update the per-(request|thread) version instead.
|
|
* This solves two problems: one is that ini entries for dl()'d
|
|
* extensions will now work, and the second is that updating the
|
|
* global hash here from dl() is not mutex protected and can
|
|
* lead to death.
|
|
*/
|
|
if (directives != EG(ini_directives)) {
|
|
directives = EG(ini_directives);
|
|
}
|
|
#endif
|
|
|
|
while (ini_entry->name) {
|
|
p = pemalloc(sizeof(zend_ini_entry), 1);
|
|
p->name = zend_string_init_interned(ini_entry->name, ini_entry->name_length, 1);
|
|
p->on_modify = ini_entry->on_modify;
|
|
p->mh_arg1 = ini_entry->mh_arg1;
|
|
p->mh_arg2 = ini_entry->mh_arg2;
|
|
p->mh_arg3 = ini_entry->mh_arg3;
|
|
p->value = NULL;
|
|
p->orig_value = NULL;
|
|
p->displayer = ini_entry->displayer;
|
|
p->modifiable = ini_entry->modifiable;
|
|
|
|
p->orig_modifiable = 0;
|
|
p->modified = 0;
|
|
p->module_number = module_number;
|
|
|
|
if (zend_hash_add_ptr(directives, p->name, (void*)p) == NULL) {
|
|
if (p->name) {
|
|
zend_string_release_ex(p->name, 1);
|
|
}
|
|
zend_unregister_ini_entries(module_number);
|
|
return FAILURE;
|
|
}
|
|
if (((default_value = zend_get_configuration_directive(p->name)) != NULL) &&
|
|
(!p->on_modify || p->on_modify(p, Z_STR_P(default_value), p->mh_arg1, p->mh_arg2, p->mh_arg3, ZEND_INI_STAGE_STARTUP) == SUCCESS)) {
|
|
|
|
p->value = zend_new_interned_string(zend_string_copy(Z_STR_P(default_value)));
|
|
} else {
|
|
p->value = ini_entry->value ?
|
|
zend_string_init_interned(ini_entry->value, ini_entry->value_length, 1) : NULL;
|
|
|
|
if (p->on_modify) {
|
|
p->on_modify(p, p->value, p->mh_arg1, p->mh_arg2, p->mh_arg3, ZEND_INI_STAGE_STARTUP);
|
|
}
|
|
}
|
|
ini_entry++;
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API void zend_unregister_ini_entries(int module_number) /* {{{ */
|
|
{
|
|
zend_hash_apply_with_argument(registered_zend_ini_directives, zend_remove_ini_entries, (void *) &module_number);
|
|
}
|
|
/* }}} */
|
|
|
|
#ifdef ZTS
|
|
ZEND_API void zend_ini_refresh_caches(int stage) /* {{{ */
|
|
{
|
|
zend_ini_entry *p;
|
|
|
|
ZEND_HASH_FOREACH_PTR(EG(ini_directives), p) {
|
|
if (p->on_modify) {
|
|
p->on_modify(p, p->value, p->mh_arg1, p->mh_arg2, p->mh_arg3, stage);
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
/* }}} */
|
|
#endif
|
|
|
|
ZEND_API int zend_alter_ini_entry(zend_string *name, zend_string *new_value, int modify_type, int stage) /* {{{ */
|
|
{
|
|
|
|
return zend_alter_ini_entry_ex(name, new_value, modify_type, stage, 0);
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_alter_ini_entry_chars(zend_string *name, const char *value, size_t value_length, int modify_type, int stage) /* {{{ */
|
|
{
|
|
int ret;
|
|
zend_string *new_value;
|
|
|
|
new_value = zend_string_init(value, value_length, !(stage & ZEND_INI_STAGE_IN_REQUEST));
|
|
ret = zend_alter_ini_entry_ex(name, new_value, modify_type, stage, 0);
|
|
zend_string_release(new_value);
|
|
return ret;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_alter_ini_entry_chars_ex(zend_string *name, const char *value, size_t value_length, int modify_type, int stage, int force_change) /* {{{ */
|
|
{
|
|
int ret;
|
|
zend_string *new_value;
|
|
|
|
new_value = zend_string_init(value, value_length, !(stage & ZEND_INI_STAGE_IN_REQUEST));
|
|
ret = zend_alter_ini_entry_ex(name, new_value, modify_type, stage, force_change);
|
|
zend_string_release(new_value);
|
|
return ret;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_alter_ini_entry_ex(zend_string *name, zend_string *new_value, int modify_type, int stage, int force_change) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
zend_string *duplicate;
|
|
uint8_t modifiable;
|
|
zend_bool modified;
|
|
|
|
if ((ini_entry = zend_hash_find_ptr(EG(ini_directives), name)) == NULL) {
|
|
return FAILURE;
|
|
}
|
|
|
|
modifiable = ini_entry->modifiable;
|
|
modified = ini_entry->modified;
|
|
|
|
if (stage == ZEND_INI_STAGE_ACTIVATE && modify_type == ZEND_INI_SYSTEM) {
|
|
ini_entry->modifiable = ZEND_INI_SYSTEM;
|
|
}
|
|
|
|
if (!force_change) {
|
|
if (!(ini_entry->modifiable & modify_type)) {
|
|
return FAILURE;
|
|
}
|
|
}
|
|
|
|
if (!EG(modified_ini_directives)) {
|
|
ALLOC_HASHTABLE(EG(modified_ini_directives));
|
|
zend_hash_init(EG(modified_ini_directives), 8, NULL, NULL, 0);
|
|
}
|
|
if (!modified) {
|
|
ini_entry->orig_value = ini_entry->value;
|
|
ini_entry->orig_modifiable = modifiable;
|
|
ini_entry->modified = 1;
|
|
zend_hash_add_ptr(EG(modified_ini_directives), ini_entry->name, ini_entry);
|
|
}
|
|
|
|
duplicate = zend_string_copy(new_value);
|
|
|
|
if (!ini_entry->on_modify
|
|
|| ini_entry->on_modify(ini_entry, duplicate, ini_entry->mh_arg1, ini_entry->mh_arg2, ini_entry->mh_arg3, stage) == SUCCESS) {
|
|
if (modified && ini_entry->orig_value != ini_entry->value) { /* we already changed the value, free the changed value */
|
|
zend_string_release(ini_entry->value);
|
|
}
|
|
ini_entry->value = duplicate;
|
|
} else {
|
|
zend_string_release(duplicate);
|
|
return FAILURE;
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_restore_ini_entry(zend_string *name, int stage) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
|
|
if ((ini_entry = zend_hash_find_ptr(EG(ini_directives), name)) == NULL ||
|
|
(stage == ZEND_INI_STAGE_RUNTIME && (ini_entry->modifiable & ZEND_INI_USER) == 0)) {
|
|
return FAILURE;
|
|
}
|
|
|
|
if (EG(modified_ini_directives)) {
|
|
if (zend_restore_ini_entry_cb(ini_entry, stage) == 0) {
|
|
zend_hash_del(EG(modified_ini_directives), name);
|
|
} else {
|
|
return FAILURE;
|
|
}
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API int zend_ini_register_displayer(char *name, uint32_t name_length, void (*displayer)(zend_ini_entry *ini_entry, int type)) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
|
|
ini_entry = zend_hash_str_find_ptr(registered_zend_ini_directives, name, name_length);
|
|
if (ini_entry == NULL) {
|
|
return FAILURE;
|
|
}
|
|
|
|
ini_entry->displayer = displayer;
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
/*
|
|
* Data retrieval
|
|
*/
|
|
|
|
ZEND_API zend_long zend_ini_long(char *name, size_t name_length, int orig) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
|
|
ini_entry = zend_hash_str_find_ptr(EG(ini_directives), name, name_length);
|
|
if (ini_entry) {
|
|
if (orig && ini_entry->modified) {
|
|
return (ini_entry->orig_value ? ZEND_STRTOL(ZSTR_VAL(ini_entry->orig_value), NULL, 0) : 0);
|
|
} else {
|
|
return (ini_entry->value ? ZEND_STRTOL(ZSTR_VAL(ini_entry->value), NULL, 0) : 0);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API double zend_ini_double(char *name, size_t name_length, int orig) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
|
|
ini_entry = zend_hash_str_find_ptr(EG(ini_directives), name, name_length);
|
|
if (ini_entry) {
|
|
if (orig && ini_entry->modified) {
|
|
return (double) (ini_entry->orig_value ? zend_strtod(ZSTR_VAL(ini_entry->orig_value), NULL) : 0.0);
|
|
} else {
|
|
return (double) (ini_entry->value ? zend_strtod(ZSTR_VAL(ini_entry->value), NULL) : 0.0);
|
|
}
|
|
}
|
|
|
|
return 0.0;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API char *zend_ini_string_ex(char *name, size_t name_length, int orig, zend_bool *exists) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
|
|
ini_entry = zend_hash_str_find_ptr(EG(ini_directives), name, name_length);
|
|
if (ini_entry) {
|
|
if (exists) {
|
|
*exists = 1;
|
|
}
|
|
|
|
if (orig && ini_entry->modified) {
|
|
return ini_entry->orig_value ? ZSTR_VAL(ini_entry->orig_value) : NULL;
|
|
} else {
|
|
return ini_entry->value ? ZSTR_VAL(ini_entry->value) : NULL;
|
|
}
|
|
} else {
|
|
if (exists) {
|
|
*exists = 0;
|
|
}
|
|
return NULL;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API char *zend_ini_string(char *name, size_t name_length, int orig) /* {{{ */
|
|
{
|
|
zend_bool exists = 1;
|
|
char *return_value;
|
|
|
|
return_value = zend_ini_string_ex(name, name_length, orig, &exists);
|
|
if (!exists) {
|
|
return NULL;
|
|
} else if (!return_value) {
|
|
return_value = "";
|
|
}
|
|
return return_value;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API zend_string *zend_ini_get_value(zend_string *name) /* {{{ */
|
|
{
|
|
zend_ini_entry *ini_entry;
|
|
|
|
ini_entry = zend_hash_find_ptr(EG(ini_directives), name);
|
|
if (ini_entry) {
|
|
return ini_entry->value ? ini_entry->value : ZSTR_EMPTY_ALLOC();
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API zend_bool zend_ini_parse_bool(zend_string *str)
|
|
{
|
|
if ((ZSTR_LEN(str) == 4 && strcasecmp(ZSTR_VAL(str), "true") == 0)
|
|
|| (ZSTR_LEN(str) == 3 && strcasecmp(ZSTR_VAL(str), "yes") == 0)
|
|
|| (ZSTR_LEN(str) == 2 && strcasecmp(ZSTR_VAL(str), "on") == 0)) {
|
|
return 1;
|
|
} else {
|
|
return atoi(ZSTR_VAL(str)) != 0;
|
|
}
|
|
}
|
|
|
|
ZEND_INI_DISP(zend_ini_boolean_displayer_cb) /* {{{ */
|
|
{
|
|
int value;
|
|
zend_string *tmp_value;
|
|
|
|
if (type == ZEND_INI_DISPLAY_ORIG && ini_entry->modified) {
|
|
tmp_value = (ini_entry->orig_value ? ini_entry->orig_value : NULL );
|
|
} else if (ini_entry->value) {
|
|
tmp_value = ini_entry->value;
|
|
} else {
|
|
tmp_value = NULL;
|
|
}
|
|
|
|
if (tmp_value) {
|
|
value = zend_ini_parse_bool(tmp_value);
|
|
} else {
|
|
value = 0;
|
|
}
|
|
|
|
if (value) {
|
|
ZEND_PUTS("On");
|
|
} else {
|
|
ZEND_PUTS("Off");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_INI_DISP(zend_ini_color_displayer_cb) /* {{{ */
|
|
{
|
|
char *value;
|
|
|
|
if (type == ZEND_INI_DISPLAY_ORIG && ini_entry->modified) {
|
|
value = ZSTR_VAL(ini_entry->orig_value);
|
|
} else if (ini_entry->value) {
|
|
value = ZSTR_VAL(ini_entry->value);
|
|
} else {
|
|
value = NULL;
|
|
}
|
|
if (value) {
|
|
if (zend_uv.html_errors) {
|
|
zend_printf("<font style=\"color: %s\">%s</font>", value, value);
|
|
} else {
|
|
ZEND_PUTS(value);
|
|
}
|
|
} else {
|
|
if (zend_uv.html_errors) {
|
|
ZEND_PUTS(NO_VALUE_HTML);
|
|
} else {
|
|
ZEND_PUTS(NO_VALUE_PLAINTEXT);
|
|
}
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_INI_DISP(display_link_numbers) /* {{{ */
|
|
{
|
|
char *value;
|
|
|
|
if (type == ZEND_INI_DISPLAY_ORIG && ini_entry->modified) {
|
|
value = ZSTR_VAL(ini_entry->orig_value);
|
|
} else if (ini_entry->value) {
|
|
value = ZSTR_VAL(ini_entry->value);
|
|
} else {
|
|
value = NULL;
|
|
}
|
|
|
|
if (value) {
|
|
if (atoi(value) == -1) {
|
|
ZEND_PUTS("Unlimited");
|
|
} else {
|
|
zend_printf("%s", value);
|
|
}
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* Standard message handlers */
|
|
ZEND_API ZEND_INI_MH(OnUpdateBool) /* {{{ */
|
|
{
|
|
zend_bool *p;
|
|
#ifndef ZTS
|
|
char *base = (char *) mh_arg2;
|
|
#else
|
|
char *base;
|
|
|
|
base = (char *) ts_resource(*((int *) mh_arg2));
|
|
#endif
|
|
|
|
p = (zend_bool *) (base+(size_t) mh_arg1);
|
|
|
|
*p = zend_ini_parse_bool(new_value);
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API ZEND_INI_MH(OnUpdateLong) /* {{{ */
|
|
{
|
|
zend_long *p;
|
|
#ifndef ZTS
|
|
char *base = (char *) mh_arg2;
|
|
#else
|
|
char *base;
|
|
|
|
base = (char *) ts_resource(*((int *) mh_arg2));
|
|
#endif
|
|
|
|
p = (zend_long *) (base+(size_t) mh_arg1);
|
|
|
|
*p = zend_atol(ZSTR_VAL(new_value), ZSTR_LEN(new_value));
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API ZEND_INI_MH(OnUpdateLongGEZero) /* {{{ */
|
|
{
|
|
zend_long *p, tmp;
|
|
#ifndef ZTS
|
|
char *base = (char *) mh_arg2;
|
|
#else
|
|
char *base;
|
|
|
|
base = (char *) ts_resource(*((int *) mh_arg2));
|
|
#endif
|
|
|
|
tmp = zend_atol(ZSTR_VAL(new_value), ZSTR_LEN(new_value));
|
|
if (tmp < 0) {
|
|
return FAILURE;
|
|
}
|
|
|
|
p = (zend_long *) (base+(size_t) mh_arg1);
|
|
*p = tmp;
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API ZEND_INI_MH(OnUpdateReal) /* {{{ */
|
|
{
|
|
double *p;
|
|
#ifndef ZTS
|
|
char *base = (char *) mh_arg2;
|
|
#else
|
|
char *base;
|
|
|
|
base = (char *) ts_resource(*((int *) mh_arg2));
|
|
#endif
|
|
|
|
p = (double *) (base+(size_t) mh_arg1);
|
|
|
|
*p = zend_strtod(ZSTR_VAL(new_value), NULL);
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API ZEND_INI_MH(OnUpdateString) /* {{{ */
|
|
{
|
|
char **p;
|
|
#ifndef ZTS
|
|
char *base = (char *) mh_arg2;
|
|
#else
|
|
char *base;
|
|
|
|
base = (char *) ts_resource(*((int *) mh_arg2));
|
|
#endif
|
|
|
|
p = (char **) (base+(size_t) mh_arg1);
|
|
|
|
*p = new_value ? ZSTR_VAL(new_value) : NULL;
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
ZEND_API ZEND_INI_MH(OnUpdateStringUnempty) /* {{{ */
|
|
{
|
|
char **p;
|
|
#ifndef ZTS
|
|
char *base = (char *) mh_arg2;
|
|
#else
|
|
char *base;
|
|
|
|
base = (char *) ts_resource(*((int *) mh_arg2));
|
|
#endif
|
|
|
|
if (new_value && !ZSTR_VAL(new_value)[0]) {
|
|
return FAILURE;
|
|
}
|
|
|
|
p = (char **) (base+(size_t) mh_arg1);
|
|
|
|
*p = new_value ? ZSTR_VAL(new_value) : NULL;
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|