diff options
Diffstat (limited to 'Zend/zend_API.c')
-rw-r--r-- | Zend/zend_API.c | 701 |
1 files changed, 701 insertions, 0 deletions
diff --git a/Zend/zend_API.c b/Zend/zend_API.c new file mode 100644 index 0000000000..6ab3072457 --- /dev/null +++ b/Zend/zend_API.c @@ -0,0 +1,701 @@ +/* + +----------------------------------------------------------------------+ + | Zend Engine | + +----------------------------------------------------------------------+ + | Copyright (c) 1998, 1999 Andi Gutmans, Zeev Suraski | + +----------------------------------------------------------------------+ + | This source file is subject to the Zend license, that is bundled | + | with this package in the file LICENSE. If you did not receive a | + | copy of the Zend license, please mail us at zend@zend.com so we can | + | send you a copy immediately. | + +----------------------------------------------------------------------+ + | Authors: Andi Gutmans <andi@zend.com> | + | Zeev Suraski <zeev@zend.com> | + +----------------------------------------------------------------------+ +*/ + +#include "zend.h" +#include "zend_variables.h" +#include "zend_execute.h" +#include "zend_API.h" +#include "modules.h" +#include "zend_constants.h" +#include "zend_operators.h" + +#if HAVE_STDARG_H +#include <stdarg.h> +#endif + +/* these variables are true statics/globals, and have to be mutex'ed on every access */ +static int module_count=0; +HashTable list_destructors, module_registry; + +/* this function doesn't check for too many parameters */ +int getParameters(HashTable *ht, int param_count,...) +{ + va_list ptr; + zval **param, **tmp = NULL, *param_ptr; + int i; + + va_start(ptr, param_count); + + for (i = 0; i < param_count; i++) { + param = va_arg(ptr, zval **); + if (zend_hash_index_find(ht, i, (void **) &tmp) == FAILURE) { + va_end(ptr); + return FAILURE; + } + param_ptr = *tmp; + if (!param_ptr->is_ref && param_ptr->refcount>1) { + zval *new_tmp; + + new_tmp = (zval *) emalloc(sizeof(zval)); + *new_tmp = *param_ptr; + zval_copy_ctor(new_tmp); + new_tmp->refcount = 1; + new_tmp->is_ref = 0; + param_ptr = new_tmp; + zend_hash_index_update(ht, i, ¶m_ptr, sizeof(zval *), NULL); + } + *param = param_ptr; + } + va_end(ptr); + return SUCCESS; +} + + +int getParametersArray(HashTable *ht, int param_count, zval **argument_array) +{ + int i; + zval **tmp = NULL, *param_ptr; + + for (i = 0; i < param_count; i++) { + if (zend_hash_index_find(ht, i, (void **) &tmp) == FAILURE) { + return FAILURE; + } + param_ptr = *tmp; + if (!param_ptr->is_ref && param_ptr->refcount>1) { + zval *new_tmp; + + new_tmp = (zval *) emalloc(sizeof(zval)); + *new_tmp = *param_ptr; + zval_copy_ctor(new_tmp); + new_tmp->refcount = 1; + new_tmp->is_ref = 0; + param_ptr = new_tmp; + zend_hash_index_update(ht, i, ¶m_ptr, sizeof(zval *), NULL); + } + argument_array[i] = param_ptr; + } + return SUCCESS; +} + + +int getThis(zval **this) +{ + /* NEEDS TO BE IMPLEMENTED FOR ZEND */ + /* + zval *data; + + if (zend_hash_find(GLOBAL(function_state.calling_symbol_table), "this", sizeof("this"), (void **)&data) == FAILURE) { + return FAILURE; + } + + *this = data; + */ + return SUCCESS; +} + +int ParameterPassedByReference(HashTable *ht, uint n) +{ + zval **tmp; + + if (zend_hash_index_find(ht, n-1, (void **) &tmp) == FAILURE) { + return 0; + } + return (*tmp)->is_ref; +} + + +ZEND_API void wrong_param_count() +{ + zend_error(E_WARNING,"Wrong parameter count for %s()",get_active_function_name()); +} + + +ZEND_API inline int array_init(zval *arg) +{ + arg->value.ht = (HashTable *) emalloc(sizeof(HashTable)); + if (!arg->value.ht || zend_hash_init(arg->value.ht, 0, NULL, PVAL_PTR_DTOR, 0)) { + zend_error(E_CORE_ERROR, "Cannot allocate memory for array"); + return FAILURE; + } + arg->type = IS_ARRAY; + return SUCCESS; +} + + +static void zval_update_const_and_ref(zval **p) +{ + zval_update_constant(*p); + zval_add_ref(p); +} + + + +ZEND_API inline int object_init_ex(zval *arg, zend_class_entry *class_type) +{ + zval *tmp; + + arg->value.obj.properties = (HashTable *) emalloc(sizeof(HashTable)); + zend_hash_init(arg->value.obj.properties, 0, NULL, PVAL_PTR_DTOR, 0); + zend_hash_copy(arg->value.obj.properties, &class_type->default_properties, (void (*)(void *)) zval_update_const_and_ref, (void *) &tmp, sizeof(zval *)); + arg->type = IS_OBJECT; + arg->value.obj.ce = class_type; + return SUCCESS; +} + + +ZEND_API inline int object_init(zval *arg) +{ + return object_init_ex(arg, &standard_class); +} + + +ZEND_API inline int add_assoc_long(zval *arg, char *key, long n) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_LONG; + tmp->value.lval = n; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.ht, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_assoc_double(zval *arg, char *key, double d) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_DOUBLE; + tmp->value.dval = d; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.ht, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_assoc_string(zval *arg, char *key, char *str, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = strlen(str); + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.ht, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_assoc_stringl(zval *arg, char *key, char *str, uint length, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = length; + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.ht, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_assoc_function(zval *arg, char *key,void (*function_ptr)(INTERNAL_FUNCTION_PARAMETERS)) +{ + zend_error(E_WARNING, "add_assoc_function() is no longer supported"); + return FAILURE; +} + + +ZEND_API inline int add_index_long(zval *arg, uint index, long n) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_LONG; + tmp->value.lval = n; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),NULL); +} + + +ZEND_API inline int add_index_double(zval *arg, uint index, double d) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_DOUBLE; + tmp->value.dval = d; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),NULL); +} + + +ZEND_API inline int add_index_string(zval *arg, uint index, char *str, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = strlen(str); + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),NULL); +} + + +ZEND_API inline int add_index_stringl(zval *arg, uint index, char *str, uint length, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = length; + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),NULL); +} + + +ZEND_API inline int add_next_index_long(zval *arg, long n) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_LONG; + tmp->value.lval = n; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_next_index_insert(arg->value.ht, &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_next_index_double(zval *arg, double d) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_DOUBLE; + tmp->value.dval = d; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_next_index_insert(arg->value.ht, &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_next_index_string(zval *arg, char *str, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = strlen(str); + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_next_index_insert(arg->value.ht, &tmp, sizeof(zval *),NULL); +} + + +ZEND_API inline int add_next_index_stringl(zval *arg, char *str, uint length, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = length; + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_next_index_insert(arg->value.ht, &tmp, sizeof(zval *),NULL); +} + + +ZEND_API inline int add_get_assoc_string(zval *arg, char *key, char *str, void **dest, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = strlen(str); + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.ht, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), dest); +} + + +ZEND_API inline int add_get_assoc_stringl(zval *arg, char *key, char *str, uint length, void **dest, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = length; + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.ht, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), dest); +} + + +ZEND_API inline int add_get_index_long(zval *arg, uint index, long l, void **dest) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_LONG; + tmp->value.lval= l; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),dest); +} + + +ZEND_API inline int add_get_index_double(zval *arg, uint index, double d, void **dest) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_DOUBLE; + tmp->value.dval= d; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),dest); +} + + +ZEND_API inline int add_get_index_string(zval *arg, uint index, char *str, void **dest, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = strlen(str); + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),dest); +} + + +ZEND_API inline int add_get_index_stringl(zval *arg, uint index, char *str, uint length, void **dest, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = length; + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_index_update(arg->value.ht, index, (void *) &tmp, sizeof(zval *),dest); +} + + +ZEND_API inline int add_property_long(zval *arg, char *key, long n) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_LONG; + tmp->value.lval = n; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.obj.properties, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_property_double(zval *arg, char *key, double d) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_DOUBLE; + tmp->value.dval = d; + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.obj.properties, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_property_string(zval *arg, char *key, char *str, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = strlen(str); + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.obj.properties, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +ZEND_API inline int add_property_stringl(zval *arg, char *key, char *str, uint length, int duplicate) +{ + zval *tmp = (zval *) emalloc(sizeof(zval)); + + tmp->type = IS_STRING; + tmp->value.str.len = length; + if (duplicate) { + tmp->value.str.val = estrndup(str,tmp->value.str.len); + } else { + tmp->value.str.val = str; + } + tmp->refcount=1; + tmp->is_ref=0; + return zend_hash_update(arg->value.obj.properties, key, strlen(key)+1, (void *) &tmp, sizeof(zval *), NULL); +} + + +int zend_startup_module(zend_module_entry *module) +{ + if (module) { + module->module_number = zend_next_free_module(); + if (module->module_startup_func) { + if (module->module_startup_func(MODULE_PERSISTENT, module->module_number)==FAILURE) { + zend_error(E_CORE_ERROR,"Unable to start %s module",module->name); + return FAILURE; + } + } + module->type = MODULE_PERSISTENT; + register_module(module); + } + return SUCCESS; +} + + +ZEND_API int _register_list_destructors(void (*list_destructor)(void *), void (*plist_destructor)(void *), int module_number) +{ + list_destructors_entry ld; + +#if 0 + printf("Registering destructors %d for module %d\n", list_destructors.nNextFreeElement, module_number); +#endif + + ld.list_destructor=(void (*)(void *)) list_destructor; + ld.plist_destructor=(void (*)(void *)) plist_destructor; + ld.module_number = module_number; + ld.resource_id = list_destructors.nNextFreeElement; + + if (zend_hash_next_index_insert(&list_destructors,(void *) &ld,sizeof(list_destructors_entry),NULL)==FAILURE) { + return FAILURE; + } + return list_destructors.nNextFreeElement-1; +} + + +/* registers all functions in *library_functions in the function hash */ +int register_functions(function_entry *functions) +{ + function_entry *ptr = functions; + zend_internal_function internal_function; + int count=0,unload=0; + CLS_FETCH(); + + internal_function.type = ZEND_INTERNAL_FUNCTION; + + while (ptr->fname) { + internal_function.handler = ptr->handler; + internal_function.arg_types = ptr->func_arg_types; + internal_function.function_name = ptr->fname; + if (!internal_function.handler) { + zend_error(E_CORE_WARNING,"Null function defined as active function"); + unregister_functions(functions,count); + return FAILURE; + } + if (zend_hash_add(CG(function_table), ptr->fname, strlen(ptr->fname)+1, &internal_function, sizeof(zend_internal_function), NULL) == FAILURE) { + unload=1; + break; + } + ptr++; + count++; + } + if (unload) { /* before unloading, display all remaining bad function in the module */ + while (ptr->fname) { + if (zend_hash_exists(CG(function_table), ptr->fname, strlen(ptr->fname)+1)) { + zend_error(E_CORE_WARNING,"Module load failed - duplicate function name - %s",ptr->fname); + } + ptr++; + } + unregister_functions(functions,count); + return FAILURE; + } + return SUCCESS; +} + +/* count=-1 means erase all functions, otherwise, + * erase the first count functions + */ +void unregister_functions(function_entry *functions,int count) +{ + function_entry *ptr = functions; + int i=0; + CLS_FETCH(); + + while (ptr->fname) { + if (count!=-1 && i>=count) { + break; + } +#if 0 + zend_printf("Unregistering %s()\n",ptr->fname); +#endif + zend_hash_del(CG(function_table),ptr->fname,strlen(ptr->fname)+1); + ptr++; + i++; + } +} + + +int register_module(zend_module_entry *module) +{ +#if 0 + zend_printf("%s: Registering module %d\n",module->name, module->module_number); +#endif + if (register_functions(module->functions)==FAILURE) { + zend_error(E_CORE_WARNING,"%s: Unable to register functions, unable to load",module->name); + return FAILURE; + } + module->module_started=1; + return zend_hash_add(&module_registry, module->name,strlen(module->name)+1,(void *)module,sizeof(zend_module_entry),NULL); +} + + +void module_destructor(zend_module_entry *module) +{ + if (module->type == MODULE_TEMPORARY) { + zend_hash_apply_with_argument(&list_destructors, (int (*)(void *,void *)) clean_module_resource_destructors, (void *) &(module->module_number)); + clean_module_constants(module->module_number); + } + + if (module->request_started && module->request_shutdown_func) { +#if 0 + zend_printf("%s: Request shutdown\n",module->name); +#endif + module->request_shutdown_func(); + } + module->request_started=0; + if (module->module_started && module->module_shutdown_func) { +#if 0 + zend_printf("%s: Module shutdown\n",module->name); +#endif + module->module_shutdown_func(); + } + module->module_started=0; + unregister_functions(module->functions,-1); + +#if HAVE_LIBDL + if (module->handle) { + dlclose(module->handle); + } +#endif +} + + +/* call request startup for all modules */ +int module_registry_request_startup(zend_module_entry *module) +{ + if (!module->request_started && module->request_startup_func) { +#if 0 + zend_printf("%s: Request startup\n",module->name); +#endif + module->request_startup_func(module->type, module->module_number); + } + module->request_started=1; + return 0; +} + + +/* for persistent modules - call request shutdown and flag NOT to erase + * for temporary modules - do nothing, and flag to erase + */ +int module_registry_cleanup(zend_module_entry *module) +{ + switch(module->type) { + case MODULE_PERSISTENT: + if (module->request_started && module->request_shutdown_func) { +#if 0 + zend_printf("%s: Request shutdown\n",module->name); +#endif + module->request_shutdown_func(); + } + module->request_started=0; + return 0; + break; + case MODULE_TEMPORARY: + return 1; + break; + } + return 0; +} + + +/* return the next free module number */ +int zend_next_free_module(void) +{ + return ++module_count; +} + + +zend_class_entry *register_internal_class(zend_class_entry *class_entry) +{ + zend_class_entry *register_class; + char *lowercase_name = zend_strndup(class_entry->name, class_entry->name_length); + CLS_FETCH(); + + zend_str_tolower(lowercase_name, class_entry->name_length); + + class_entry->type = ZEND_INTERNAL_CLASS; + class_entry->parent = NULL; + zend_hash_init(&class_entry->default_properties, 0, NULL, PVAL_PTR_DTOR, 1); + zend_hash_init(&class_entry->function_table, 0, NULL, (void (*)(void *)) destroy_zend_function, 1); + + zend_hash_update(CG(class_table), lowercase_name, class_entry->name_length+1, class_entry, sizeof(zend_class_entry), (void **) ®ister_class); + free(lowercase_name); + return register_class; +} |