summaryrefslogtreecommitdiff
path: root/ext/standard/basic_functions.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/standard/basic_functions.c')
-rw-r--r--ext/standard/basic_functions.c2153
1 files changed, 2153 insertions, 0 deletions
diff --git a/ext/standard/basic_functions.c b/ext/standard/basic_functions.c
new file mode 100644
index 0000000000..5b7b807024
--- /dev/null
+++ b/ext/standard/basic_functions.c
@@ -0,0 +1,2153 @@
+/*
+ +----------------------------------------------------------------------+
+ | PHP HTML Embedded Scripting Language Version 3.0 |
+ +----------------------------------------------------------------------+
+ | Copyright (c) 1997,1998 PHP Development Team (See Credits file) |
+ +----------------------------------------------------------------------+
+ | This program is free software; you can redistribute it and/or modify |
+ | it under the terms of one of the following licenses: |
+ | |
+ | A) the GNU General Public License as published by the Free Software |
+ | Foundation; either version 2 of the License, or (at your option) |
+ | any later version. |
+ | |
+ | B) the PHP License as published by the PHP Development Team and |
+ | included in the distribution in the file: LICENSE |
+ | |
+ | This program is distributed in the hope that it will be useful, |
+ | but WITHOUT ANY WARRANTY; without even the implied warranty of |
+ | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
+ | GNU General Public License for more details. |
+ | |
+ | You should have received a copy of both licenses referred to here. |
+ | If you did not, or have any questions about PHP licensing, please |
+ | contact core@php.net. |
+ +----------------------------------------------------------------------+
+ | Authors: Andi Gutmans <andi@zend.com> |
+ | Zeev Suraski <zeev@zend.com> |
+ +----------------------------------------------------------------------+
+ */
+
+
+#define HASH_OF(p) ((p)->type==IS_ARRAY ? (p)->value.ht : (((p)->type==IS_OBJECT ? (p)->value.obj.properties : NULL)))
+#ifdef THREAD_SAFE
+#include "tls.h"
+#endif
+#include "php.h"
+#include "php_ini.h"
+#include "internal_functions_registry.h"
+#include "php3_standard.h"
+#include "zend_operators.h"
+#include <stdarg.h>
+#include <stdlib.h>
+#include <math.h>
+#include <time.h>
+#include <stdio.h>
+#if HAVE_UNISTD_H
+#include <unistd.h>
+#endif
+#if HAVE_STRING_H
+#include <string.h>
+#else
+#include <strings.h>
+#endif
+#include "safe_mode.h"
+/*
+#include "functions/basic_functions.h"
+#include "functions/phpmath.h"
+#include "functions/php3_string.h"
+#include "functions/dns.h"
+#include "functions/md5.h"
+#include "functions/html.h"
+#include "functions/post.h"
+#include "functions/exec.h"
+#include "functions/info.h"
+#include "functions/url.h"
+#include "functions/datetime.h"
+#include "functions/fsock.h"
+#include "functions/image.h"
+#include "functions/php3_link.h"
+#include "functions/php3_filestat.h"
+#include "functions/microtime.h"
+#include "functions/pageinfo.h"
+#include "functions/uniqid.h"
+#include "functions/base64.h"
+#include "functions/php3_mail.h"
+#include "functions/php3_var.h"
+#include "functions/php3_iptc.h"
+#include "functions/quot_print.h"
+#include "functions/cyr_convert.h"
+*/
+#if WIN32|WINNT
+#include "win32/unistd.h"
+#endif
+#include "zend_globals.h"
+
+#include "php_globals.h"
+
+static unsigned char second_and_third_args_force_ref[] = { 3, BYREF_NONE, BYREF_FORCE, BYREF_FORCE };
+static unsigned char third_and_fourth_args_force_ref[] = { 4, BYREF_NONE, BYREF_NONE, BYREF_FORCE, BYREF_FORCE };
+static pval *user_compare_func_name;
+static HashTable *user_shutdown_function_names;
+
+/* some prototypes for local functions */
+void user_shutdown_function_dtor(pval *user_shutdown_function_name);
+int user_shutdown_function_executor(pval *user_shutdown_function_name);
+void php3_call_shutdown_functions(void);
+
+function_entry basic_functions[] = {
+ {"intval", int_value, NULL},
+ {"doubleval", double_value, NULL},
+ {"strval", string_value, NULL},
+ PHP_FE(define, NULL)
+ PHP_FE(defined, NULL)
+ {"short_tags", php3_toggle_short_open_tag, NULL},
+ {"sleep", php3_sleep, NULL},
+ {"usleep", php3_usleep, NULL},
+ {"ksort", php3_key_sort, first_arg_force_ref},
+ {"asort", php3_asort, first_arg_force_ref},
+ {"arsort", php3_arsort, first_arg_force_ref},
+ {"sort", php3_sort, first_arg_force_ref},
+ {"rsort", php3_rsort, first_arg_force_ref},
+ {"usort", php3_user_sort, first_arg_force_ref},
+ {"uasort", php3_auser_sort, first_arg_force_ref},
+ {"uksort", php3_user_key_sort, first_arg_force_ref},
+ {"array_walk", php3_array_walk, first_arg_force_ref},
+ {"sizeof", php3_count, first_arg_allow_ref},
+ {"count", php3_count, first_arg_allow_ref},
+ {"time", php3_time, NULL},
+ {"mktime", php3_mktime, NULL},
+ {"gmmktime", php3_gmmktime, NULL},
+#if HAVE_STRFTIME
+ {"strftime", php3_strftime, NULL},
+#endif
+ {"date", php3_date, NULL},
+ {"gmdate", php3_gmdate, NULL},
+ {"getdate", php3_getdate, NULL},
+ {"checkdate", php3_checkdate, NULL},
+ {"chr", php3_chr, NULL},
+ {"ord", php3_ord, NULL},
+ {"flush", php3_flush, NULL},
+ {"end", array_end, first_arg_force_ref},
+ {"prev", array_prev, first_arg_force_ref},
+ {"next", array_next, first_arg_force_ref},
+ {"reset", array_reset, first_arg_force_ref},
+ {"current", array_current, first_arg_force_ref},
+ {"key", array_current_key, first_arg_force_ref},
+ {"each", array_each, first_arg_force_ref},
+ {"gettype", php3_gettype, NULL},
+ {"settype", php3_settype, first_arg_force_ref},
+ {"min", php3_min, NULL},
+ {"max", php3_max, NULL},
+
+ {"addslashes", php3_addslashes, NULL},
+ {"chop", php3_chop, NULL},
+ {"str_replace", php3_str_replace, NULL},
+ {"chunk_split", php3_chunk_split, NULL},
+ {"trim", php3_trim, NULL},
+ {"ltrim", php3_ltrim, NULL},
+ {"rtrim", php3_chop, NULL},
+ {"pos", array_current, first_arg_force_ref},
+
+ {"fsockopen", php3_fsockopen, third_and_fourth_args_force_ref},
+ {"getimagesize", php3_getimagesize, NULL},
+ {"htmlspecialchars", php3_htmlspecialchars, NULL},
+ {"htmlentities", php3_htmlentities, NULL},
+ {"md5", php3_md5, NULL},
+
+ {"iptcparse", php3_iptcparse, NULL},
+ {"parse_url", php3_parse_url, NULL},
+
+ {"parse_str", php3_parsestr, NULL},
+ {"phpinfo", php3_info, NULL},
+ {"phpversion", php3_version, NULL},
+ {"strlen", php3_strlen, NULL},
+ {"strcmp", php3_strcmp, NULL},
+ {"strspn", php3_strspn, NULL},
+ {"strcspn", php3_strcspn, NULL},
+ {"strcasecmp", php3_strcasecmp, NULL},
+ {"strtok", php3_strtok, NULL},
+ {"strtoupper", php3_strtoupper, NULL},
+ {"strtolower", php3_strtolower, NULL},
+ {"strchr", php3_strstr, NULL},
+ {"strpos", php3_strpos, NULL},
+ {"strrpos", php3_strrpos, NULL},
+ {"strrev", php3_strrev, NULL},
+ {"hebrev", php3_hebrev, NULL},
+ {"hebrevc", php3_hebrev_with_conversion,NULL},
+ {"nl2br", php3_newline_to_br, NULL},
+ {"basename", php3_basename, NULL},
+ {"dirname", php3_dirname, NULL},
+ {"stripslashes", php3_stripslashes, NULL},
+ {"strstr", php3_strstr, NULL},
+ {"stristr", php3_stristr, NULL},
+ {"strrchr", php3_strrchr, NULL},
+ {"substr", php3_substr, NULL},
+ {"quotemeta", php3_quotemeta, NULL},
+ {"urlencode", php3_urlencode, NULL},
+ {"urldecode", php3_urldecode, NULL},
+ {"rawurlencode", php3_rawurlencode, NULL},
+ {"rawurldecode", php3_rawurldecode, NULL},
+ {"ucfirst", php3_ucfirst, NULL},
+ {"ucwords", php3_ucwords, NULL},
+ {"strtr", php3_strtr, NULL},
+ {"sprintf", php3_user_sprintf, NULL},
+ {"printf", php3_user_printf, NULL},
+ {"setlocale", php3_setlocale, NULL},
+
+ {"exec", php3_exec, second_and_third_args_force_ref},
+ {"system", php3_system, second_arg_force_ref},
+ {"escapeshellcmd", php3_escapeshellcmd, NULL},
+ {"passthru", php3_passthru, second_arg_force_ref},
+ PHP_FE(shell_exec, NULL)
+
+ {"soundex", soundex, NULL},
+
+ {"rand", php3_rand, NULL},
+ {"srand", php3_srand, NULL},
+ {"getrandmax", php3_getrandmax, NULL},
+ {"mt_rand", php3_mt_rand, NULL},
+ {"mt_srand", php3_mt_srand, NULL},
+ {"mt_getrandmax", php3_mt_getrandmax, NULL},
+ {"gethostbyaddr", php3_gethostbyaddr, NULL},
+ {"gethostbyname", php3_gethostbyname, NULL},
+ {"gethostbynamel", php3_gethostbynamel, NULL},
+#if !(WIN32|WINNT)||HAVE_BINDLIB
+ {"checkdnsrr", php3_checkdnsrr, NULL},
+ {"getmxrr", php3_getmxrr, second_and_third_args_force_ref},
+#endif
+ {"explode", php3_explode, NULL},
+ {"implode", php3_implode, NULL},
+ {"getenv", php3_getenv, NULL},
+ {"error_reporting", php3_error_reporting, NULL},
+ {"clearstatcache", php3_clearstatcache, NULL},
+
+ {"unlink", php3_unlink, NULL},
+
+ {"getmyuid", php3_getmyuid, NULL},
+ {"getmypid", php3_getmypid, NULL},
+ /*getmyiid is here for forward compatibility with 3.1
+ See pageinfo.c in 3.1 for more information*/
+ {"getmyiid", php3_getmypid, NULL},
+ {"getmyinode", php3_getmyinode, NULL},
+ {"getlastmod", php3_getlastmod, NULL},
+
+ {"base64_decode", php3_base64_decode, NULL},
+ {"base64_encode", php3_base64_encode, NULL},
+
+ {"abs", php3_abs, NULL},
+ {"ceil", php3_ceil, NULL},
+ {"floor", php3_floor, NULL},
+ {"round", php3_round, NULL},
+ {"sin", php3_sin, NULL},
+ {"cos", php3_cos, NULL},
+ {"tan", php3_tan, NULL},
+ {"asin", php3_asin, NULL},
+ {"acos", php3_acos, NULL},
+ {"atan", php3_atan, NULL},
+ {"atan2", php3_atan2, NULL},
+ {"pi", php3_pi, NULL},
+ {"pow", php3_pow, NULL},
+ {"exp", php3_exp, NULL},
+ {"log", php3_log, NULL},
+ {"log10", php3_log10, NULL},
+ {"sqrt", php3_sqrt, NULL},
+ {"deg2rad", php3_deg2rad, NULL},
+ {"rad2deg", php3_rad2deg, NULL},
+ {"bindec", php3_bindec, NULL},
+ {"hexdec", php3_hexdec, NULL},
+ {"octdec", php3_octdec, NULL},
+ {"decbin", php3_decbin, NULL},
+ {"decoct", php3_decoct, NULL},
+ {"dechex", php3_dechex, NULL},
+ {"base_convert",php3_base_convert, NULL},
+ {"number_format", php3_number_format, NULL},
+
+#if HAVE_PUTENV
+ {"putenv", php3_putenv, NULL},
+#endif
+ {"microtime", php3_microtime, NULL},
+ {"uniqid", php3_uniqid, NULL},
+ {"linkinfo", php3_linkinfo, NULL},
+ {"readlink", php3_readlink, NULL},
+ {"symlink", php3_symlink, NULL},
+ {"link", php3_link, NULL},
+ {"quoted_printable_decode", php3_quoted_printable_decode, NULL},
+ {"convert_cyr_string", php3_convert_cyr_string, NULL},
+ {"get_current_user", php3_get_current_user, NULL},
+ {"set_time_limit", php3_set_time_limit, NULL},
+
+ {"get_cfg_var", php3_get_cfg_var, NULL},
+ {"magic_quotes_runtime", php3_set_magic_quotes_runtime, NULL},
+ {"set_magic_quotes_runtime", php3_set_magic_quotes_runtime, NULL},
+ {"get_magic_quotes_gpc", php3_get_magic_quotes_gpc, NULL},
+ {"get_magic_quotes_runtime", php3_get_magic_quotes_runtime, NULL},
+
+ {"is_long", php3_is_long, first_arg_allow_ref},
+ {"is_int", php3_is_long, first_arg_allow_ref},
+ {"is_integer", php3_is_long, first_arg_allow_ref},
+ {"is_float", php3_is_double, first_arg_allow_ref},
+ {"is_double", php3_is_double, first_arg_allow_ref},
+ {"is_real", php3_is_double, first_arg_allow_ref},
+ {"is_string", php3_is_string, first_arg_allow_ref},
+ {"is_array", php3_is_array, first_arg_allow_ref},
+ {"is_object", php3_is_object, first_arg_allow_ref},
+
+ {"leak", php3_leak, NULL},
+ {"error_log", php3_error_log, NULL},
+ {"call_user_func", php3_call_user_func, NULL},
+ {"call_user_method", php3_call_user_method, NULL},
+
+ PHP_FE(var_dump, NULL)
+ PHP_FE(serialize, first_arg_allow_ref)
+ PHP_FE(unserialize, first_arg_allow_ref)
+
+ PHP_FE(register_shutdown_function, NULL)
+
+ PHP_FE(highlight_file, NULL)
+ PHP_NAMED_FE(show_source, php3_highlight_file, NULL)
+ PHP_FE(highlight_string, NULL)
+
+ PHP_FE(ob_start, NULL)
+ PHP_FE(ob_end_flush, NULL)
+ PHP_FE(ob_end_clean, NULL)
+ PHP_FE(ob_get_contents, NULL)
+
+ PHP_FE(ini_get, NULL)
+ PHP_FE(ini_alter, NULL)
+ PHP_FE(ini_restore, NULL)
+
+ PHP_FE(print_r, NULL)
+
+ {NULL, NULL, NULL}
+};
+
+
+PHP_INI_BEGIN()
+ PHP_INI_ENTRY("highlight.string", "#foobar", PHP_INI_ALL, NULL, NULL)
+ PHP_INI_ENTRY("test2", "testing", PHP_INI_SYSTEM, NULL, NULL)
+PHP_INI_END()
+
+
+php3_module_entry basic_functions_module = {
+ "Basic Functions", /* extension name */
+ basic_functions, /* function list */
+ php3_minit_basic, /* process startup */
+ php3_mshutdown_basic, /* process shutdown */
+ php3_rinit_basic, /* request startup */
+ php3_rshutdown_basic, /* request shutdown */
+ NULL, /* extension info */
+ STANDARD_MODULE_PROPERTIES
+};
+
+#if HAVE_PUTENV
+static HashTable putenv_ht;
+
+static void _php3_putenv_destructor(putenv_entry *pe)
+{
+ if (pe->previous_value) {
+ putenv(pe->previous_value);
+ } else {
+# if HAVE_UNSETENV
+ unsetenv(pe->key);
+# else
+ char **env;
+
+ for (env = environ; env != NULL && *env != NULL; env++) {
+ if (!strncmp(*env,pe->key,pe->key_len) && (*env)[pe->key_len]=='=') { /* found it */
+ *env = "";
+ break;
+ }
+ }
+# endif
+ }
+ efree(pe->putenv_string);
+ efree(pe->key);
+}
+#endif
+
+#ifndef M_PI
+#define M_PI 3.14159265358979323846
+#endif
+
+void test_class_startup();
+
+int php3_minit_basic(INIT_FUNC_ARGS)
+{
+ TLS_VARS;
+
+ REGISTER_DOUBLE_CONSTANT("M_PI", M_PI, CONST_CS | CONST_PERSISTENT);
+ test_class_startup();
+ REGISTER_INI_ENTRIES();
+ return SUCCESS;
+}
+
+
+int php3_mshutdown_basic(SHUTDOWN_FUNC_ARGS)
+{
+ UNREGISTER_INI_ENTRIES();
+ return SUCCESS;
+}
+
+
+int php3_rinit_basic(INIT_FUNC_ARGS)
+{
+ TLS_VARS;
+ GLOBAL(strtok_string) = NULL;
+#if HAVE_PUTENV
+ if (_php3_hash_init(&putenv_ht, 1, NULL, (void (*)(void *)) _php3_putenv_destructor, 0) == FAILURE) {
+ return FAILURE;
+ }
+#endif
+ user_compare_func_name=NULL;
+ user_shutdown_function_names=NULL;
+ return SUCCESS;
+}
+
+
+int php3_rshutdown_basic(SHUTDOWN_FUNC_ARGS)
+{
+ TLS_VARS;
+ STR_FREE(GLOBAL(strtok_string));
+#if HAVE_PUTENV
+ _php3_hash_destroy(&putenv_ht);
+#endif
+ return SUCCESS;
+}
+
+/********************
+ * System Functions *
+ ********************/
+
+void php3_getenv(INTERNAL_FUNCTION_PARAMETERS)
+{
+#if FHTTPD
+ int i;
+#endif
+ pval *str;
+ char *ptr;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &str) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_string(str);
+
+#if FHTTPD
+ ptr=NULL;
+ if (str->type == IS_STRING && req){
+ for(i=0;i<req->nlines;i++){
+ if (req->lines[i].paramc>1){
+ if (req->lines[i].params[0]){
+ if (!strcmp(req->lines[i].params[0],
+ str->value.str.val)){
+ ptr=req->lines[i].params[1];
+ i=req->nlines;
+ }
+ }
+ }
+ }
+ }
+ if (!ptr) ptr = getenv(str->value.str.val);
+ if (ptr
+#else
+
+ if (str->type == IS_STRING &&
+#if APACHE
+ ((ptr = (char *)table_get(GLOBAL(php3_rqst)->subprocess_env, str->value.str.val)) || (ptr = getenv(str->value.str.val)))
+#endif
+#if CGI_BINARY
+ (ptr = getenv(str->value.str.val))
+#endif
+
+#if USE_SAPI
+ (ptr = GLOBAL(sapi_rqst)->getenv(GLOBAL(sapi_rqst)->scid,str->value.str.val))
+#endif
+#endif
+ ) {
+ RETURN_STRING(ptr,1);
+ }
+ RETURN_FALSE;
+}
+
+
+#if HAVE_PUTENV
+void php3_putenv(INTERNAL_FUNCTION_PARAMETERS)
+{
+
+ pval *str;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &str) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_string(str);
+
+ if (str->value.str.val && *(str->value.str.val)) {
+ int ret;
+ char *p,**env;
+ putenv_entry pe;
+
+ pe.putenv_string = estrndup(str->value.str.val,str->value.str.len);
+ pe.key = str->value.str.val;
+ if ((p=strchr(pe.key,'='))) { /* nullify the '=' if there is one */
+ *p='\0';
+ }
+ pe.key_len = strlen(pe.key);
+ pe.key = estrndup(pe.key,pe.key_len);
+
+ _php3_hash_del(&putenv_ht,pe.key,pe.key_len+1);
+
+ /* find previous value */
+ pe.previous_value = NULL;
+ for (env = environ; env != NULL && *env != NULL; env++) {
+ if (!strncmp(*env,pe.key,pe.key_len) && (*env)[pe.key_len]=='=') { /* found it */
+ pe.previous_value = *env;
+ break;
+ }
+ }
+
+ if ((ret=putenv(pe.putenv_string))==0) { /* success */
+ _php3_hash_add(&putenv_ht,pe.key,pe.key_len+1,(void **) &pe,sizeof(putenv_entry),NULL);
+ RETURN_TRUE;
+ } else {
+ efree(pe.putenv_string);
+ efree(pe.key);
+ RETURN_FALSE;
+ }
+ }
+}
+#endif
+
+
+void php3_error_reporting(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *arg;
+ int old_error_reporting;
+ ELS_FETCH();
+
+ old_error_reporting = EG(error_reporting);
+ switch (ARG_COUNT(ht)) {
+ case 0:
+ break;
+ case 1:
+ if (getParameters(ht,1,&arg) == FAILURE) {
+ RETURN_FALSE;
+ }
+ convert_to_long(arg);
+ EG(error_reporting)=arg->value.lval;
+ break;
+ default:
+ WRONG_PARAM_COUNT;
+ break;
+ }
+
+ RETVAL_LONG(old_error_reporting);
+}
+
+void php3_toggle_short_open_tag(INTERNAL_FUNCTION_PARAMETERS)
+{
+ /* has to be implemented within Zend */
+#if 0
+ pval *value;
+ int ret;
+ TLS_VARS;
+
+ ret = php3_ini.short_open_tag;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht,1,&value) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_long(value);
+ php3_ini.short_open_tag = value->value.lval;
+ RETURN_LONG(ret);
+#endif
+}
+
+/*******************
+ * Basic Functions *
+ *******************/
+
+void int_value(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *num, *arg_base;
+ int base;
+
+ switch(ARG_COUNT(ht)) {
+ case 1:
+ if (getParameters(ht, 1, &num) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ base = 10;
+ break;
+ case 2:
+ if (getParameters(ht, 2, &num, &arg_base) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_long(arg_base);
+ base = arg_base->value.lval;
+ break;
+ default:
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_long_base(num, base);
+ *return_value = *num;
+}
+
+
+void double_value(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *num;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &num) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_double(num);
+ *return_value = *num;
+}
+
+
+void string_value(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *num;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &num) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_string(num);
+ *return_value = *num;
+ pval_copy_constructor(return_value);
+}
+
+static int array_key_compare(const void *a, const void *b)
+{
+ Bucket *first;
+ Bucket *second;
+ int min, r;
+
+ first = *((Bucket **) a);
+ second = *((Bucket **) b);
+
+ if (first->arKey == NULL && second->arKey == NULL) {
+ return (first->h - second->h);
+ } else if (first->arKey == NULL) {
+ return -1;
+ } else if (second->arKey == NULL) {
+ return 1;
+ }
+ min = MIN(first->nKeyLength, second->nKeyLength);
+ if ((r = memcmp(first->arKey, second->arKey, min)) == 0) {
+ return (first->nKeyLength - second->nKeyLength);
+ } else {
+ return r;
+ }
+}
+
+
+void php3_key_sort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in ksort() call");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to ksort()");
+ return;
+ }
+ if (_php3_hash_sort(target_hash, array_key_compare,0) == FAILURE) {
+ return;
+ }
+ RETURN_TRUE;
+}
+
+
+/* the current implementation of count() is a definite example of what
+ * user functions should NOT look like. It's a hack, until we get
+ * unset() to work right in 3.1
+ */
+void php3_count(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ if (array->type == IS_STRING && array->value.str.val==undefined_variable_string) {
+ RETURN_LONG(0);
+ } else {
+ RETURN_LONG(1);
+ }
+ }
+
+ RETURN_LONG(_php3_hash_num_elements(target_hash));
+}
+
+
+/* Numbers are always smaller than strings int this function as it
+ * anyway doesn't make much sense to compare two different data types.
+ * This keeps it consistant and simple.
+ */
+static int array_data_compare(const void *a, const void *b)
+{
+ Bucket *f;
+ Bucket *s;
+ pval *first;
+ pval *second;
+ double dfirst, dsecond;
+
+ f = *((Bucket **) a);
+ s = *((Bucket **) b);
+
+ first = *((pval **) f->pData);
+ second = *((pval **) s->pData);
+
+ if ((first->type == IS_LONG || first->type == IS_DOUBLE) &&
+ (second->type == IS_LONG || second->type == IS_DOUBLE)) {
+ if (first->type == IS_LONG) {
+ dfirst = (double) first->value.lval;
+ } else {
+ dfirst = first->value.dval;
+ }
+ if (second->type == IS_LONG) {
+ dsecond = (double) second->value.lval;
+ } else {
+ dsecond = second->value.dval;
+ }
+ if (dfirst < dsecond) {
+ return -1;
+ } else if (dfirst == dsecond) {
+ return 0;
+ } else {
+ return 1;
+ }
+ }
+ if ((first->type == IS_LONG || first->type == IS_DOUBLE) &&
+ second->type == IS_STRING) {
+ return -1;
+ } else if ((first->type == IS_STRING) &&
+ (second->type == IS_LONG || second->type == IS_DOUBLE)) {
+ return 1;
+ }
+ if (first->type == IS_STRING && second->type == IS_STRING) {
+ return strcmp(first->value.str.val, second->value.str.val);
+ }
+ return 0; /* Anything else is equal as it can't be compared */
+}
+
+static int array_reverse_data_compare(const void *a, const void *b)
+{
+ return array_data_compare(a,b)*-1;
+}
+
+void php3_asort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in asort() call");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to asort()");
+ return;
+ }
+ if (_php3_hash_sort(target_hash, array_data_compare,0) == FAILURE) {
+ return;
+ }
+ RETURN_TRUE;
+}
+
+void php3_arsort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in arsort() call");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to arsort()");
+ return;
+ }
+ if (_php3_hash_sort(target_hash, array_reverse_data_compare,0) == FAILURE) {
+ return;
+ }
+ RETURN_TRUE;
+}
+
+void php3_sort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in sort() call");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to sort()");
+ return;
+ }
+ if (_php3_hash_sort(target_hash, array_data_compare,1) == FAILURE) {
+ return;
+ }
+ RETURN_TRUE;
+}
+
+void php3_rsort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in rsort() call");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to rsort()");
+ return;
+ }
+ if (_php3_hash_sort(target_hash, array_reverse_data_compare,1) == FAILURE) {
+ return;
+ }
+ RETURN_TRUE;
+}
+
+
+static int array_user_compare(const void *a, const void *b)
+{
+ Bucket *f;
+ Bucket *s;
+ pval *args[2];
+ pval retval;
+ CLS_FETCH();
+
+ f = *((Bucket **) a);
+ s = *((Bucket **) b);
+
+ args[0] = (pval *) f->pData;
+ args[1] = (pval *) s->pData;
+
+ if (call_user_function(CG(function_table), NULL, user_compare_func_name, &retval, 2, args)==SUCCESS) {
+ convert_to_long(&retval);
+ return retval.value.lval;
+ } else {
+ return 0;
+ }
+}
+
+
+void php3_user_sort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ pval *old_compare_func;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ old_compare_func = user_compare_func_name;
+ if (ARG_COUNT(ht) != 2 || getParameters(ht, 2, &array, &user_compare_func_name) == FAILURE) {
+ user_compare_func_name = old_compare_func;
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in usort() call");
+ user_compare_func_name = old_compare_func;
+ return;
+ }
+ convert_to_string(user_compare_func_name);
+ if (_php3_hash_sort(target_hash, array_user_compare, 1) == FAILURE) {
+ user_compare_func_name = old_compare_func;
+ return;
+ }
+ user_compare_func_name = old_compare_func;
+ RETURN_TRUE;
+}
+
+void php3_auser_sort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ pval *old_compare_func;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ old_compare_func = user_compare_func_name;
+ if (ARG_COUNT(ht) != 2 || getParameters(ht, 2, &array, &user_compare_func_name) == FAILURE) {
+ user_compare_func_name = old_compare_func;
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in uasort() call");
+ user_compare_func_name = old_compare_func;
+ return;
+ }
+ convert_to_string(user_compare_func_name);
+ if (_php3_hash_sort(target_hash, array_user_compare, 0) == FAILURE) {
+ user_compare_func_name = old_compare_func;
+ return;
+ }
+ user_compare_func_name = old_compare_func;
+ RETURN_TRUE;
+}
+
+
+static int array_user_key_compare(const void *a, const void *b)
+{
+ Bucket *f;
+ Bucket *s;
+ pval key1, key2;
+ pval *args[2];
+ pval retval;
+ int status;
+ CLS_FETCH();
+
+ args[0] = &key1;
+ args[1] = &key2;
+
+ f = *((Bucket **) a);
+ s = *((Bucket **) b);
+
+ if (f->arKey) {
+ key1.value.str.val = estrndup(f->arKey, f->nKeyLength);
+ key1.value.str.len = f->nKeyLength;
+ key1.type = IS_STRING;
+ } else {
+ key1.value.lval = f->h;
+ key1.type = IS_LONG;
+ }
+ if (s->arKey) {
+ key2.value.str.val = estrndup(s->arKey, s->nKeyLength);
+ key2.value.str.len = s->nKeyLength;
+ key2.type = IS_STRING;
+ } else {
+ key2.value.lval = s->h;
+ key2.type = IS_LONG;
+ }
+
+ status = call_user_function(CG(function_table), NULL, user_compare_func_name, &retval, 2, args);
+
+ pval_destructor(&key1);
+ pval_destructor(&key2);
+
+ if (status==SUCCESS) {
+ convert_to_long(&retval);
+ return retval.value.lval;
+ } else {
+ return 0;
+ }
+}
+
+
+void php3_user_key_sort(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ pval *old_compare_func;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ old_compare_func = user_compare_func_name;
+ if (ARG_COUNT(ht) != 2 || getParameters(ht, 2, &array, &user_compare_func_name) == FAILURE) {
+ user_compare_func_name = old_compare_func;
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in uksort() call");
+ user_compare_func_name = old_compare_func;
+ return;
+ }
+ convert_to_string(user_compare_func_name);
+ if (_php3_hash_sort(target_hash, array_user_key_compare, 0) == FAILURE) {
+ user_compare_func_name = old_compare_func;
+ return;
+ }
+ user_compare_func_name = old_compare_func;
+ RETURN_TRUE;
+}
+
+
+void array_end(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array, **entry;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Variable passed to end() is not an array or object");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to end()");
+ }
+ _php3_hash_internal_pointer_end(target_hash);
+ if (_php3_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
+ RETURN_FALSE;
+ }
+ *return_value = **entry;
+ pval_copy_constructor(return_value);
+}
+
+
+void array_prev(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array, **entry;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Variable passed to prev() is not an array or object");
+ RETURN_FALSE;
+ }
+ _php3_hash_move_backwards(target_hash);
+ if (_php3_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
+ RETURN_FALSE;
+ }
+
+ *return_value = **entry;
+ pval_copy_constructor(return_value);
+}
+
+
+void array_next(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array, **entry;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Variable passed to next() is not an array or object");
+ RETURN_FALSE;
+ }
+ _php3_hash_move_forward(target_hash);
+ if (_php3_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
+ RETURN_FALSE;
+ }
+
+ *return_value = **entry;
+ pval_copy_constructor(return_value);
+}
+
+
+void array_each(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array,*entry,**entry_ptr, *tmp;
+ char *string_key;
+ ulong num_key;
+ pval **inserted_pointer;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING,"Variable passed to each() is not an array or object");
+ return;
+ }
+ if (_php3_hash_get_current_data(target_hash, (void **) &entry_ptr)==FAILURE) {
+ RETURN_FALSE;
+ }
+ array_init(return_value);
+ entry = *entry_ptr;
+
+ /* add value elements */
+ if (entry->is_ref) {
+ tmp = (pval *)emalloc(sizeof(pval));
+ *tmp = *entry;
+ pval_copy_constructor(tmp);
+ tmp->is_ref=0;
+ tmp->refcount=0;
+ entry=tmp;
+ }
+ _php3_hash_index_update(return_value->value.ht, 1, &entry, sizeof(pval *), NULL);
+ entry->refcount++;
+ _php3_hash_update_ptr(return_value->value.ht, "value", sizeof("value"), entry, sizeof(pval *), NULL);
+ entry->refcount++;
+
+ /* add the key elements */
+ switch (_php3_hash_get_current_key(target_hash, &string_key, &num_key)) {
+ case HASH_KEY_IS_STRING:
+ add_get_index_string(return_value,0,string_key,(void **) &inserted_pointer,0);
+ break;
+ case HASH_KEY_IS_LONG:
+ add_get_index_long(return_value,0,num_key, (void **) &inserted_pointer);
+ break;
+ }
+ _php3_hash_update(return_value->value.ht, "key", sizeof("key"), inserted_pointer, sizeof(pval *), NULL);
+ (*inserted_pointer)->refcount++;
+ _php3_hash_move_forward(target_hash);
+}
+
+
+void array_reset(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array, **entry;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Variable passed to reset() is not an array or object");
+ return;
+ }
+ _php3_hash_internal_pointer_reset(target_hash);
+ if (_php3_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
+ return;
+ }
+
+ *return_value = **entry;
+ pval_copy_constructor(return_value);
+ return_value->refcount=1;
+ return_value->is_ref=0;
+}
+
+void array_current(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array, **entry;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Variable passed to current() is not an array or object");
+ return;
+ }
+ if (_php3_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
+ return;
+ }
+ *return_value = **entry;
+ pval_copy_constructor(return_value);
+}
+
+
+void array_current_key(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *array;
+ char *string_key;
+ ulong num_key;
+ HashTable *target_hash;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &array) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Variable passed to key() is not an array or object");
+ return;
+ }
+ if (!ParameterPassedByReference(ht,1)) {
+ php3_error(E_WARNING, "Array not passed by reference in call to key()");
+ }
+ switch (_php3_hash_get_current_key(target_hash, &string_key, &num_key)) {
+ case HASH_KEY_IS_STRING:
+ return_value->value.str.val = string_key;
+ return_value->value.str.len = strlen(string_key);
+ return_value->type = IS_STRING;
+ break;
+ case HASH_KEY_IS_LONG:
+ return_value->type = IS_LONG;
+ return_value->value.lval = num_key;
+ break;
+ case HASH_KEY_NON_EXISTANT:
+ return;
+ }
+}
+
+#ifdef __cplusplus
+void php3_flush(HashTable *)
+#else
+void php3_flush(INTERNAL_FUNCTION_PARAMETERS)
+#endif
+{
+#if APACHE
+ TLS_VARS;
+# if MODULE_MAGIC_NUMBER > 19970110
+ rflush(GLOBAL(php3_rqst));
+# else
+ bflush(GLOBAL(php3_rqst)->connection->client);
+# endif
+#endif
+#if FHTTPD
+ /*FIXME -- what does it flush really? the whole response?*/
+#endif
+#if CGI_BINARY
+ fflush(stdout);
+#endif
+#if USE_SAPI
+ TLS_VARS;
+ GLOBAL(sapi_rqst)->flush(GLOBAL(sapi_rqst)->scid);
+#endif
+}
+
+
+void php3_sleep(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *num;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &num) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_long(num);
+ sleep(num->value.lval);
+}
+
+void php3_usleep(INTERNAL_FUNCTION_PARAMETERS)
+{
+#if HAVE_USLEEP
+ pval *num;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &num) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_long(num);
+ usleep(num->value.lval);
+#endif
+}
+
+void php3_gettype(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *arg;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 1 || getParameters(ht, 1, &arg) == FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ switch (arg->type) {
+ case IS_LONG:
+ RETVAL_STRING("integer",1);
+ break;
+ case IS_DOUBLE:
+ RETVAL_STRING("double",1);
+ break;
+ case IS_STRING:
+ RETVAL_STRING("string",1);
+ break;
+ case IS_ARRAY:
+ RETVAL_STRING("array",1);
+ break;
+ case IS_OBJECT: {
+ char *result;
+ int res_len;
+
+ res_len = sizeof("object of type ")-1 + arg->value.obj.ce->name_length;
+ result = (char *) emalloc(res_len+1);
+ sprintf(result, "object of type %s", arg->value.obj.ce->name);
+ RETVAL_STRINGL(result, res_len, 0);
+ }
+ break;
+ default:
+ RETVAL_STRING("unknown type",1);
+ }
+}
+
+
+void php3_settype(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *var, *type;
+ char *new_type;
+ TLS_VARS;
+
+ if (ARG_COUNT(ht) != 2 || getParameters(ht, 2, &var, &type) ==
+ FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_string(type);
+ new_type = type->value.str.val;
+
+ if (!strcasecmp(new_type, "integer")) {
+ convert_to_long(var);
+ } else if (!strcasecmp(new_type, "double")) {
+ convert_to_double(var);
+ } else if (!strcasecmp(new_type, "string")) {
+ convert_to_string(var);
+ } else if (!strcasecmp(new_type, "array")) {
+ convert_to_array(var);
+ } else if (!strcasecmp(new_type, "object")) {
+ convert_to_object(var);
+ } else {
+ php3_error(E_WARNING, "settype: invalid type");
+ RETURN_FALSE;
+ }
+ RETVAL_TRUE;
+}
+
+
+void php3_min(INTERNAL_FUNCTION_PARAMETERS)
+{
+ int argc=ARG_COUNT(ht);
+ pval **result;
+
+ if (argc<=0) {
+ php3_error(E_WARNING, "min: must be passed at least 1 value");
+ var_uninit(return_value);
+ return;
+ }
+ if (argc == 1) {
+ pval *arr;
+
+ if (getParameters(ht, 1, &arr) == FAILURE ||
+ arr->type != IS_ARRAY) {
+ WRONG_PARAM_COUNT;
+ }
+ if (_php3_hash_minmax(arr->value.ht, array_data_compare, 0, (void **) &result)==SUCCESS) {
+ *return_value = **result;
+ pval_copy_constructor(return_value);
+ } else {
+ php3_error(E_WARNING, "min: array must contain at least 1 element");
+ var_uninit(return_value);
+ }
+ } else {
+ if (_php3_hash_minmax(ht, array_data_compare, 0, (void **) &result)==SUCCESS) {
+ *return_value = **result;
+ pval_copy_constructor(return_value);
+ }
+ }
+}
+
+
+void php3_max(INTERNAL_FUNCTION_PARAMETERS)
+{
+ int argc=ARG_COUNT(ht);
+ pval **result;
+
+ if (argc<=0) {
+ php3_error(E_WARNING, "max: must be passed at least 1 value");
+ var_uninit(return_value);
+ return;
+ }
+ if (argc == 1) {
+ pval *arr;
+
+ if (getParameters(ht, 1, &arr) == FAILURE ||
+ arr->type != IS_ARRAY) {
+ WRONG_PARAM_COUNT;
+ }
+ if (_php3_hash_minmax(arr->value.ht, array_data_compare, 1, (void **) &result)==SUCCESS) {
+ *return_value = **result;
+ pval_copy_constructor(return_value);
+ } else {
+ php3_error(E_WARNING, "max: array must contain at least 1 element");
+ var_uninit(return_value);
+ }
+ } else {
+ if (_php3_hash_minmax(ht, array_data_compare, 1, (void **) &result)==SUCCESS) {
+ *return_value = **result;
+ pval_copy_constructor(return_value);
+ }
+ }
+}
+
+static pval *php3_array_walk_func_name;
+
+static int _php3_array_walk(const void *a)
+{
+ pval *args[1];
+ pval retval;
+ CLS_FETCH();
+
+ args[0] = (pval *)a;
+
+ call_user_function(CG(function_table), NULL, php3_array_walk_func_name, &retval, 1, args);
+ return 0;
+}
+
+void php3_array_walk(INTERNAL_FUNCTION_PARAMETERS) {
+ pval *array, *old_walk_func_name;
+ HashTable *target_hash;
+ TLS_VARS;
+
+ old_walk_func_name = php3_array_walk_func_name;
+ if (ARG_COUNT(ht) != 2 || getParameters(ht, 2, &array, &php3_array_walk_func_name) == FAILURE) {
+ php3_array_walk_func_name = old_walk_func_name;
+ WRONG_PARAM_COUNT;
+ }
+ target_hash = HASH_OF(array);
+ if (!target_hash) {
+ php3_error(E_WARNING, "Wrong datatype in array_walk() call");
+ php3_array_walk_func_name = old_walk_func_name;
+ return;
+ }
+ convert_to_string(php3_array_walk_func_name);
+ _php3_hash_apply(target_hash, (int (*)(void *))_php3_array_walk);
+ php3_array_walk_func_name = old_walk_func_name;
+ RETURN_TRUE;
+}
+
+#if 0
+void php3_max(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval **argv;
+ int argc, i;
+ unsigned short max_type = IS_LONG;
+ TLS_VARS;
+
+ argc = ARG_COUNT(ht);
+ /* if there is one parameter and this parameter is an array of
+ * 2 or more elements, use that array
+ */
+ if (argc == 1) {
+ argv = (pval **)emalloc(sizeof(pval *) * argc);
+ if (getParametersArray(ht, argc, argv) == FAILURE ||
+ argv[0]->type != IS_ARRAY) {
+ WRONG_PARAM_COUNT;
+ }
+ if (argv[0]->value.ht->nNumOfElements < 2) {
+ php3_error(E_WARNING,
+ "min: array must contain at least 2 elements");
+ RETURN_FALSE;
+ }
+ /* replace the function parameters with the array */
+ ht = argv[0]->value.ht;
+ argc = _php3_hash_num_elements(ht);
+ efree(argv);
+ } else if (argc < 2) {
+ WRONG_PARAM_COUNT;
+ }
+ argv = (pval **)emalloc(sizeof(pval *) * argc);
+ if (getParametersArray(ht, argc, argv) == FAILURE) {
+ efree(argv);
+ WRONG_PARAM_COUNT;
+ }
+ /* figure out what types to compare
+ * if the arguments contain a double, convert all of them to a double
+ * else convert all of them to long
+ */
+ for (i = 0; i < argc; i++) {
+ if (argv[i]->type == IS_DOUBLE) {
+ max_type = IS_DOUBLE;
+ break;
+ }
+ }
+ if (max_type == IS_LONG) {
+ convert_to_long(argv[0]);
+ return_value->value.lval = argv[0]->value.lval;
+ for (i = 1; i < argc; i++) {
+ convert_to_long(argv[i]);
+ if (argv[i]->value.lval > return_value->value.lval) {
+ return_value->value.lval = argv[i]->value.lval;
+ }
+ }
+ } else {
+ convert_to_double(argv[0]);
+ return_value->value.dval = argv[0]->value.dval;
+ for (i = 1; i < argc; i++) {
+ convert_to_double(argv[i]);
+ if (argv[i]->value.dval > return_value->value.dval) {
+ return_value->value.dval = argv[i]->value.dval;
+ }
+ }
+ }
+ efree(argv);
+ return_value->type = max_type;
+}
+#endif
+
+void php3_get_current_user(INTERNAL_FUNCTION_PARAMETERS)
+{
+ TLS_VARS;
+
+ RETURN_STRING(_php3_get_current_user(),1);
+}
+
+
+void php3_get_cfg_var(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *varname;
+ char *value;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &varname)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_string(varname);
+
+ if (cfg_get_string(varname->value.str.val,&value)==FAILURE) {
+ RETURN_FALSE;
+ }
+ RETURN_STRING(value,1);
+}
+
+void php3_set_magic_quotes_runtime(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *new_setting;
+ PLS_FETCH();
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &new_setting)==FAILURE) {
+ RETURN_FALSE;
+ }
+ convert_to_long(new_setting);
+
+ PG(magic_quotes_runtime)=new_setting->value.lval;
+ RETURN_TRUE;
+}
+
+void php3_get_magic_quotes_runtime(INTERNAL_FUNCTION_PARAMETERS)
+{
+ PLS_FETCH();
+
+ RETURN_LONG(PG(magic_quotes_runtime));
+}
+
+void php3_get_magic_quotes_gpc(INTERNAL_FUNCTION_PARAMETERS)
+{
+ PLS_FETCH();
+
+ RETURN_LONG(PG(magic_quotes_gpc));
+}
+
+
+void php3_is_type(INTERNAL_FUNCTION_PARAMETERS,int type)
+{
+ pval *arg;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &arg)==FAILURE) {
+ RETURN_FALSE;
+ }
+ if (arg->type == type) {
+ RETURN_TRUE;
+ } else {
+ RETURN_FALSE;
+ }
+}
+
+
+void php3_is_long(INTERNAL_FUNCTION_PARAMETERS) { php3_is_type(INTERNAL_FUNCTION_PARAM_PASSTHRU, IS_LONG); }
+void php3_is_double(INTERNAL_FUNCTION_PARAMETERS) { php3_is_type(INTERNAL_FUNCTION_PARAM_PASSTHRU, IS_DOUBLE); }
+void php3_is_string(INTERNAL_FUNCTION_PARAMETERS) { php3_is_type(INTERNAL_FUNCTION_PARAM_PASSTHRU, IS_STRING); }
+void php3_is_array(INTERNAL_FUNCTION_PARAMETERS) { php3_is_type(INTERNAL_FUNCTION_PARAM_PASSTHRU, IS_ARRAY); }
+void php3_is_object(INTERNAL_FUNCTION_PARAMETERS) { php3_is_type(INTERNAL_FUNCTION_PARAM_PASSTHRU, IS_OBJECT); }
+
+
+void php3_leak(INTERNAL_FUNCTION_PARAMETERS)
+{
+ int leakbytes=3;
+ pval *leak;
+
+ if (ARG_COUNT(ht)>=1) {
+ if (getParameters(ht, 1, &leak)==SUCCESS) {
+ convert_to_long(leak);
+ leakbytes = leak->value.lval;
+ }
+ }
+
+ emalloc(leakbytes);
+}
+
+/*
+ 1st arg = error message
+ 2nd arg = error option
+ 3rd arg = optional parameters (email address or tcp address)
+ 4th arg = used for additional headers if email
+
+ error options
+ 0 = send to php3_error_log (uses syslog or file depending on ini setting)
+ 1 = send via email to 3rd parameter 4th option = additional headers
+ 2 = send via tcp/ip to 3rd parameter (name or ip:port)
+ 3 = save to file in 3rd parameter
+*/
+
+void php3_error_log(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval *string, *erropt = NULL, *option = NULL, *emailhead = NULL;
+ int opt_err = 0;
+ char *message, *opt=NULL, *headers=NULL;
+ TLS_VARS;
+
+ switch(ARG_COUNT(ht)) {
+ case 1:
+ if (getParameters(ht,1,&string) == FAILURE) {
+ php3_error(E_WARNING,"Invalid argument 1 in error_log");
+ RETURN_FALSE;
+ }
+ break;
+ case 2:
+ if (getParameters(ht,2,&string,&erropt) == FAILURE) {
+ php3_error(E_WARNING,"Invalid arguments in error_log");
+ RETURN_FALSE;
+ }
+ convert_to_long(erropt);
+ opt_err=erropt->value.lval;
+ break;
+ case 3:
+ if (getParameters(ht,3,&string,&erropt,&option) == FAILURE){
+ php3_error(E_WARNING,"Invalid arguments in error_log");
+ RETURN_FALSE;
+ }
+ convert_to_long(erropt);
+ opt_err=erropt->value.lval;
+ convert_to_string(option);
+ opt=option->value.str.val;
+ break;
+ case 4:
+ if (getParameters(ht,4,&string,&erropt,&option,&emailhead) == FAILURE){
+ php3_error(E_WARNING,"Invalid arguments in error_log");
+ RETURN_FALSE;
+ }
+ break;
+ default:
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_string(string);
+ message=string->value.str.val;
+ if (erropt != NULL) {
+ convert_to_long(erropt);
+ opt_err=erropt->value.lval;
+ }
+ if (option != NULL) {
+ convert_to_string(option);
+ opt=option->value.str.val;
+ }
+ if (emailhead != NULL) {
+ convert_to_string(emailhead);
+ headers=emailhead->value.str.val;
+ }
+
+ if (_php3_error_log(opt_err,message,opt,headers)==FAILURE) {
+ RETURN_FALSE;
+ }
+
+ RETURN_TRUE;
+}
+
+PHPAPI int _php3_error_log(int opt_err,char *message,char *opt,char *headers){
+ FILE *logfile;
+ int issock=0, socketd=0;;
+
+ switch(opt_err){
+ case 1: /*send an email*/
+ {
+#if HAVE_SENDMAIL
+ if (!_php3_mail(opt,"PHP3 error_log message",message,headers)){
+ return FAILURE;
+ }
+#else
+ php3_error(E_WARNING,"Mail option not available!");
+ return FAILURE;
+#endif
+ }
+ break;
+ case 2: /*send to an address */
+ php3_error(E_WARNING,"TCP/IP option not available!");
+ return FAILURE;
+ break;
+ case 3: /*save to a file*/
+ logfile=php3_fopen_wrapper(opt,"a", (IGNORE_URL|ENFORCE_SAFE_MODE), &issock, &socketd);
+ if(!logfile) {
+ php3_error(E_WARNING,"error_log: Unable to write to %s",opt);
+ return FAILURE;
+ }
+ fwrite(message,strlen(message),1,logfile);
+ fclose(logfile);
+ break;
+ default:
+ php3_log_err(message);
+ break;
+ }
+ return SUCCESS;
+}
+
+
+void php3_call_user_func(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval **params;
+ pval retval;
+ int arg_count=ARG_COUNT(ht);
+ CLS_FETCH();
+
+ if (arg_count<1) {
+ WRONG_PARAM_COUNT;
+ }
+ params = (pval **) emalloc(sizeof(pval)*arg_count);
+
+ if (getParametersArray(ht, arg_count, params)==FAILURE) {
+ efree(params);
+ RETURN_FALSE;
+ }
+ convert_to_string(params[0]);
+ if (call_user_function(CG(function_table), NULL, params[0], &retval, arg_count-1, params+1)==SUCCESS) {
+ *return_value = retval;
+ } else {
+ php3_error(E_WARNING,"Unable to call %s() - function does not exist", params[0]->value.str.val);
+ }
+ efree(params);
+}
+
+
+void php3_call_user_method(INTERNAL_FUNCTION_PARAMETERS)
+{
+ pval **params;
+ pval retval;
+ int arg_count=ARG_COUNT(ht);
+
+ if (arg_count<2) {
+ WRONG_PARAM_COUNT;
+ }
+ params = (pval **) emalloc(sizeof(pval)*arg_count);
+
+ if (getParametersArray(ht, arg_count, params)==FAILURE) {
+ efree(params);
+ RETURN_FALSE;
+ }
+ if (params[1]->type != IS_OBJECT) {
+ php3_error(E_WARNING,"2nd argument is not an object\n");
+ efree(params);
+ RETURN_FALSE;
+ }
+ convert_to_string(params[0]);
+ if (call_user_function(CG(function_table), params[1], params[0], &retval, arg_count-2, params+2)==SUCCESS) {
+ *return_value = retval;
+ } else {
+ php3_error(E_WARNING,"Unable to call %s() - function does not exist", params[0]->value.str.val);
+ }
+ efree(params);
+}
+
+
+void user_shutdown_function_dtor(pval *user_shutdown_function_name)
+{
+ pval retval;
+ CLS_FETCH();
+
+ if (call_user_function(CG(function_table), NULL, user_shutdown_function_name, &retval, 0, NULL)==SUCCESS) {
+ pval_destructor(&retval);
+ }
+ pval_destructor(user_shutdown_function_name);
+}
+
+
+void php3_call_shutdown_functions(void)
+{
+ if (user_shutdown_function_names) {
+ _php3_hash_destroy(user_shutdown_function_names);
+ efree(user_shutdown_function_names);
+ }
+}
+
+/* {{{ proto void register_shutdown_function(string function_name)
+ Register a user-level function to be called on request termination */
+PHP_FUNCTION(register_shutdown_function)
+{
+ pval *arg, shutdown_function_name;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &arg)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_string(arg);
+ if (!user_shutdown_function_names) {
+ user_shutdown_function_names = (HashTable *) emalloc(sizeof(HashTable));
+ _php3_hash_init(user_shutdown_function_names, 0, NULL, (void (*)(void *))user_shutdown_function_dtor, 0);
+ }
+
+ shutdown_function_name = *arg;
+ pval_copy_constructor(&shutdown_function_name);
+
+ _php3_hash_next_index_insert(user_shutdown_function_names, &shutdown_function_name, sizeof(pval), NULL);
+}
+/* }}} */
+
+
+void php_get_highlight_struct(zend_syntax_highlighter_ini *syntax_highlighter_ini)
+{
+ syntax_highlighter_ini->highlight_comment = INI_STR("highlight_comment");
+ syntax_highlighter_ini->highlight_default = INI_STR("highlight_default");
+ syntax_highlighter_ini->highlight_html = INI_STR("highlight_html");
+ syntax_highlighter_ini->highlight_keyword = INI_STR("highlight_keyword");
+ syntax_highlighter_ini->highlight_string = INI_STR("highlight_string");
+}
+
+
+/* {{{ proto void highlight_file(string file_name)
+ Syntax highlight a source file */
+PHP_FUNCTION(highlight_file)
+{
+ pval *filename;
+ zend_syntax_highlighter_ini syntax_highlighter_ini;
+
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &filename)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+ convert_to_string(filename);
+
+ php_get_highlight_struct(&syntax_highlighter_ini);
+
+ if (highlight_file(filename->value.str.val, &syntax_highlighter_ini)==FAILURE) {
+ RETURN_FALSE;
+ }
+ RETURN_TRUE;
+}
+/* }}} */
+
+
+/* {{{ proto void highlight_file(string file_name)
+ Syntax highlight a source file */
+PHP_FUNCTION(highlight_string)
+{
+ pval *expr;
+ zend_syntax_highlighter_ini syntax_highlighter_ini;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &expr)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_string(expr);
+
+ php_get_highlight_struct(&syntax_highlighter_ini);
+
+ if (highlight_string(expr, &syntax_highlighter_ini)==FAILURE) {
+ RETURN_FALSE;
+ }
+ RETURN_TRUE;
+}
+/* }}} */
+
+
+pval test_class_get_property(zend_property_reference *property_reference)
+{
+ pval result;
+ zend_overloaded_element *overloaded_property;
+ zend_llist_element *element;
+
+
+ printf("Reading a property from a TestClass object:\n");
+
+ for (element=property_reference->elements_list.head; element; element=element->next) {
+ overloaded_property = (zend_overloaded_element *) element->data;
+ switch (overloaded_property->type) {
+ case IS_ARRAY:
+ printf("Array offset: ");
+ break;
+ case IS_OBJECT:
+ printf("Object property: ");
+ break;
+ }
+ switch (overloaded_property->element.type) {
+ case IS_LONG:
+ printf("%d (numeric)\n", overloaded_property->element.value.lval);
+ break;
+ case IS_STRING:
+ printf("'%s'\n", overloaded_property->element.value.str.val);
+ break;
+ }
+ pval_destructor(&overloaded_property->element);
+ }
+
+ result.value.str.val = estrndup("testing", 7);
+ result.value.str.len = 7;
+ result.type = IS_STRING;
+ return result;
+}
+
+
+int test_class_set_property(zend_property_reference *property_reference, pval *value)
+{
+ zend_overloaded_element *overloaded_property;
+ zend_llist_element *element;
+
+ printf("Writing to a property from a TestClass object:\n");
+ printf("Writing '");
+ zend_print_variable(value);
+ printf("'\n");
+
+ for (element=property_reference->elements_list.head; element; element=element->next) {
+ overloaded_property = (zend_overloaded_element *) element->data;
+ switch (overloaded_property->type) {
+ case IS_ARRAY:
+ printf("Array offset: ");
+ break;
+ case IS_OBJECT:
+ printf("Object property: ");
+ break;
+ }
+ switch (overloaded_property->element.type) {
+ case IS_LONG:
+ printf("%d (numeric)\n", overloaded_property->element.value.lval);
+ break;
+ case IS_STRING:
+ printf("'%s'\n", overloaded_property->element.value.str.val);
+ break;
+ }
+ pval_destructor(&overloaded_property->element);
+ }
+
+ return 0;
+}
+
+
+
+void test_class_call_function(INTERNAL_FUNCTION_PARAMETERS, zend_property_reference *property_reference)
+{
+ zend_overloaded_element *overloaded_property;
+ zend_llist_element *element;
+
+
+ printf("Reading a property from a TestClass object:\n");
+
+ for (element=property_reference->elements_list.head; element; element=element->next) {
+ overloaded_property = (zend_overloaded_element *) element->data;
+ switch (overloaded_property->type) {
+ case IS_ARRAY:
+ printf("Array offset: ");
+ break;
+ case IS_OBJECT:
+ printf("Object property: ");
+ break;
+ case IS_METHOD:
+ printf("Overloaded method: ");
+ }
+ switch (overloaded_property->element.type) {
+ case IS_LONG:
+ printf("%d (numeric)\n", overloaded_property->element.value.lval);
+ break;
+ case IS_STRING:
+ printf("'%s'\n", overloaded_property->element.value.str.val);
+ break;
+ }
+ pval_destructor(&overloaded_property->element);
+ }
+
+ printf("%d arguments\n", zend_hash_num_elements(ht));
+ return_value->value.str.val = estrndup("testing", 7);
+ return_value->value.str.len = 7;
+ return_value->type = IS_STRING;
+}
+
+
+void test_class_startup()
+{
+ zend_class_entry class_entry;
+
+ class_entry.name = strdup("TestClass");
+ class_entry.name_length = sizeof("TestClass")-1;
+
+ class_entry.handle_property_get = test_class_get_property;
+ class_entry.handle_property_set = test_class_set_property;
+ class_entry.handle_function_call = test_class_call_function;
+ register_internal_class(&class_entry);
+}
+
+
+PHP_FUNCTION(ob_start)
+{
+ zend_start_ob_buffering();
+}
+
+
+PHP_FUNCTION(ob_end_flush)
+{
+ zend_end_ob_buffering(1);
+}
+
+
+PHP_FUNCTION(ob_end_clean)
+{
+ zend_end_ob_buffering(0);
+}
+
+
+PHP_FUNCTION(ob_get_contents)
+{
+ if (zend_ob_get_buffer(return_value)==FAILURE) {
+ RETURN_FALSE;
+ }
+}
+
+
+PHP_FUNCTION(ini_get)
+{
+ pval *varname;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &varname)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ return_value->value.str.val = php_ini_string(varname->value.str.val, varname->value.str.len+1, 0);
+
+ if (!return_value->value.str.val) {
+ RETURN_FALSE;
+ }
+
+ return_value->value.str.len = strlen(return_value->value.str.val);
+ return_value->type = IS_STRING;
+ pval_copy_constructor(return_value);
+}
+
+
+PHP_FUNCTION(ini_alter)
+{
+ pval *varname, *new_value;
+ char *old_value;
+
+ if (ARG_COUNT(ht)!=2 || getParameters(ht, 2, &varname, &new_value)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ old_value = php_ini_string(varname->value.str.val, varname->value.str.len+1, 0);
+
+
+ convert_to_string(new_value);
+
+ if (php_alter_ini_entry(varname->value.str.val, varname->value.str.len+1, new_value->value.str.val, new_value->value.str.len, PHP_INI_USER)==FAILURE) {
+ RETURN_FALSE;
+ }
+ if (old_value) {
+ RETURN_STRING(old_value, 1);
+ } else {
+ RETURN_FALSE;
+ }
+}
+
+
+
+PHP_FUNCTION(ini_restore)
+{
+ pval *varname;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &varname)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_string(varname);
+
+ php_restore_ini_entry(varname->value.str.val, varname->value.str.len);
+}
+
+
+PHP_FUNCTION(print_r)
+{
+ pval *expr;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &expr)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ zend_print_pval_r(expr, 0);
+ RETURN_TRUE;
+}
+
+
+/* This should go back to PHP */
+PHP_FUNCTION(define)
+{
+ pval *var, *val, *non_cs;
+ int case_sensitive;
+ zend_constant c;
+
+ switch(ARG_COUNT(ht)) {
+ case 2:
+ if (getParameters(ht, 2, &var, &val)==FAILURE) {
+ RETURN_FALSE;
+ }
+ case_sensitive = CONST_CS;
+ break;
+ case 3:
+ if (getParameters(ht, 3, &var, &val, &non_cs)==FAILURE) {
+ RETURN_FALSE;
+ }
+ convert_to_long(non_cs);
+ if (non_cs->value.lval) {
+ case_sensitive = 0;
+ } else {
+ case_sensitive = CONST_CS;
+ }
+ break;
+ default:
+ WRONG_PARAM_COUNT;
+ break;
+ }
+ switch(val->type) {
+ case IS_LONG:
+ case IS_DOUBLE:
+ case IS_STRING:
+ break;
+ default:
+ php3_error(E_WARNING,"Constants may only evaluate to scalar values");
+ RETURN_FALSE;
+ break;
+ }
+ convert_to_string(var);
+
+ c.value = *val;
+ pval_copy_constructor(&c.value);
+ c.flags = case_sensitive | ~CONST_PERSISTENT; /* non persistent */
+ c.name = php3_strndup(var->value.str.val, var->value.str.len);
+ c.name_len = var->value.str.len+1;
+ zend_register_constant(&c);
+ RETURN_TRUE;
+}
+
+
+PHP_FUNCTION(defined)
+{
+ pval *var;
+ pval c;
+
+ if (ARG_COUNT(ht)!=1 || getParameters(ht, 1, &var)==FAILURE) {
+ WRONG_PARAM_COUNT;
+ }
+
+ convert_to_string(var);
+ if (zend_get_constant(var->value.str.val, var->value.str.len, &c)) {
+ pval_destructor(&c);
+ RETURN_LONG(1);
+ } else {
+ RETURN_LONG(0);
+ }
+}
+
+/*
+ * Local variables:
+ * tab-width: 4
+ * c-basic-offset: 4
+ * End:
+ */