summaryrefslogtreecommitdiff
path: root/Zend/zend_API.c
diff options
context:
space:
mode:
Diffstat (limited to 'Zend/zend_API.c')
-rw-r--r--Zend/zend_API.c701
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, &param_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, &param_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 **) &register_class);
+ free(lowercase_name);
+ return register_class;
+}