1999-11-14 04:31:54 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
2001-12-11 23:32:16 +08:00
|
|
|
| PHP Version 4 |
|
1999-11-14 04:31:54 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2003-01-01 00:08:15 +08:00
|
|
|
| Copyright (c) 1997-2003 The PHP Group |
|
1999-11-14 04:31:54 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2000-05-18 23:34:45 +08:00
|
|
|
| This source file is subject to version 2.02 of the PHP license, |
|
1999-11-14 04:31:54 +08:00
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
|
|
| available at through the world-wide-web at |
|
2000-05-18 23:34:45 +08:00
|
|
|
| http://www.php.net/license/2_02.txt. |
|
1999-11-14 04:31:54 +08:00
|
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Authors: Andi Gutmans <andi@zend.com> |
|
|
|
|
| Zeev Suraski <zeev@zend.com> |
|
2000-03-07 04:37:11 +08:00
|
|
|
| Rasmus Lerdorf <rasmus@php.net> |
|
2002-05-14 01:28:38 +08:00
|
|
|
| Andrei Zmievski <andrei@php.net> |
|
2000-10-22 01:48:11 +08:00
|
|
|
| Stig Venaas <venaas@php.net> |
|
2001-02-20 13:36:40 +08:00
|
|
|
| Jason Greene <jason@php.net> |
|
1999-11-14 04:31:54 +08:00
|
|
|
+----------------------------------------------------------------------+
|
2000-07-24 09:40:02 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* $Id$ */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
#include "php.h"
|
|
|
|
#include "php_ini.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#if HAVE_STRING_H
|
|
|
|
#include <string.h>
|
|
|
|
#else
|
|
|
|
#include <strings.h>
|
|
|
|
#endif
|
2000-02-11 23:59:30 +08:00
|
|
|
#ifdef PHP_WIN32
|
1999-11-14 04:31:54 +08:00
|
|
|
#include "win32/unistd.h"
|
|
|
|
#endif
|
|
|
|
#include "zend_globals.h"
|
|
|
|
#include "php_globals.h"
|
|
|
|
#include "php_array.h"
|
1999-12-02 01:21:52 +08:00
|
|
|
#include "basic_functions.h"
|
2000-04-13 03:39:02 +08:00
|
|
|
#include "php_string.h"
|
2000-05-10 03:27:00 +08:00
|
|
|
#include "php_rand.h"
|
2002-02-02 04:17:32 +08:00
|
|
|
#include "php_smart_str.h"
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-12-12 06:42:01 +08:00
|
|
|
#ifdef ZTS
|
|
|
|
int array_globals_id;
|
|
|
|
#else
|
|
|
|
php_array_globals array_globals;
|
|
|
|
#endif
|
|
|
|
|
2002-01-25 17:01:02 +08:00
|
|
|
#define EXTR_OVERWRITE 0
|
|
|
|
#define EXTR_SKIP 1
|
|
|
|
#define EXTR_PREFIX_SAME 2
|
|
|
|
#define EXTR_PREFIX_ALL 3
|
|
|
|
#define EXTR_PREFIX_INVALID 4
|
|
|
|
#define EXTR_PREFIX_IF_EXISTS 5
|
|
|
|
#define EXTR_IF_EXISTS 6
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-09-10 04:05:20 +08:00
|
|
|
#define EXTR_REFS 0x100
|
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
#define SORT_REGULAR 0
|
|
|
|
#define SORT_NUMERIC 1
|
|
|
|
#define SORT_STRING 2
|
2000-05-18 08:47:57 +08:00
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
#define SORT_DESC 3
|
|
|
|
#define SORT_ASC 4
|
2000-07-12 00:48:03 +08:00
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
#define CASE_LOWER 0
|
|
|
|
#define CASE_UPPER 1
|
2001-10-06 02:38:53 +08:00
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
#define COUNT_NORMAL 0
|
|
|
|
#define COUNT_RECURSIVE 1
|
2001-10-06 02:38:53 +08:00
|
|
|
|
2002-09-12 16:04:42 +08:00
|
|
|
#define DIFF_NORMAL 0
|
|
|
|
#define DIFF_ASSOC 1
|
|
|
|
|
2002-09-21 22:50:04 +08:00
|
|
|
#define INTERSECT_NORMAL 0
|
|
|
|
#define INTERSECT_ASSOC 1
|
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_MINIT_FUNCTION(array)
|
|
|
|
{
|
1999-12-12 06:42:01 +08:00
|
|
|
#ifdef ZTS
|
2001-07-27 18:16:41 +08:00
|
|
|
ts_allocate_id(&array_globals_id, sizeof(php_array_globals), NULL, NULL);
|
1999-12-12 06:42:01 +08:00
|
|
|
#endif
|
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
REGISTER_LONG_CONSTANT("EXTR_OVERWRITE", EXTR_OVERWRITE, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("EXTR_SKIP", EXTR_SKIP, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("EXTR_PREFIX_SAME", EXTR_PREFIX_SAME, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("EXTR_PREFIX_ALL", EXTR_PREFIX_ALL, CONST_CS | CONST_PERSISTENT);
|
2001-01-23 01:27:02 +08:00
|
|
|
REGISTER_LONG_CONSTANT("EXTR_PREFIX_INVALID", EXTR_PREFIX_INVALID, CONST_CS | CONST_PERSISTENT);
|
2002-01-25 17:01:02 +08:00
|
|
|
REGISTER_LONG_CONSTANT("EXTR_PREFIX_IF_EXISTS", EXTR_PREFIX_IF_EXISTS, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("EXTR_IF_EXISTS", EXTR_IF_EXISTS, CONST_CS | CONST_PERSISTENT);
|
2002-09-10 04:05:20 +08:00
|
|
|
REGISTER_LONG_CONSTANT("EXTR_REFS", EXTR_REFS, CONST_CS | CONST_PERSISTENT);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-12-14 03:42:26 +08:00
|
|
|
REGISTER_LONG_CONSTANT("SORT_ASC", SORT_ASC, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("SORT_DESC", SORT_DESC, CONST_CS | CONST_PERSISTENT);
|
|
|
|
|
2000-05-18 08:47:57 +08:00
|
|
|
REGISTER_LONG_CONSTANT("SORT_REGULAR", SORT_REGULAR, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("SORT_NUMERIC", SORT_NUMERIC, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("SORT_STRING", SORT_STRING, CONST_CS | CONST_PERSISTENT);
|
2001-10-06 02:38:53 +08:00
|
|
|
REGISTER_LONG_CONSTANT("CASE_LOWER", CASE_LOWER, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("CASE_UPPER", CASE_UPPER, CONST_CS | CONST_PERSISTENT);
|
2000-05-18 08:47:57 +08:00
|
|
|
|
2001-12-30 04:59:59 +08:00
|
|
|
REGISTER_LONG_CONSTANT("COUNT_NORMAL", COUNT_NORMAL, CONST_CS | CONST_PERSISTENT);
|
|
|
|
REGISTER_LONG_CONSTANT("COUNT_RECURSIVE", COUNT_RECURSIVE, CONST_CS | CONST_PERSISTENT);
|
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
return SUCCESS;
|
|
|
|
}
|
|
|
|
|
1999-12-12 06:42:01 +08:00
|
|
|
PHP_MSHUTDOWN_FUNCTION(array)
|
|
|
|
{
|
|
|
|
#ifdef ZTS
|
2002-06-18 21:16:33 +08:00
|
|
|
ts_free_id(array_globals_id);
|
1999-12-12 06:42:01 +08:00
|
|
|
#endif
|
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
return SUCCESS;
|
1999-12-12 06:42:01 +08:00
|
|
|
}
|
|
|
|
|
2001-08-06 11:50:52 +08:00
|
|
|
static void set_compare_func(int sort_type TSRMLS_DC)
|
2000-05-18 08:47:57 +08:00
|
|
|
{
|
|
|
|
switch (sort_type) {
|
|
|
|
case SORT_NUMERIC:
|
|
|
|
ARRAYG(compare_func) = numeric_compare_function;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SORT_STRING:
|
|
|
|
ARRAYG(compare_func) = string_compare_function;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SORT_REGULAR:
|
|
|
|
default:
|
|
|
|
ARRAYG(compare_func) = compare_function;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_key_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
1999-11-15 03:59:52 +08:00
|
|
|
Bucket *f;
|
|
|
|
Bucket *s;
|
2001-09-18 05:02:53 +08:00
|
|
|
zval result;
|
|
|
|
zval first;
|
|
|
|
zval second;
|
1999-11-15 03:59:52 +08:00
|
|
|
|
|
|
|
f = *((Bucket **) a);
|
|
|
|
s = *((Bucket **) b);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-11-15 03:59:52 +08:00
|
|
|
if (f->nKeyLength == 0) {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE(first) = IS_LONG;
|
|
|
|
Z_LVAL(first) = f->h;
|
1999-11-15 03:59:52 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE(first) = IS_STRING;
|
|
|
|
Z_STRVAL(first) = f->arKey;
|
2001-12-30 04:47:25 +08:00
|
|
|
Z_STRLEN(first) = f->nKeyLength-1;
|
1999-11-15 03:59:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (s->nKeyLength == 0) {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE(second) = IS_LONG;
|
|
|
|
Z_LVAL(second) = s->h;
|
1999-11-15 03:59:52 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE(second) = IS_STRING;
|
|
|
|
Z_STRVAL(second) = s->arKey;
|
2001-12-30 04:47:25 +08:00
|
|
|
Z_STRLEN(second) = s->nKeyLength-1;
|
1999-11-15 03:59:52 +08:00
|
|
|
}
|
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
if (ARRAYG(compare_func)(&result, &first, &second TSRMLS_CC) == FAILURE) {
|
|
|
|
return 0;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
if (Z_TYPE(result) == IS_DOUBLE) {
|
|
|
|
if (Z_DVAL(result) < 0) {
|
1999-11-22 01:13:39 +08:00
|
|
|
return -1;
|
2002-06-18 21:16:33 +08:00
|
|
|
} else if (Z_DVAL(result) > 0) {
|
1999-11-22 01:13:39 +08:00
|
|
|
return 1;
|
2002-06-18 21:16:33 +08:00
|
|
|
} else {
|
1999-11-22 01:13:39 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2002-06-18 21:16:33 +08:00
|
|
|
}
|
1999-11-22 01:13:39 +08:00
|
|
|
|
1999-11-15 03:59:52 +08:00
|
|
|
convert_to_long(&result);
|
1999-11-22 01:13:39 +08:00
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_LVAL(result) < 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
return -1;
|
2000-11-27 21:31:21 +08:00
|
|
|
} else if (Z_LVAL(result) > 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
return 1;
|
1999-11-22 01:13:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_reverse_key_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
2001-09-18 05:02:53 +08:00
|
|
|
return array_key_compare(a, b TSRMLS_CC) * -1;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool krsort(array array_arg [, int sort_flags])
|
|
|
|
Sort an array by key value in reverse order */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(krsort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval *array;
|
|
|
|
long sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {
|
|
|
|
RETURN_FALSE;
|
2000-05-18 08:47:57 +08:00
|
|
|
}
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
|
|
|
|
target_hash = HASH_OF(array);
|
|
|
|
set_compare_func(sort_type TSRMLS_CC);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_reverse_key_compare, 0 TSRMLS_CC) == FAILURE) {
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool ksort(array array_arg [, int sort_flags])
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array by key */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(ksort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval *array;
|
|
|
|
long sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {
|
|
|
|
RETURN_FALSE;
|
2000-05-18 08:47:57 +08:00
|
|
|
}
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
|
|
|
|
target_hash = HASH_OF(array);
|
|
|
|
set_compare_func(sort_type TSRMLS_CC);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_key_compare, 0 TSRMLS_CC) == FAILURE) {
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2001-05-23 09:50:11 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-12-30 04:59:59 +08:00
|
|
|
|
2002-03-12 14:56:57 +08:00
|
|
|
static int php_count_recursive(zval *array, long mode TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
2002-01-11 05:31:06 +08:00
|
|
|
long cnt = 0;
|
2001-12-30 04:59:59 +08:00
|
|
|
zval **element;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-09-22 00:10:33 +08:00
|
|
|
if (Z_TYPE_P(array) == IS_ARRAY) {
|
2002-10-10 01:15:56 +08:00
|
|
|
cnt = zend_hash_num_elements(Z_ARRVAL_P(array));
|
2001-12-30 04:59:59 +08:00
|
|
|
if (mode == COUNT_RECURSIVE) {
|
2002-01-11 05:31:06 +08:00
|
|
|
HashPosition pos;
|
|
|
|
|
|
|
|
for (zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(array), &pos);
|
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_P(array), (void **) &element, &pos) == SUCCESS;
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_P(array), &pos)) {
|
2002-03-12 14:56:57 +08:00
|
|
|
cnt += php_count_recursive(*element, COUNT_RECURSIVE TSRMLS_CC);
|
2001-12-30 04:59:59 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
2002-01-11 05:31:06 +08:00
|
|
|
|
2001-12-30 04:59:59 +08:00
|
|
|
return cnt;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-12-30 04:59:59 +08:00
|
|
|
/* {{{ proto int count(mixed var [, int mode])
|
|
|
|
Count the number of elements in a variable (usually an array) */
|
|
|
|
PHP_FUNCTION(count)
|
|
|
|
{
|
|
|
|
zval *array;
|
|
|
|
long mode = COUNT_NORMAL;
|
|
|
|
|
|
|
|
if (zend_parse_parameters (ZEND_NUM_ARGS() TSRMLS_CC, "z|l", &array, &mode) == FAILURE)
|
|
|
|
return;
|
|
|
|
|
2002-01-10 00:03:36 +08:00
|
|
|
switch (Z_TYPE_P(array)) {
|
|
|
|
case IS_NULL:
|
|
|
|
RETURN_LONG(0);
|
|
|
|
break;
|
|
|
|
case IS_ARRAY:
|
2002-03-12 15:20:00 +08:00
|
|
|
RETURN_LONG (php_count_recursive (array, mode TSRMLS_CC));
|
2002-01-10 00:03:36 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
RETURN_LONG(1);
|
|
|
|
break;
|
2001-12-30 04:59:59 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* 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.
|
2000-10-22 19:18:21 +08:00
|
|
|
*
|
2000-12-29 07:50:42 +08:00
|
|
|
* This is not correct any more, depends on what compare_func is set to.
|
1999-11-14 04:31:54 +08:00
|
|
|
*/
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_data_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
Bucket *f;
|
|
|
|
Bucket *s;
|
|
|
|
pval result;
|
|
|
|
pval *first;
|
|
|
|
pval *second;
|
|
|
|
|
|
|
|
f = *((Bucket **) a);
|
|
|
|
s = *((Bucket **) b);
|
|
|
|
|
|
|
|
first = *((pval **) f->pData);
|
|
|
|
second = *((pval **) s->pData);
|
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
if (ARRAYG(compare_func)(&result, first, second TSRMLS_CC) == FAILURE) {
|
|
|
|
return 0;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
if (Z_TYPE(result) == IS_DOUBLE) {
|
|
|
|
if (Z_DVAL(result) < 0) {
|
1999-11-22 01:13:39 +08:00
|
|
|
return -1;
|
2002-06-18 21:16:33 +08:00
|
|
|
} else if (Z_DVAL(result) > 0) {
|
1999-11-22 01:13:39 +08:00
|
|
|
return 1;
|
2002-06-18 21:16:33 +08:00
|
|
|
} else {
|
1999-11-22 01:13:39 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2002-06-18 21:16:33 +08:00
|
|
|
}
|
1999-11-22 01:13:39 +08:00
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
convert_to_long(&result);
|
1999-11-22 01:13:39 +08:00
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_LVAL(result) < 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
return -1;
|
2000-11-27 21:31:21 +08:00
|
|
|
} else if (Z_LVAL(result) > 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
return 1;
|
1999-11-22 01:13:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_reverse_data_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
2001-09-18 05:02:53 +08:00
|
|
|
return array_data_compare(a, b TSRMLS_CC)*-1;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2000-04-13 03:39:02 +08:00
|
|
|
static int array_natural_general_compare(const void *a, const void *b, int fold_case)
|
|
|
|
{
|
|
|
|
Bucket *f, *s;
|
|
|
|
zval *fval, *sval;
|
|
|
|
zval first, second;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
f = *((Bucket **) a);
|
|
|
|
s = *((Bucket **) b);
|
|
|
|
|
|
|
|
fval = *((pval **) f->pData);
|
|
|
|
sval = *((pval **) s->pData);
|
|
|
|
first = *fval;
|
|
|
|
second = *sval;
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(fval) != IS_STRING) {
|
2000-04-13 03:39:02 +08:00
|
|
|
zval_copy_ctor(&first);
|
|
|
|
convert_to_string(&first);
|
|
|
|
}
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(sval) != IS_STRING) {
|
2002-08-21 19:23:44 +08:00
|
|
|
zval_copy_ctor(&second);
|
2000-04-13 03:39:02 +08:00
|
|
|
convert_to_string(&second);
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
result = strnatcmp_ex(Z_STRVAL(first), Z_STRLEN(first),
|
|
|
|
Z_STRVAL(second), Z_STRLEN(second), fold_case);
|
2000-04-13 03:39:02 +08:00
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(fval) != IS_STRING)
|
2000-04-13 03:39:02 +08:00
|
|
|
zval_dtor(&first);
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(sval) != IS_STRING)
|
2000-04-13 03:39:02 +08:00
|
|
|
zval_dtor(&second);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_natural_compare(const void *a, const void *b TSRMLS_DC)
|
2000-04-13 03:39:02 +08:00
|
|
|
{
|
|
|
|
return array_natural_general_compare(a, b, 0);
|
|
|
|
}
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_natural_case_compare(const void *a, const void *b TSRMLS_DC)
|
2000-04-13 03:39:02 +08:00
|
|
|
{
|
|
|
|
return array_natural_general_compare(a, b, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void php_natsort(INTERNAL_FUNCTION_PARAMETERS, int fold_case)
|
|
|
|
{
|
|
|
|
zval **array;
|
|
|
|
HashTable *target_hash;
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
2000-04-13 03:39:02 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
2000-04-13 03:39:02 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fold_case) {
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_natural_case_compare, 0 TSRMLS_CC) == FAILURE) {
|
2000-04-13 03:39:02 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_natural_compare, 0 TSRMLS_CC) == FAILURE) {
|
2000-04-13 03:39:02 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto void natsort(array array_arg)
|
|
|
|
Sort an array using natural sort */
|
|
|
|
PHP_FUNCTION(natsort)
|
|
|
|
{
|
|
|
|
php_natsort(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto void natcasesort(array array_arg)
|
|
|
|
Sort an array using case-insensitive natural sort */
|
|
|
|
PHP_FUNCTION(natcasesort)
|
|
|
|
{
|
|
|
|
php_natsort(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool asort(array array_arg [, int sort_flags])
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array and maintain index association */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(asort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval *array;
|
|
|
|
long sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {
|
|
|
|
RETURN_FALSE;
|
2000-05-18 08:47:57 +08:00
|
|
|
}
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
|
|
|
|
target_hash = HASH_OF(array);
|
|
|
|
set_compare_func(sort_type TSRMLS_CC);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_data_compare, 0 TSRMLS_CC) == FAILURE) {
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool arsort(array array_arg [, int sort_flags])
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array in reverse order and maintain index association */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(arsort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval *array;
|
|
|
|
long sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
|
|
|
|
target_hash = HASH_OF(array);
|
|
|
|
set_compare_func(sort_type TSRMLS_CC);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_reverse_data_compare, 0 TSRMLS_CC) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool sort(array array_arg [, int sort_flags])
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(sort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval *array;
|
|
|
|
long sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
|
|
|
|
target_hash = HASH_OF(array);
|
|
|
|
set_compare_func(sort_type TSRMLS_CC);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_data_compare, 1 TSRMLS_CC) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool rsort(array array_arg [, int sort_flags])
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array in reverse order */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(rsort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval *array;
|
|
|
|
long sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a|l", &array, &sort_type) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
|
|
|
|
target_hash = HASH_OF(array);
|
|
|
|
set_compare_func(sort_type TSRMLS_CC);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_reverse_data_compare, 1 TSRMLS_CC) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2001-10-21 15:42:35 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_user_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
Bucket *f;
|
|
|
|
Bucket *s;
|
2001-09-18 05:02:53 +08:00
|
|
|
zval **args[2];
|
|
|
|
zval *retval_ptr;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
f = *((Bucket **) a);
|
|
|
|
s = *((Bucket **) b);
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
args[0] = (zval **) f->pData;
|
|
|
|
args[1] = (zval **) s->pData;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-07-30 16:24:42 +08:00
|
|
|
if (call_user_function_ex(EG(function_table), NULL, *BG(user_compare_func_name), &retval_ptr, 2, args, 0, NULL TSRMLS_CC)==SUCCESS
|
1999-12-20 02:58:27 +08:00
|
|
|
&& retval_ptr) {
|
|
|
|
long retval;
|
|
|
|
|
|
|
|
convert_to_long_ex(&retval_ptr);
|
2000-11-27 21:31:21 +08:00
|
|
|
retval = Z_LVAL_P(retval_ptr);
|
1999-12-20 02:58:27 +08:00
|
|
|
zval_ptr_dtor(&retval_ptr);
|
|
|
|
return retval;
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool usort(array array_arg, string cmp_function)
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array by values using a user-defined comparison function */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(usort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval **array;
|
|
|
|
zval **old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
1999-12-02 01:21:52 +08:00
|
|
|
old_compare_func = BG(user_compare_func_name);
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array, &BG(user_compare_func_name)) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_user_compare, 1 TSRMLS_CC) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool uasort(array array_arg, string cmp_function)
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array with a user-defined comparison function and maintain index association */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(uasort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval **array;
|
|
|
|
zval **old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
1999-12-02 01:21:52 +08:00
|
|
|
old_compare_func = BG(user_compare_func_name);
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array, &BG(user_compare_func_name)) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_user_compare, 0 TSRMLS_CC) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
static int array_user_key_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
Bucket *f;
|
|
|
|
Bucket *s;
|
|
|
|
pval key1, key2;
|
|
|
|
pval *args[2];
|
|
|
|
pval retval;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
args[0] = &key1;
|
|
|
|
args[1] = &key2;
|
|
|
|
INIT_PZVAL(&key1);
|
|
|
|
INIT_PZVAL(&key2);
|
|
|
|
|
|
|
|
f = *((Bucket **) a);
|
|
|
|
s = *((Bucket **) b);
|
|
|
|
|
|
|
|
if (f->nKeyLength) {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_STRVAL(key1) = estrndup(f->arKey, f->nKeyLength);
|
|
|
|
Z_STRLEN(key1) = f->nKeyLength-1;
|
|
|
|
Z_TYPE(key1) = IS_STRING;
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_LVAL(key1) = f->h;
|
|
|
|
Z_TYPE(key1) = IS_LONG;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
if (s->nKeyLength) {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_STRVAL(key2) = estrndup(s->arKey, s->nKeyLength);
|
|
|
|
Z_STRLEN(key2) = s->nKeyLength-1;
|
|
|
|
Z_TYPE(key2) = IS_STRING;
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_LVAL(key2) = s->h;
|
|
|
|
Z_TYPE(key2) = IS_LONG;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2001-07-30 16:24:42 +08:00
|
|
|
status = call_user_function(EG(function_table), NULL, *BG(user_compare_func_name), &retval, 2, args TSRMLS_CC);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_dtor(&key1);
|
|
|
|
zval_dtor(&key2);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-09-22 00:10:33 +08:00
|
|
|
if (status == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
convert_to_long(&retval);
|
2000-11-27 21:31:21 +08:00
|
|
|
return Z_LVAL(retval);
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool uksort(array array_arg, string cmp_function)
|
2000-03-29 19:19:01 +08:00
|
|
|
Sort an array by keys using a user-defined comparison function */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(uksort)
|
|
|
|
{
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
zval **array;
|
|
|
|
zval **old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *target_hash;
|
|
|
|
|
1999-12-02 01:21:52 +08:00
|
|
|
old_compare_func = BG(user_compare_func_name);
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array, &BG(user_compare_func_name)) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-09-18 05:02:53 +08:00
|
|
|
if (zend_hash_sort(target_hash, zend_qsort, array_user_key_compare, 0 TSRMLS_CC) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(user_compare_func_name) = old_compare_func;
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-03-29 19:19:01 +08:00
|
|
|
/* {{{ proto mixed end(array array_arg)
|
|
|
|
Advances array argument's internal pointer to the last element and return it */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(end)
|
|
|
|
{
|
|
|
|
pval **array, **entry;
|
|
|
|
HashTable *target_hash;
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Passed variable is not an array or object");
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
zend_hash_internal_pointer_end(target_hash);
|
1999-11-21 20:13:44 +08:00
|
|
|
|
|
|
|
if (return_value_used) {
|
|
|
|
if (zend_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*return_value = **entry;
|
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-03-29 19:19:01 +08:00
|
|
|
/* {{{ proto mixed prev(array array_arg)
|
|
|
|
Move array argument's internal pointer to the previous element and return it */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(prev)
|
|
|
|
{
|
2002-10-11 03:24:45 +08:00
|
|
|
pval **array, **entry;
|
|
|
|
HashTable *target_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-10-11 03:24:45 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Passed variable is not an array or object");
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
2002-10-11 03:24:45 +08:00
|
|
|
zend_hash_move_backwards(target_hash);
|
1999-11-21 20:13:44 +08:00
|
|
|
|
|
|
|
if (return_value_used) {
|
2002-10-11 03:24:45 +08:00
|
|
|
if (zend_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
|
1999-11-21 20:13:44 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-11-21 20:13:44 +08:00
|
|
|
*return_value = **entry;
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-03-29 19:19:01 +08:00
|
|
|
/* {{{ proto mixed next(array array_arg)
|
|
|
|
Move array argument's internal pointer to the next element and return it */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(next)
|
|
|
|
{
|
2002-10-11 03:24:45 +08:00
|
|
|
pval **array, **entry;
|
|
|
|
HashTable *target_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-10-11 03:24:45 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Passed variable is not an array or object");
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
2002-10-11 03:24:45 +08:00
|
|
|
zend_hash_move_forward(target_hash);
|
1999-11-21 20:13:44 +08:00
|
|
|
|
|
|
|
if (return_value_used) {
|
2002-10-11 03:24:45 +08:00
|
|
|
if (zend_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
|
1999-11-21 20:13:44 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*return_value = **entry;
|
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-03-29 19:19:01 +08:00
|
|
|
/* {{{ proto mixed reset(array array_arg)
|
|
|
|
Set array argument's internal pointer to the first element and return it */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(reset)
|
|
|
|
{
|
2002-10-11 03:24:45 +08:00
|
|
|
pval **array, **entry;
|
|
|
|
HashTable *target_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-10-11 03:24:45 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Passed variable is not an array or object");
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2002-10-11 03:24:45 +08:00
|
|
|
zend_hash_internal_pointer_reset(target_hash);
|
1999-11-17 02:47:47 +08:00
|
|
|
|
1999-11-17 03:12:29 +08:00
|
|
|
if (return_value_used) {
|
2002-10-11 03:24:45 +08:00
|
|
|
if (zend_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
|
1999-11-21 20:13:44 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
1999-11-17 02:47:47 +08:00
|
|
|
*return_value = **entry;
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_copy_ctor(return_value);
|
1999-11-17 02:47:47 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-03-29 19:19:01 +08:00
|
|
|
/* {{{ proto mixed current(array array_arg)
|
|
|
|
Return the element currently pointed to by the internal array pointer */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(current)
|
|
|
|
{
|
2002-10-11 03:24:45 +08:00
|
|
|
pval **array, **entry;
|
|
|
|
HashTable *target_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-10-11 03:24:45 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Passed variable is not an array or object");
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2002-10-11 03:24:45 +08:00
|
|
|
if (zend_hash_get_current_data(target_hash, (void **) &entry) == FAILURE) {
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
*return_value = **entry;
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-03-29 19:19:01 +08:00
|
|
|
/* {{{ proto mixed key(array array_arg)
|
|
|
|
Return the key of the element currently pointed to by the internal array pointer */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(key)
|
|
|
|
{
|
2002-10-11 03:24:45 +08:00
|
|
|
pval **array;
|
|
|
|
char *string_key;
|
|
|
|
uint string_length;
|
|
|
|
ulong num_key;
|
|
|
|
HashTable *target_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-10-11 03:24:45 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Passed variable is not an array or object");
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2002-10-11 03:24:45 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(target_hash, &string_key, &string_length, &num_key, 0, NULL)) {
|
1999-11-14 04:31:54 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2002-10-11 03:24:45 +08:00
|
|
|
RETVAL_STRINGL(string_key, string_length - 1, 1);
|
|
|
|
break;
|
1999-11-14 04:31:54 +08:00
|
|
|
case HASH_KEY_IS_LONG:
|
2002-10-11 03:24:45 +08:00
|
|
|
RETVAL_LONG(num_key);
|
|
|
|
break;
|
|
|
|
case HASH_KEY_NON_EXISTANT:
|
|
|
|
return;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-06-25 21:21:37 +08:00
|
|
|
/* {{{ proto mixed min(mixed arg1 [, mixed arg2 [, mixed ...]])
|
2000-03-29 19:19:01 +08:00
|
|
|
Return the lowest value in an array or a series of arguments */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(min)
|
|
|
|
{
|
2000-06-06 03:47:54 +08:00
|
|
|
int argc=ZEND_NUM_ARGS();
|
1999-11-14 04:31:54 +08:00
|
|
|
pval **result;
|
|
|
|
|
|
|
|
if (argc<=0) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Atleast one value should be passed");
|
2000-03-29 20:43:13 +08:00
|
|
|
RETURN_NULL();
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-08-06 11:50:52 +08:00
|
|
|
set_compare_func(SORT_REGULAR TSRMLS_CC);
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
pval **arr;
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (zend_get_parameters_ex(1, &arr) == FAILURE || Z_TYPE_PP(arr) != IS_ARRAY) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2002-09-22 00:10:33 +08:00
|
|
|
if (zend_hash_minmax(Z_ARRVAL_PP(arr), array_data_compare, 0, (void **) &result TSRMLS_CC) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
*return_value = **result;
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Array must contain atleast one element");
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
} else {
|
2000-06-06 03:47:54 +08:00
|
|
|
pval ***args = (pval ***) emalloc(sizeof(pval **)*ZEND_NUM_ARGS());
|
1999-11-14 04:31:54 +08:00
|
|
|
pval **min, result;
|
|
|
|
int i;
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (zend_get_parameters_array_ex(ZEND_NUM_ARGS(), args)==FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
min = args[0];
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
for (i=1; i<ZEND_NUM_ARGS(); i++) {
|
2001-07-30 12:58:07 +08:00
|
|
|
is_smaller_function(&result, *args[i], *min TSRMLS_CC);
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_LVAL(result) == 1) {
|
1999-11-14 04:31:54 +08:00
|
|
|
min = args[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*return_value = **min;
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
efree(args);
|
|
|
|
}
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-06-25 21:21:37 +08:00
|
|
|
/* {{{ proto mixed max(mixed arg1 [, mixed arg2 [, mixed ...]])
|
2000-03-29 19:19:01 +08:00
|
|
|
Return the highest value in an array or a series of arguments */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(max)
|
|
|
|
{
|
2000-06-06 03:47:54 +08:00
|
|
|
int argc=ZEND_NUM_ARGS();
|
1999-11-14 04:31:54 +08:00
|
|
|
pval **result;
|
|
|
|
|
|
|
|
if (argc<=0) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Atleast one value should be passed");
|
2000-03-29 20:43:13 +08:00
|
|
|
RETURN_NULL();
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-08-06 11:50:52 +08:00
|
|
|
set_compare_func(SORT_REGULAR TSRMLS_CC);
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc == 1) {
|
|
|
|
pval **arr;
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (zend_get_parameters_ex(1, &arr) == FAILURE || Z_TYPE_PP(arr) != IS_ARRAY) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2002-09-22 00:10:33 +08:00
|
|
|
if (zend_hash_minmax(Z_ARRVAL_PP(arr), array_data_compare, 1, (void **) &result TSRMLS_CC) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
*return_value = **result;
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Array must contain atleast one element");
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
} else {
|
2000-06-06 03:47:54 +08:00
|
|
|
pval ***args = (pval ***) emalloc(sizeof(pval **)*ZEND_NUM_ARGS());
|
1999-11-14 04:31:54 +08:00
|
|
|
pval **max, result;
|
|
|
|
int i;
|
|
|
|
|
2002-09-22 00:10:33 +08:00
|
|
|
if (zend_get_parameters_array_ex(ZEND_NUM_ARGS(), args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
max = args[0];
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
for (i=1; i<ZEND_NUM_ARGS(); i++) {
|
2001-07-30 12:58:07 +08:00
|
|
|
is_smaller_or_equal_function(&result, *args[i], *max TSRMLS_CC);
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_LVAL(result) == 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
max = args[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*return_value = **max;
|
1999-11-21 20:13:44 +08:00
|
|
|
zval_copy_ctor(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
efree(args);
|
|
|
|
}
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2003-02-06 01:56:08 +08:00
|
|
|
static int php_array_walk(HashTable *target_hash, zval **userdata, int recursive TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
zval **args[3], /* Arguments to userland function */
|
1999-12-20 02:58:27 +08:00
|
|
|
*retval_ptr, /* Return value - unused */
|
1999-11-14 04:31:54 +08:00
|
|
|
*key; /* Entry key */
|
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
1999-11-14 04:31:54 +08:00
|
|
|
ulong num_key;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Set up known arguments */
|
|
|
|
args[1] = &key;
|
|
|
|
args[2] = userdata;
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(target_hash, &pos);
|
2000-04-14 02:57:37 +08:00
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
/* Iterate through hash */
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(target_hash, (void **)&args[0], &pos) == SUCCESS) {
|
2003-02-06 01:56:08 +08:00
|
|
|
if (recursive && Z_TYPE_PP(args[0]) == IS_ARRAY) {
|
|
|
|
HashTable *thash;
|
|
|
|
|
|
|
|
thash = HASH_OF(*(args[0]));
|
|
|
|
if (thash == target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "recursion detected");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
php_array_walk(thash, userdata, recursive TSRMLS_CC);
|
2002-09-22 00:10:33 +08:00
|
|
|
} else {
|
2003-02-06 01:56:08 +08:00
|
|
|
/* Allocate space for key */
|
|
|
|
MAKE_STD_ZVAL(key);
|
2002-11-27 06:12:40 +08:00
|
|
|
|
2003-02-06 01:56:08 +08:00
|
|
|
/* Set up the key */
|
|
|
|
if (zend_hash_get_current_key_ex(target_hash, &string_key, &string_key_len, &num_key, 0, &pos) == HASH_KEY_IS_LONG) {
|
|
|
|
Z_TYPE_P(key) = IS_LONG;
|
|
|
|
Z_LVAL_P(key) = num_key;
|
2002-11-27 06:12:40 +08:00
|
|
|
} else {
|
2003-02-06 01:56:08 +08:00
|
|
|
ZVAL_STRINGL(key, string_key, string_key_len-1, 1);
|
2002-11-27 05:21:32 +08:00
|
|
|
}
|
2003-02-06 01:56:08 +08:00
|
|
|
|
|
|
|
/* Call the userland function */
|
|
|
|
if (call_user_function_ex(EG(function_table), NULL, *BG(array_walk_func_name),
|
|
|
|
&retval_ptr, userdata ? 3 : 2, args, 0, NULL TSRMLS_CC) == SUCCESS) {
|
|
|
|
|
|
|
|
zval_ptr_dtor(&retval_ptr);
|
|
|
|
} else {
|
|
|
|
char *func_name;
|
2002-11-27 06:12:40 +08:00
|
|
|
|
2003-02-06 01:56:08 +08:00
|
|
|
if (zend_is_callable(*BG(array_walk_func_name), 0, &func_name)) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unable to call %s()", func_name);
|
|
|
|
} else {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unable to call %s() - function does not exist", func_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
efree(func_name);
|
|
|
|
break;
|
|
|
|
}
|
2002-09-22 00:10:33 +08:00
|
|
|
}
|
2000-02-12 05:14:42 +08:00
|
|
|
|
2002-12-05 00:41:28 +08:00
|
|
|
zval_ptr_dtor(&key);
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(target_hash, &pos);
|
2002-06-18 21:16:33 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
Modified array_walk, arsort, asort, krsort, ksort, rsort, shuffle, sort,
uasort, uksort and usort.
Corrected prototype comments for all.
Converted to use zend_parse_parameters, except for the u.?sort functions.
Changed return values for failure from implicit NULL to explicit FALSE in
asort(), krsort(), and ksort().
Changed pval to zval in the u.?sort functions.
Added test to help ensure that fixes to code do not change function
behavior. Tests are quite large (~237k) but compress down to ~16k.
Please let me know if this is a problem!
2001-12-22 19:49:56 +08:00
|
|
|
/* {{{ proto bool array_walk(array input, string funcname [, mixed userdata])
|
1999-11-14 04:31:54 +08:00
|
|
|
Apply a user function to every member of an array */
|
2001-07-28 19:36:37 +08:00
|
|
|
PHP_FUNCTION(array_walk)
|
|
|
|
{
|
2002-06-18 21:16:33 +08:00
|
|
|
int argc;
|
1999-11-14 04:31:54 +08:00
|
|
|
zval **array,
|
|
|
|
**userdata = NULL,
|
|
|
|
**old_walk_func_name;
|
|
|
|
HashTable *target_hash;
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
1999-12-02 01:21:52 +08:00
|
|
|
old_walk_func_name = BG(array_walk_func_name);
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc < 2 || argc > 3 ||
|
1999-12-19 06:40:35 +08:00
|
|
|
zend_get_parameters_ex(argc, &array, &BG(array_walk_func_name), &userdata) == FAILURE) {
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
2000-08-23 02:39:29 +08:00
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
if (Z_TYPE_PP(BG(array_walk_func_name)) != IS_ARRAY &&
|
|
|
|
Z_TYPE_PP(BG(array_walk_func_name)) != IS_STRING) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Wrong syntax for function name");
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2003-02-06 01:56:08 +08:00
|
|
|
php_array_walk(target_hash, userdata, 0 TSRMLS_CC);
|
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ proto bool array_walk_recursive(array input, string funcname [, mixed userdata])
|
|
|
|
Apply a user function recursively to every member of an array */
|
|
|
|
PHP_FUNCTION(array_walk_recursive)
|
|
|
|
{
|
|
|
|
int argc;
|
|
|
|
zval **array,
|
|
|
|
**userdata = NULL,
|
|
|
|
**old_walk_func_name;
|
|
|
|
HashTable *target_hash;
|
|
|
|
|
|
|
|
argc = ZEND_NUM_ARGS();
|
|
|
|
old_walk_func_name = BG(array_walk_func_name);
|
|
|
|
if (argc < 2 || argc > 3 ||
|
|
|
|
zend_get_parameters_ex(argc, &array, &BG(array_walk_func_name), &userdata) == FAILURE) {
|
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
if (Z_TYPE_PP(BG(array_walk_func_name)) != IS_ARRAY && Z_TYPE_PP(BG(array_walk_func_name)) != IS_STRING) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Wrong syntax for function name");
|
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
php_array_walk(target_hash, userdata, 1 TSRMLS_CC);
|
1999-12-02 01:21:52 +08:00
|
|
|
BG(array_walk_func_name) = old_walk_func_name;
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2003-02-06 01:56:08 +08:00
|
|
|
|
2001-02-20 13:36:40 +08:00
|
|
|
/* void php_search_array(INTERNAL_FUNCTION_PARAMETERS, int behavior)
|
2002-06-18 21:16:33 +08:00
|
|
|
* 0 = return boolean
|
|
|
|
* 1 = return key
|
2001-02-20 13:36:40 +08:00
|
|
|
*/
|
|
|
|
static void php_search_array(INTERNAL_FUNCTION_PARAMETERS, int behavior)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
2001-02-20 13:36:40 +08:00
|
|
|
zval **value, /* value to check for */
|
1999-11-14 04:31:54 +08:00
|
|
|
**array, /* array to check in */
|
2000-06-01 21:52:08 +08:00
|
|
|
**strict, /* strict comparison or not */
|
|
|
|
**entry, /* pointer to array entry */
|
1999-11-14 04:31:54 +08:00
|
|
|
res; /* comparison result */
|
|
|
|
HashTable *target_hash; /* array hashtable */
|
2000-06-18 11:43:58 +08:00
|
|
|
HashPosition pos; /* hash iterator */
|
2001-02-20 13:36:40 +08:00
|
|
|
ulong num_key;
|
2001-10-06 05:58:41 +08:00
|
|
|
uint str_key_len;
|
2001-02-20 13:36:40 +08:00
|
|
|
char *string_key;
|
2001-10-07 21:49:08 +08:00
|
|
|
int (*is_equal_func)(zval *, zval *, zval * TSRMLS_DC) = is_equal_function;
|
2002-06-18 21:16:33 +08:00
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() < 2 || ZEND_NUM_ARGS() > 3 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &value, &array, &strict) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2002-12-09 23:54:38 +08:00
|
|
|
|
|
|
|
#ifndef ZEND_ENGINE_2
|
2001-08-31 12:52:10 +08:00
|
|
|
if (Z_TYPE_PP(value) == IS_OBJECT) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Wrong datatype for first argument");
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2002-12-09 23:54:38 +08:00
|
|
|
#endif
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(array) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Wrong datatype for second argument");
|
2000-03-29 19:19:01 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() == 3) {
|
2000-06-01 21:52:08 +08:00
|
|
|
convert_to_boolean_ex(strict);
|
2001-02-20 13:36:40 +08:00
|
|
|
if (Z_LVAL_PP(strict)) {
|
2001-10-07 21:49:08 +08:00
|
|
|
is_equal_func = is_identical_function;
|
2002-06-18 21:16:33 +08:00
|
|
|
}
|
2000-06-01 21:52:08 +08:00
|
|
|
}
|
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
target_hash = HASH_OF(*array);
|
2000-06-18 11:43:58 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(target_hash, &pos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(target_hash, (void **)&entry, &pos) == SUCCESS) {
|
2002-06-18 21:16:33 +08:00
|
|
|
is_equal_func(&res, *value, *entry TSRMLS_CC);
|
2001-10-06 04:14:33 +08:00
|
|
|
if (Z_LVAL(res)) {
|
2002-09-22 00:10:33 +08:00
|
|
|
if (behavior == 0) {
|
2001-02-20 13:36:40 +08:00
|
|
|
RETURN_TRUE;
|
|
|
|
} else {
|
|
|
|
/* Return current key */
|
2001-10-06 05:58:41 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(target_hash, &string_key, &str_key_len, &num_key, 0, &pos)) {
|
2001-02-20 13:36:40 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2001-10-06 05:58:41 +08:00
|
|
|
RETURN_STRINGL(string_key, str_key_len-1, 1);
|
2001-02-20 13:36:40 +08:00
|
|
|
break;
|
|
|
|
case HASH_KEY_IS_LONG:
|
2001-10-06 05:58:41 +08:00
|
|
|
RETURN_LONG(num_key);
|
2001-02-20 13:36:40 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2000-06-18 11:43:58 +08:00
|
|
|
zend_hash_move_forward_ex(target_hash, &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-02-20 13:36:40 +08:00
|
|
|
|
2001-10-06 04:14:33 +08:00
|
|
|
RETURN_FALSE;
|
2001-02-20 13:36:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto bool in_array(mixed needle, array haystack [, bool strict])
|
|
|
|
Checks if the given value exists in the array */
|
|
|
|
PHP_FUNCTION(in_array)
|
|
|
|
{
|
|
|
|
php_search_array(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2001-02-20 22:23:03 +08:00
|
|
|
/* {{{ proto mixed array_search(mixed needle, array haystack [, bool strict])
|
2001-02-20 13:36:40 +08:00
|
|
|
Searches the array for a given value and returns the corresponding key if successful */
|
2001-02-20 22:23:03 +08:00
|
|
|
PHP_FUNCTION(array_search)
|
2001-02-20 13:36:40 +08:00
|
|
|
{
|
|
|
|
php_search_array(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
static int php_valid_var_name(char *var_name)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
int len, i;
|
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
if (!var_name)
|
1999-11-14 04:31:54 +08:00
|
|
|
return 0;
|
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
len = strlen(var_name);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
if (!isalpha((int)var_name[0]) && var_name[0] != '_')
|
1999-11-14 04:31:54 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (len > 1) {
|
2002-09-22 00:10:33 +08:00
|
|
|
for (i=1; i<len; i++) {
|
2001-01-23 01:27:02 +08:00
|
|
|
if (!isalnum((int)var_name[i]) && var_name[i] != '_') {
|
1999-11-14 04:31:54 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
/* {{{ proto int extract(array var_array [, int extract_type [, string prefix]])
|
1999-11-14 04:31:54 +08:00
|
|
|
Imports variables into symbol table from an array */
|
|
|
|
PHP_FUNCTION(extract)
|
|
|
|
{
|
2001-01-23 01:27:02 +08:00
|
|
|
zval **var_array, **z_extract_type, **prefix;
|
2000-03-24 22:56:19 +08:00
|
|
|
zval **entry, *data;
|
2002-02-02 04:17:32 +08:00
|
|
|
char *var_name;
|
|
|
|
smart_str final_name = {0};
|
2001-08-21 20:57:53 +08:00
|
|
|
ulong num_key;
|
|
|
|
uint var_name_len;
|
2001-01-23 01:27:02 +08:00
|
|
|
int var_exists, extract_type, key_type, count = 0;
|
2002-09-10 04:05:20 +08:00
|
|
|
zend_bool extract_refs = 0;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-02-02 04:17:32 +08:00
|
|
|
switch (ZEND_NUM_ARGS()) {
|
1999-11-14 04:31:54 +08:00
|
|
|
case 1:
|
1999-12-19 06:40:35 +08:00
|
|
|
if (zend_get_parameters_ex(1, &var_array) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2001-01-23 01:27:02 +08:00
|
|
|
extract_type = EXTR_OVERWRITE;
|
1999-11-14 04:31:54 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
2001-01-23 01:27:02 +08:00
|
|
|
if (zend_get_parameters_ex(2, &var_array, &z_extract_type) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2001-01-23 01:27:02 +08:00
|
|
|
convert_to_long_ex(z_extract_type);
|
|
|
|
extract_type = Z_LVAL_PP(z_extract_type);
|
2002-09-10 04:05:20 +08:00
|
|
|
extract_refs = (extract_type & EXTR_REFS)>>8;
|
|
|
|
extract_type &= 0xff;
|
2002-01-25 17:01:02 +08:00
|
|
|
if (extract_type > EXTR_SKIP && extract_type <= EXTR_PREFIX_IF_EXISTS) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Prefix expected to be specified");
|
2001-01-23 01:27:02 +08:00
|
|
|
return;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
2001-01-23 01:27:02 +08:00
|
|
|
if (zend_get_parameters_ex(3, &var_array, &z_extract_type, &prefix) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2001-01-23 01:27:02 +08:00
|
|
|
convert_to_long_ex(z_extract_type);
|
|
|
|
extract_type = Z_LVAL_PP(z_extract_type);
|
2002-09-10 04:05:20 +08:00
|
|
|
extract_refs = (extract_type & EXTR_REFS)>>8;
|
|
|
|
extract_type &= 0xff;
|
1999-11-14 04:31:54 +08:00
|
|
|
convert_to_string_ex(prefix);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2002-01-25 17:01:02 +08:00
|
|
|
if (extract_type < EXTR_OVERWRITE || extract_type > EXTR_IF_EXISTS) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Unknown extract type");
|
2001-01-23 01:27:02 +08:00
|
|
|
return;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(var_array) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "First argument should be an array");
|
2001-01-23 01:27:02 +08:00
|
|
|
return;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(var_array), &pos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(var_array), (void **)&entry, &pos) == SUCCESS) {
|
2001-03-17 03:29:23 +08:00
|
|
|
key_type = zend_hash_get_current_key_ex(Z_ARRVAL_PP(var_array), &var_name, &var_name_len, &num_key, 0, &pos);
|
2001-01-23 01:27:02 +08:00
|
|
|
var_exists = 0;
|
|
|
|
|
|
|
|
if (key_type == HASH_KEY_IS_STRING) {
|
|
|
|
var_name_len--;
|
|
|
|
var_exists = zend_hash_exists(EG(active_symbol_table), var_name, var_name_len + 1);
|
|
|
|
} else if (extract_type == EXTR_PREFIX_ALL || extract_type == EXTR_PREFIX_INVALID) {
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_appendl(&final_name, Z_STRVAL_PP(prefix), Z_STRLEN_PP(prefix));
|
|
|
|
smart_str_appendc(&final_name, '_');
|
|
|
|
smart_str_append_long(&final_name, num_key);
|
2001-01-23 01:27:02 +08:00
|
|
|
} else {
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(var_array), &pos);
|
2001-01-23 01:27:02 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (extract_type) {
|
2002-01-25 17:01:02 +08:00
|
|
|
case EXTR_IF_EXISTS:
|
2002-02-02 04:17:32 +08:00
|
|
|
if (!var_exists) break;
|
2002-01-25 17:01:02 +08:00
|
|
|
/* break omitted intentionally */
|
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
case EXTR_OVERWRITE:
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2001-01-23 01:27:02 +08:00
|
|
|
break;
|
|
|
|
|
2002-01-25 17:01:02 +08:00
|
|
|
case EXTR_PREFIX_IF_EXISTS:
|
2002-02-02 04:17:32 +08:00
|
|
|
if (var_exists) {
|
|
|
|
smart_str_appendl(&final_name, Z_STRVAL_PP(prefix), Z_STRLEN_PP(prefix));
|
|
|
|
smart_str_appendc(&final_name, '_');
|
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2002-01-25 17:01:02 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
case EXTR_PREFIX_SAME:
|
|
|
|
if (!var_exists)
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2001-01-23 01:27:02 +08:00
|
|
|
/* break omitted intentionally */
|
|
|
|
|
|
|
|
case EXTR_PREFIX_ALL:
|
2002-02-02 04:17:32 +08:00
|
|
|
if (final_name.len == 0) {
|
|
|
|
smart_str_appendl(&final_name, Z_STRVAL_PP(prefix), Z_STRLEN_PP(prefix));
|
|
|
|
smart_str_appendc(&final_name, '_');
|
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2001-01-23 01:27:02 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EXTR_PREFIX_INVALID:
|
2002-02-02 04:17:32 +08:00
|
|
|
if (final_name.len == 0) {
|
2001-01-23 01:27:02 +08:00
|
|
|
if (!php_valid_var_name(var_name)) {
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_appendl(&final_name, Z_STRVAL_PP(prefix), Z_STRLEN_PP(prefix));
|
|
|
|
smart_str_appendc(&final_name, '_');
|
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2001-01-23 01:27:02 +08:00
|
|
|
} else
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2001-01-23 01:27:02 +08:00
|
|
|
}
|
|
|
|
break;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
default:
|
|
|
|
if (!var_exists)
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_appendl(&final_name, var_name, var_name_len);
|
2001-01-23 01:27:02 +08:00
|
|
|
break;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-02-02 04:17:32 +08:00
|
|
|
if (final_name.len) {
|
|
|
|
smart_str_0(&final_name);
|
|
|
|
if (php_valid_var_name(final_name.c)) {
|
2002-09-10 04:05:20 +08:00
|
|
|
if (extract_refs) {
|
|
|
|
zval **orig_var;
|
|
|
|
|
|
|
|
(*entry)->is_ref = 1;
|
|
|
|
(*entry)->refcount++;
|
2002-09-22 00:10:33 +08:00
|
|
|
if (zend_hash_find(EG(active_symbol_table), final_name.c, final_name.len+1, (void **) &orig_var) == SUCCESS
|
2002-09-10 04:05:20 +08:00
|
|
|
&& PZVAL_IS_REF(*orig_var)) {
|
|
|
|
|
|
|
|
(*entry)->refcount += (*orig_var)->refcount-2;
|
|
|
|
zval_dtor(*orig_var);
|
|
|
|
FREE_ZVAL(*orig_var);
|
|
|
|
*orig_var = *entry;
|
|
|
|
} else {
|
|
|
|
zend_hash_update(EG(active_symbol_table), final_name.c, final_name.len+1, entry, sizeof(zval *), NULL);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
MAKE_STD_ZVAL(data);
|
|
|
|
*data = **entry;
|
|
|
|
zval_copy_ctor(data);
|
|
|
|
|
|
|
|
ZEND_SET_SYMBOL(EG(active_symbol_table), final_name.c, data);
|
|
|
|
}
|
2001-01-09 04:57:59 +08:00
|
|
|
|
2001-01-23 01:27:02 +08:00
|
|
|
count++;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2002-02-02 04:17:32 +08:00
|
|
|
final_name.len = 0;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(var_array), &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2001-01-09 04:57:59 +08:00
|
|
|
|
2002-02-02 04:17:32 +08:00
|
|
|
smart_str_free(&final_name);
|
|
|
|
|
2001-01-09 04:57:59 +08:00
|
|
|
RETURN_LONG(count);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
static void php_compact_var(HashTable *eg_active_symbol_table, zval *return_value, zval *entry)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
zval **value_ptr, *value, *data;
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(entry) == IS_STRING) {
|
|
|
|
if (zend_hash_find(eg_active_symbol_table, Z_STRVAL_P(entry),
|
|
|
|
Z_STRLEN_P(entry)+1, (void **)&value_ptr) != FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
value = *value_ptr;
|
1999-12-27 05:21:33 +08:00
|
|
|
ALLOC_ZVAL(data);
|
1999-11-14 04:31:54 +08:00
|
|
|
*data = *value;
|
|
|
|
zval_copy_ctor(data);
|
|
|
|
INIT_PZVAL(data);
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), Z_STRVAL_P(entry),
|
|
|
|
Z_STRLEN_P(entry)+1, &data, sizeof(zval *), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
2000-11-27 21:31:21 +08:00
|
|
|
else if (Z_TYPE_P(entry) == IS_ARRAY) {
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(entry), &pos);
|
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_P(entry), (void**)&value_ptr, &pos) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
value = *value_ptr;
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
php_compact_var(eg_active_symbol_table, return_value, value);
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_P(entry), &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-05 04:31:54 +08:00
|
|
|
/* {{{ proto array compact(mixed var_names [, mixed ...])
|
1999-11-14 04:31:54 +08:00
|
|
|
Creates a hash containing variables and their values */
|
|
|
|
PHP_FUNCTION(compact)
|
|
|
|
{
|
|
|
|
zval ***args; /* function arguments array */
|
|
|
|
int i;
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
args = (zval ***)emalloc(ZEND_NUM_ARGS() * sizeof(zval **));
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (zend_get_parameters_array_ex(ZEND_NUM_ARGS(), args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
for (i=0; i<ZEND_NUM_ARGS(); i++) {
|
|
|
|
php_compact_var(EG(active_symbol_table), return_value, *args[i]);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
efree(args);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2001-10-24 02:46:35 +08:00
|
|
|
/* {{{ proto array array_fill(int start_key, int num, mixed val)
|
2001-10-21 15:42:35 +08:00
|
|
|
Create an array containing num elements starting with index start_key each initialized to val */
|
2001-10-24 02:46:35 +08:00
|
|
|
PHP_FUNCTION(array_fill)
|
2001-10-21 15:42:35 +08:00
|
|
|
{
|
|
|
|
zval **start_key, **num, **val, *newval;
|
2001-10-23 04:49:27 +08:00
|
|
|
long i;
|
2001-10-21 15:42:35 +08:00
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() != 3 || zend_get_parameters_ex(3, &start_key, &num, &val) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2002-06-18 21:16:33 +08:00
|
|
|
/* allocate an array for return */
|
2002-12-06 06:28:02 +08:00
|
|
|
array_init(return_value);
|
2001-10-21 15:42:35 +08:00
|
|
|
|
2002-09-22 00:10:33 +08:00
|
|
|
switch (Z_TYPE_PP(start_key)) {
|
2001-10-21 15:42:35 +08:00
|
|
|
case IS_STRING:
|
|
|
|
case IS_LONG:
|
|
|
|
case IS_DOUBLE:
|
2002-09-22 00:10:33 +08:00
|
|
|
if (PZVAL_IS_REF(*val)) {
|
2001-10-23 04:49:27 +08:00
|
|
|
SEPARATE_ZVAL(val);
|
|
|
|
}
|
2001-10-21 15:42:35 +08:00
|
|
|
convert_to_long_ex(start_key);
|
2001-10-23 04:49:27 +08:00
|
|
|
zval_add_ref(val);
|
|
|
|
zend_hash_index_update(Z_ARRVAL_P(return_value), Z_LVAL_PP(start_key), val, sizeof(val), NULL);
|
2001-10-21 15:42:35 +08:00
|
|
|
break;
|
|
|
|
default:
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Wrong data type for start key");
|
2001-10-21 15:42:35 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
convert_to_long_ex(num);
|
|
|
|
i = Z_LVAL_PP(num) - 1;
|
2002-09-22 00:10:33 +08:00
|
|
|
if (i < 0) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Number of elements must be positive");
|
2001-10-21 15:42:35 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
2001-10-23 04:49:27 +08:00
|
|
|
newval = *val;
|
2002-09-22 00:10:33 +08:00
|
|
|
while (i--) {
|
2002-11-06 00:19:19 +08:00
|
|
|
#ifndef ZEND_ENGINE_2
|
|
|
|
if (newval->refcount >= 62000) {
|
2001-10-23 04:49:27 +08:00
|
|
|
MAKE_STD_ZVAL(newval);
|
|
|
|
*newval = **val;
|
|
|
|
zval_copy_ctor(newval);
|
2002-11-06 00:19:19 +08:00
|
|
|
newval->refcount = 0;
|
2001-10-23 04:49:27 +08:00
|
|
|
}
|
2002-11-06 00:19:19 +08:00
|
|
|
#endif
|
2001-10-23 04:49:27 +08:00
|
|
|
zval_add_ref(&newval);
|
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(return_value), &newval, sizeof(zval *), NULL);
|
2001-10-21 15:42:35 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2002-11-14 13:46:10 +08:00
|
|
|
/* {{{ proto array range(mixed low, mixed high[, int step])
|
2001-07-10 04:36:47 +08:00
|
|
|
Create an array containing the range of integers or characters from low to high (inclusive) */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(range)
|
|
|
|
{
|
2002-12-21 01:16:31 +08:00
|
|
|
zval *zlow, *zhigh, *zstep = NULL;
|
|
|
|
int err = 0, is_step_double = 0;
|
|
|
|
double step = 1.0;
|
2002-11-14 13:46:10 +08:00
|
|
|
|
2002-12-26 04:02:03 +08:00
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z/z/|z/", &zlow, &zhigh, &zstep) == FAILURE) {
|
2002-11-14 13:46:10 +08:00
|
|
|
RETURN_FALSE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2002-12-21 01:16:31 +08:00
|
|
|
if (zstep) {
|
|
|
|
if (Z_TYPE_P(zstep) == IS_DOUBLE || (Z_TYPE_P(zstep) == IS_STRING && is_numeric_string(Z_STRVAL_P(zstep), Z_STRLEN_P(zstep), NULL, NULL, 0) == IS_DOUBLE)) {
|
|
|
|
is_step_double = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
convert_to_double_ex(&zstep);
|
|
|
|
step = Z_DVAL_P(zstep);
|
|
|
|
|
|
|
|
/* We only want positive step values. */
|
|
|
|
if (step < 0.0) {
|
|
|
|
step *= -1;
|
|
|
|
}
|
|
|
|
}
|
2002-11-14 13:46:10 +08:00
|
|
|
|
|
|
|
/* Initialize the return_value as an array. */
|
2002-12-06 06:28:02 +08:00
|
|
|
array_init(return_value);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2002-11-14 13:46:10 +08:00
|
|
|
/* If the range is given as strings, generate an array of characters. */
|
|
|
|
if (Z_TYPE_P(zlow) == IS_STRING && Z_TYPE_P(zhigh) == IS_STRING) {
|
2002-12-21 01:16:31 +08:00
|
|
|
int type1, type2;
|
2002-11-23 19:27:56 +08:00
|
|
|
unsigned char *low, *high;
|
2002-12-21 01:16:31 +08:00
|
|
|
long lstep = (long) step;
|
2002-11-14 13:46:10 +08:00
|
|
|
|
2002-12-21 01:16:31 +08:00
|
|
|
type1 = is_numeric_string(Z_STRVAL_P(zlow), Z_STRLEN_P(zlow), NULL, NULL, 0);
|
|
|
|
type2 = is_numeric_string(Z_STRVAL_P(zhigh), Z_STRLEN_P(zhigh), NULL, NULL, 0);
|
|
|
|
|
|
|
|
if (type1 == IS_DOUBLE || type2 == IS_DOUBLE || is_step_double) {
|
|
|
|
goto double_str;
|
|
|
|
} else if (type1 == IS_LONG || type2 == IS_LONG) {
|
|
|
|
goto long_str;
|
|
|
|
}
|
|
|
|
|
2002-12-26 04:02:03 +08:00
|
|
|
convert_to_string(zlow);
|
|
|
|
convert_to_string(zhigh);
|
2002-11-23 19:27:56 +08:00
|
|
|
low = (unsigned char *)Z_STRVAL_P(zlow);
|
|
|
|
high = (unsigned char *)Z_STRVAL_P(zhigh);
|
2002-11-14 13:46:10 +08:00
|
|
|
|
|
|
|
if (*low > *high) { /* Negative steps */
|
2002-12-21 01:16:31 +08:00
|
|
|
if (*low - *high < lstep || lstep <= 0) {
|
|
|
|
err = 1;
|
|
|
|
goto err;
|
2002-11-23 19:27:56 +08:00
|
|
|
}
|
|
|
|
for (; *low >= *high; (*low) -= (unsigned int)step) {
|
2001-07-10 04:36:47 +08:00
|
|
|
add_next_index_stringl(return_value, low, 1, 1);
|
2002-11-14 13:46:10 +08:00
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
} else if (*high > *low) { /* Positive steps */
|
|
|
|
if (*high - *low < lstep || lstep <= 0) {
|
|
|
|
err = 1;
|
|
|
|
goto err;
|
2002-11-23 19:27:56 +08:00
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
for (; *low <= *high; (*low) += (unsigned int)lstep) {
|
2001-07-10 04:36:47 +08:00
|
|
|
add_next_index_stringl(return_value, low, 1, 1);
|
2002-11-14 13:46:10 +08:00
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
} else {
|
|
|
|
add_next_index_stringl(return_value, low, 1, 1);
|
|
|
|
}
|
|
|
|
} else if (Z_TYPE_P(zlow) == IS_DOUBLE || Z_TYPE_P(zhigh) == IS_DOUBLE || is_step_double) {
|
|
|
|
double low, high;
|
|
|
|
double_str:
|
2002-12-26 04:02:03 +08:00
|
|
|
convert_to_double(zlow);
|
|
|
|
convert_to_double(zhigh);
|
2002-12-21 01:16:31 +08:00
|
|
|
low = Z_DVAL_P(zlow);
|
|
|
|
high = Z_DVAL_P(zhigh);
|
|
|
|
|
|
|
|
if (low > high) { /* Negative steps */
|
|
|
|
if (low - high < step || step <= 0) {
|
|
|
|
err = 1;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
for (; low >= high; low -= step) {
|
|
|
|
add_next_index_double(return_value, low);
|
|
|
|
}
|
|
|
|
} else if (high > low) { /* Positive steps */
|
|
|
|
if (high - low < step || step <= 0) {
|
|
|
|
err = 1;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
for (; low <= high; low += step) {
|
|
|
|
add_next_index_double(return_value, low);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
add_next_index_double(return_value, low);
|
2001-07-10 04:36:47 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int low, high;
|
2002-12-21 01:16:31 +08:00
|
|
|
long lstep;
|
|
|
|
long_str:
|
2002-12-26 04:02:03 +08:00
|
|
|
convert_to_long(zlow);
|
|
|
|
convert_to_long(zhigh);
|
2002-11-14 13:46:10 +08:00
|
|
|
low = Z_LVAL_P(zlow);
|
|
|
|
high = Z_LVAL_P(zhigh);
|
2002-12-21 01:16:31 +08:00
|
|
|
lstep = (long) step;
|
|
|
|
|
2002-11-14 13:46:10 +08:00
|
|
|
if (low > high) { /* Negative steps */
|
2002-12-21 01:16:31 +08:00
|
|
|
if (low - high < lstep || lstep <= 0) {
|
|
|
|
err = 1;
|
|
|
|
goto err;
|
2002-11-23 19:27:56 +08:00
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
for (; low >= high; low -= lstep) {
|
2001-07-10 04:36:47 +08:00
|
|
|
add_next_index_long(return_value, low);
|
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
} else if (high > low) { /* Positive steps */
|
|
|
|
if (high - low < lstep || lstep <= 0) {
|
|
|
|
err = 1;
|
|
|
|
goto err;
|
2002-11-23 19:27:56 +08:00
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
for (; low <= high; low += lstep) {
|
2001-07-10 04:36:47 +08:00
|
|
|
add_next_index_long(return_value, low);
|
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
} else {
|
|
|
|
add_next_index_long(return_value, low);
|
2001-07-10 04:36:47 +08:00
|
|
|
}
|
|
|
|
}
|
2002-12-21 01:16:31 +08:00
|
|
|
err:
|
|
|
|
if (err) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "step exceeds the specified range");
|
|
|
|
zval_dtor(return_value);
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2002-06-10 20:21:58 +08:00
|
|
|
static void array_data_shuffle(zval *array TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
2002-06-10 10:28:32 +08:00
|
|
|
Bucket **elems, *temp;
|
|
|
|
HashTable *hash;
|
2002-08-16 03:09:05 +08:00
|
|
|
int j, n_elems, rnd_idx, n_left;
|
2002-06-10 10:28:32 +08:00
|
|
|
|
|
|
|
n_elems = zend_hash_num_elements(Z_ARRVAL_P(array));
|
2002-06-10 20:21:58 +08:00
|
|
|
|
2002-06-10 10:28:32 +08:00
|
|
|
if (n_elems <= 1) {
|
2002-06-10 20:21:58 +08:00
|
|
|
return;
|
2002-06-10 10:28:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
elems = (Bucket **)emalloc(n_elems * sizeof(Bucket *));
|
|
|
|
hash = Z_ARRVAL_P(array);
|
|
|
|
n_left = n_elems;
|
|
|
|
|
|
|
|
for (j = 0, temp = hash->pListHead; temp; temp = temp->pListNext)
|
|
|
|
elems[j++] = temp;
|
|
|
|
while (--n_left) {
|
|
|
|
rnd_idx = php_rand(TSRMLS_C);
|
2002-08-16 03:09:05 +08:00
|
|
|
RAND_RANGE(rnd_idx, 0, n_left, PHP_RAND_MAX);
|
|
|
|
if (rnd_idx != n_left) {
|
|
|
|
temp = elems[n_left];
|
|
|
|
elems[n_left] = elems[rnd_idx];
|
2002-06-10 10:28:32 +08:00
|
|
|
elems[rnd_idx] = temp;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2002-06-10 10:28:32 +08:00
|
|
|
|
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
|
|
|
hash->pListHead = elems[0];
|
|
|
|
hash->pListTail = NULL;
|
|
|
|
hash->pInternalPointer = hash->pListHead;
|
|
|
|
|
|
|
|
for (j = 0; j < n_elems; j++) {
|
|
|
|
if (hash->pListTail) {
|
|
|
|
hash->pListTail->pListNext = elems[j];
|
|
|
|
}
|
|
|
|
elems[j]->pListLast = hash->pListTail;
|
|
|
|
elems[j]->pListNext = NULL;
|
|
|
|
hash->pListTail = elems[j];
|
|
|
|
}
|
|
|
|
temp = hash->pListHead;
|
|
|
|
j = 0;
|
|
|
|
while (temp != NULL) {
|
|
|
|
temp->nKeyLength = 0;
|
|
|
|
temp->h = j++;
|
|
|
|
temp = temp->pListNext;
|
|
|
|
}
|
|
|
|
hash->nNextFreeElement = n_elems;
|
|
|
|
zend_hash_rehash(hash);
|
|
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
|
|
|
|
efree(elems);
|
2002-06-10 20:21:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* {{{ proto bool shuffle(array array_arg)
|
|
|
|
Randomly shuffle the contents of an array */
|
|
|
|
PHP_FUNCTION(shuffle)
|
|
|
|
{
|
|
|
|
zval *array;
|
|
|
|
|
|
|
|
if (zend_parse_parameters(1 TSRMLS_CC, "a", &array) == FAILURE) {
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_data_shuffle(array TSRMLS_CC);
|
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2000-05-18 08:47:57 +08:00
|
|
|
/* HashTable* php_splice(HashTable *in_hash, int offset, int length,
|
|
|
|
zval ***list, int list_count, HashTable **removed) */
|
|
|
|
HashTable* php_splice(HashTable *in_hash, int offset, int length,
|
|
|
|
zval ***list, int list_count, HashTable **removed)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
HashTable *out_hash = NULL; /* Output hashtable */
|
|
|
|
int num_in, /* Number of entries in the input hashtable */
|
|
|
|
pos, /* Current position in the hashtable */
|
|
|
|
i; /* Loop counter */
|
|
|
|
Bucket *p; /* Pointer to hash bucket */
|
|
|
|
zval *entry; /* Hash entry */
|
|
|
|
|
|
|
|
/* If input hash doesn't exist, we have nothing to do */
|
|
|
|
if (!in_hash)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Get number of entries in the input hash */
|
|
|
|
num_in = zend_hash_num_elements(in_hash);
|
|
|
|
|
|
|
|
/* Clamp the offset.. */
|
|
|
|
if (offset > num_in)
|
|
|
|
offset = num_in;
|
|
|
|
else if (offset < 0 && (offset=num_in+offset) < 0)
|
|
|
|
offset = 0;
|
|
|
|
|
|
|
|
/* ..and the length */
|
2002-09-22 00:10:33 +08:00
|
|
|
if (length < 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
length = num_in-offset+length;
|
2002-09-22 00:10:33 +08:00
|
|
|
} else if (offset+length > num_in) {
|
1999-11-14 04:31:54 +08:00
|
|
|
length = num_in-offset;
|
2002-09-22 00:10:33 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Create and initialize output hash */
|
2001-06-20 00:03:35 +08:00
|
|
|
ALLOC_HASHTABLE(out_hash);
|
1999-12-22 01:14:31 +08:00
|
|
|
zend_hash_init(out_hash, 0, NULL, ZVAL_PTR_DTOR, 0);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Start at the beginning of the input hash and copy
|
|
|
|
entries to output hash until offset is reached */
|
|
|
|
for (pos=0, p=in_hash->pListHead; pos<offset && p ; pos++, p=p->pListNext) {
|
|
|
|
/* Get entry and increase reference count */
|
|
|
|
entry = *((zval **)p->pData);
|
|
|
|
entry->refcount++;
|
|
|
|
|
|
|
|
/* Update output hash depending on key type */
|
|
|
|
if (p->nKeyLength)
|
|
|
|
zend_hash_update(out_hash, p->arKey, p->nKeyLength, &entry, sizeof(zval *), NULL);
|
|
|
|
else
|
2000-06-29 04:09:04 +08:00
|
|
|
zend_hash_next_index_insert(out_hash, &entry, sizeof(zval *), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If hash for removed entries exists, go until offset+length
|
|
|
|
and copy the entries to it */
|
|
|
|
if (removed != NULL) {
|
2002-09-22 00:10:33 +08:00
|
|
|
for ( ; pos<offset+length && p; pos++, p=p->pListNext) {
|
1999-11-14 04:31:54 +08:00
|
|
|
entry = *((zval **)p->pData);
|
|
|
|
entry->refcount++;
|
|
|
|
if (p->nKeyLength)
|
|
|
|
zend_hash_update(*removed, p->arKey, p->nKeyLength, &entry, sizeof(zval *), NULL);
|
|
|
|
else
|
2000-06-29 04:09:04 +08:00
|
|
|
zend_hash_next_index_insert(*removed, &entry, sizeof(zval *), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
} else /* otherwise just skip those entries */
|
2002-09-22 00:10:33 +08:00
|
|
|
for ( ; pos<offset+length && p; pos++, p=p->pListNext);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* If there are entries to insert.. */
|
|
|
|
if (list != NULL) {
|
|
|
|
/* ..for each one, create a new zval, copy entry into it
|
|
|
|
and copy it into the output hash */
|
|
|
|
for (i=0; i<list_count; i++) {
|
|
|
|
entry = *list[i];
|
2002-03-01 20:00:05 +08:00
|
|
|
if (entry->refcount>=1000) {
|
|
|
|
zval *tmp = (zval *) emalloc(sizeof(zval));
|
|
|
|
|
|
|
|
*tmp = *entry;
|
|
|
|
zval_copy_ctor(tmp);
|
|
|
|
tmp->refcount = 1;
|
|
|
|
tmp->is_ref = 0;
|
|
|
|
entry = tmp;
|
|
|
|
} else {
|
|
|
|
entry->refcount++;
|
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
zend_hash_next_index_insert(out_hash, &entry, sizeof(zval *), NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy the remaining input hash entries to the output hash */
|
|
|
|
for ( ; p ; p=p->pListNext) {
|
|
|
|
entry = *((zval **)p->pData);
|
|
|
|
entry->refcount++;
|
|
|
|
if (p->nKeyLength)
|
|
|
|
zend_hash_update(out_hash, p->arKey, p->nKeyLength, &entry, sizeof(zval *), NULL);
|
|
|
|
else
|
|
|
|
zend_hash_next_index_insert(out_hash, &entry, sizeof(zval *), NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
zend_hash_internal_pointer_reset(out_hash);
|
|
|
|
return out_hash;
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2000-06-25 21:21:37 +08:00
|
|
|
/* {{{ proto int array_push(array stack, mixed var [, mixed ...])
|
1999-11-14 04:31:54 +08:00
|
|
|
Pushes elements onto the end of the array */
|
|
|
|
PHP_FUNCTION(array_push)
|
|
|
|
{
|
|
|
|
zval ***args, /* Function arguments array */
|
2002-06-18 21:16:33 +08:00
|
|
|
*stack, /* Input array */
|
|
|
|
*new_var; /* Variable to be pushed */
|
1999-11-14 04:31:54 +08:00
|
|
|
int i, /* Loop counter */
|
|
|
|
argc; /* Number of function arguments */
|
|
|
|
|
|
|
|
/* Get the argument count and check it */
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc < 2) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
1999-12-19 06:40:35 +08:00
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get first argument and check that it's an array */
|
|
|
|
stack = *args[0];
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(stack) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "First argument should be an array");
|
2001-02-22 01:22:26 +08:00
|
|
|
efree(args);
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For each subsequent argument, make it a reference, increase refcount,
|
|
|
|
and add it to the end of the array */
|
|
|
|
for (i=1; i<argc; i++) {
|
|
|
|
new_var = *args[i];
|
|
|
|
new_var->refcount++;
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(stack), &new_var, sizeof(zval *), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Clean up and return the number of values in the stack */
|
|
|
|
efree(args);
|
2000-11-27 21:31:21 +08:00
|
|
|
RETVAL_LONG(zend_hash_num_elements(Z_ARRVAL_P(stack)));
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ void _phpi_pop(INTERNAL_FUNCTION_PARAMETERS, int which_end) */
|
|
|
|
static void _phpi_pop(INTERNAL_FUNCTION_PARAMETERS, int off_the_end)
|
|
|
|
{
|
|
|
|
zval **stack, /* Input stack */
|
|
|
|
**val; /* Value to be popped */
|
2002-08-01 12:38:11 +08:00
|
|
|
char *key = NULL;
|
2002-08-02 01:34:31 +08:00
|
|
|
int key_len = 0;
|
2002-08-01 20:07:15 +08:00
|
|
|
ulong index;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Get the arguments and do error-checking */
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &stack) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(stack) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (zend_hash_num_elements(Z_ARRVAL_PP(stack)) == 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the first or last value and copy it into the return value */
|
|
|
|
if (off_the_end)
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_internal_pointer_end(Z_ARRVAL_PP(stack));
|
1999-11-14 04:31:54 +08:00
|
|
|
else
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_internal_pointer_reset(Z_ARRVAL_PP(stack));
|
|
|
|
zend_hash_get_current_data(Z_ARRVAL_PP(stack), (void **)&val);
|
1999-11-14 04:31:54 +08:00
|
|
|
*return_value = **val;
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
INIT_PZVAL(return_value);
|
|
|
|
|
|
|
|
/* Delete the first or last value */
|
2002-08-01 12:38:11 +08:00
|
|
|
zend_hash_get_current_key_ex(Z_ARRVAL_PP(stack), &key, &key_len, &index, 0, NULL);
|
|
|
|
zend_hash_del_key_or_index(Z_ARRVAL_PP(stack), key, key_len, index, (key) ? HASH_DEL_KEY : HASH_DEL_INDEX);
|
|
|
|
|
2002-08-01 20:06:46 +08:00
|
|
|
/* If we did a shift... re-index like it did before */
|
2002-08-02 00:39:52 +08:00
|
|
|
if (!off_the_end) {
|
2002-08-02 01:34:31 +08:00
|
|
|
int k = 0;
|
2003-02-22 21:55:11 +08:00
|
|
|
int should_rehash = 0;
|
2002-08-02 01:34:31 +08:00
|
|
|
Bucket *p = Z_ARRVAL_PP(stack)->pListHead;
|
|
|
|
while (p != NULL) {
|
2003-02-06 23:27:53 +08:00
|
|
|
if (p->nKeyLength == 0) {
|
2003-02-22 21:55:11 +08:00
|
|
|
if (p->h != k) {
|
|
|
|
p->h = k++;
|
|
|
|
should_rehash = 1;
|
|
|
|
} else {
|
|
|
|
k++;
|
|
|
|
}
|
2003-02-06 23:27:53 +08:00
|
|
|
}
|
2002-08-02 01:34:31 +08:00
|
|
|
p = p->pListNext;
|
2002-08-01 12:38:11 +08:00
|
|
|
}
|
2003-02-06 23:27:53 +08:00
|
|
|
Z_ARRVAL_PP(stack)->nNextFreeElement = k;
|
2003-02-22 21:55:11 +08:00
|
|
|
if (should_rehash) {
|
2003-02-22 18:54:26 +08:00
|
|
|
zend_hash_rehash(Z_ARRVAL_PP(stack));
|
|
|
|
}
|
2002-09-22 00:10:33 +08:00
|
|
|
} else if (!key_len) {
|
2002-08-02 01:34:31 +08:00
|
|
|
Z_ARRVAL_PP(stack)->nNextFreeElement = Z_ARRVAL_PP(stack)->nNextFreeElement - 1;
|
2002-08-01 12:38:11 +08:00
|
|
|
}
|
2003-02-04 00:57:02 +08:00
|
|
|
|
|
|
|
zend_hash_internal_pointer_reset(Z_ARRVAL_PP(stack));
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ proto mixed array_pop(array stack)
|
|
|
|
Pops an element off the end of the array */
|
|
|
|
PHP_FUNCTION(array_pop)
|
|
|
|
{
|
|
|
|
_phpi_pop(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto mixed array_shift(array stack)
|
|
|
|
Pops an element off the beginning of the array */
|
|
|
|
PHP_FUNCTION(array_shift)
|
|
|
|
{
|
|
|
|
_phpi_pop(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2000-06-25 21:21:37 +08:00
|
|
|
/* {{{ proto int array_unshift(array stack, mixed var [, mixed ...])
|
1999-11-14 04:31:54 +08:00
|
|
|
Pushes elements onto the beginning of the array */
|
|
|
|
PHP_FUNCTION(array_unshift)
|
|
|
|
{
|
|
|
|
zval ***args, /* Function arguments array */
|
|
|
|
*stack; /* Input stack */
|
|
|
|
HashTable *new_hash; /* New hashtable for the stack */
|
|
|
|
int argc; /* Number of function arguments */
|
|
|
|
|
|
|
|
|
|
|
|
/* Get the argument count and check it */
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc < 2) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
1999-12-19 06:40:35 +08:00
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get first argument and check that it's an array */
|
|
|
|
stack = *args[0];
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(stack) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be an array");
|
2001-02-22 01:22:26 +08:00
|
|
|
efree(args);
|
1999-11-14 04:31:54 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Use splice to insert the elements at the beginning. Destroy old
|
|
|
|
hashtable and replace it with new one */
|
2000-11-27 21:31:21 +08:00
|
|
|
new_hash = php_splice(Z_ARRVAL_P(stack), 0, 0, &args[1], argc-1, NULL);
|
|
|
|
zend_hash_destroy(Z_ARRVAL_P(stack));
|
|
|
|
efree(Z_ARRVAL_P(stack));
|
|
|
|
Z_ARRVAL_P(stack) = new_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Clean up and return the number of elements in the stack */
|
|
|
|
efree(args);
|
2000-11-27 21:31:21 +08:00
|
|
|
RETVAL_LONG(zend_hash_num_elements(Z_ARRVAL_P(stack)));
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_splice(array input, int offset [, int length [, array replacement]])
|
2000-02-24 16:39:02 +08:00
|
|
|
Removes the elements designated by offset and length and replace them with supplied array */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(array_splice)
|
|
|
|
{
|
|
|
|
zval ***args, /* Function arguments array */
|
|
|
|
*array, /* Input array */
|
|
|
|
***repl = NULL; /* Replacement elements */
|
|
|
|
HashTable *new_hash = NULL; /* Output array's hash */
|
|
|
|
Bucket *p; /* Bucket used for traversing hash */
|
|
|
|
int argc, /* Number of function arguments */
|
|
|
|
i,
|
|
|
|
offset,
|
|
|
|
length,
|
|
|
|
repl_num = 0; /* Number of replacement elements */
|
|
|
|
|
|
|
|
/* Get the argument count and check it */
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc < 2 || argc > 4) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
1999-12-19 06:40:35 +08:00
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get first argument and check that it's an array */
|
|
|
|
array = *args[0];
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_P(array) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the next two arguments. If length is omitted,
|
|
|
|
it's assumed to be until the end of the array */
|
|
|
|
convert_to_long_ex(args[1]);
|
2000-11-27 21:31:21 +08:00
|
|
|
offset = Z_LVAL_PP(args[1]);
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc > 2) {
|
|
|
|
convert_to_long_ex(args[2]);
|
2000-11-27 21:31:21 +08:00
|
|
|
length = Z_LVAL_PP(args[2]);
|
1999-11-14 04:31:54 +08:00
|
|
|
} else
|
2000-11-27 21:31:21 +08:00
|
|
|
length = zend_hash_num_elements(Z_ARRVAL_P(array));
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
if (argc == 4) {
|
|
|
|
/* Make sure the last argument, if passed, is an array */
|
|
|
|
convert_to_array_ex(args[3]);
|
|
|
|
|
|
|
|
/* Create the array of replacement elements */
|
2000-11-27 21:31:21 +08:00
|
|
|
repl_num = zend_hash_num_elements(Z_ARRVAL_PP(args[3]));
|
1999-11-14 04:31:54 +08:00
|
|
|
repl = (zval ***)emalloc(repl_num * sizeof(zval **));
|
2000-11-27 21:31:21 +08:00
|
|
|
for (p=Z_ARRVAL_PP(args[3])->pListHead, i=0; p; p=p->pListNext, i++) {
|
1999-11-14 04:31:54 +08:00
|
|
|
repl[i] = ((zval **)p->pData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize return value */
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
/* Perform splice */
|
2000-11-27 21:31:21 +08:00
|
|
|
new_hash = php_splice(Z_ARRVAL_P(array), offset, length,
|
1999-11-14 04:31:54 +08:00
|
|
|
repl, repl_num,
|
2000-11-27 21:31:21 +08:00
|
|
|
&Z_ARRVAL_P(return_value));
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Replace input array's hashtable with the new one */
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_destroy(Z_ARRVAL_P(array));
|
|
|
|
efree(Z_ARRVAL_P(array));
|
|
|
|
Z_ARRVAL_P(array) = new_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Clean up */
|
|
|
|
if (argc == 4)
|
|
|
|
efree(repl);
|
|
|
|
efree(args);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_slice(array input, int offset [, int length])
|
|
|
|
Returns elements specified by offset and length */
|
|
|
|
PHP_FUNCTION(array_slice)
|
|
|
|
{
|
|
|
|
zval **input, /* Input array */
|
|
|
|
**offset, /* Offset to get elements from */
|
|
|
|
**length, /* How many elements to get */
|
|
|
|
**entry; /* An array entry */
|
|
|
|
int offset_val, /* Value of the offset argument */
|
|
|
|
length_val, /* Value of the length argument */
|
|
|
|
num_in, /* Number of elements in the input array */
|
|
|
|
pos, /* Current position in the array */
|
|
|
|
argc; /* Number of function arguments */
|
|
|
|
|
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
1999-11-14 04:31:54 +08:00
|
|
|
ulong num_key;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition hpos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
|
|
|
|
/* Get the arguments and do error-checking */
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
1999-12-19 06:40:35 +08:00
|
|
|
if (argc < 2 || argc > 3 || zend_get_parameters_ex(argc, &input, &offset, &length)) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure offset and length are integers and assume
|
|
|
|
we want all entries from offset to the end if length
|
|
|
|
is not passed */
|
|
|
|
convert_to_long_ex(offset);
|
2000-11-27 21:31:21 +08:00
|
|
|
offset_val = Z_LVAL_PP(offset);
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc == 3) {
|
|
|
|
convert_to_long_ex(length);
|
2000-11-27 21:31:21 +08:00
|
|
|
length_val = Z_LVAL_PP(length);
|
2002-09-22 00:10:33 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
length_val = zend_hash_num_elements(Z_ARRVAL_PP(input));
|
2002-09-22 00:10:33 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Initialize returned array */
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
/* Get number of entries in the input hash */
|
2000-11-27 21:31:21 +08:00
|
|
|
num_in = zend_hash_num_elements(Z_ARRVAL_PP(input));
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Clamp the offset.. */
|
|
|
|
if (offset_val > num_in)
|
|
|
|
return;
|
|
|
|
else if (offset_val < 0 && (offset_val=num_in+offset_val) < 0)
|
|
|
|
offset_val = 0;
|
|
|
|
|
|
|
|
/* ..and the length */
|
2002-09-22 00:10:33 +08:00
|
|
|
if (length_val < 0) {
|
1999-11-14 04:31:54 +08:00
|
|
|
length_val = num_in-offset_val+length_val;
|
2002-09-22 00:10:33 +08:00
|
|
|
} else if (offset_val+length_val > num_in) {
|
1999-11-14 04:31:54 +08:00
|
|
|
length_val = num_in-offset_val;
|
2002-09-22 00:10:33 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
if (length_val == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Start at the beginning and go until we hit offset */
|
|
|
|
pos = 0;
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &hpos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (pos < offset_val &&
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&entry, &hpos) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
pos++;
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &hpos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Copy elements from input array to the one that's returned */
|
2002-09-22 00:10:33 +08:00
|
|
|
while (pos < offset_val+length_val &&
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&entry, &hpos) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
(*entry)->refcount++;
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(Z_ARRVAL_PP(input), &string_key, &string_key_len, &num_key, 0, &hpos)) {
|
1999-11-14 04:31:54 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), string_key, string_key_len,
|
1999-11-14 04:31:54 +08:00
|
|
|
entry, sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HASH_KEY_IS_LONG:
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(return_value),
|
1999-11-14 04:31:54 +08:00
|
|
|
entry, sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pos++;
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &hpos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2002-09-11 06:36:43 +08:00
|
|
|
PHPAPI int php_array_merge(HashTable *dest, HashTable *src, int recursive TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
2000-05-31 01:03:56 +08:00
|
|
|
zval **src_entry,
|
|
|
|
**dest_entry;
|
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
2000-05-31 01:03:56 +08:00
|
|
|
ulong num_key;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
2000-05-31 01:03:56 +08:00
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(src, &pos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(src, (void **)&src_entry, &pos) == SUCCESS) {
|
2001-03-17 03:29:23 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(src, &string_key, &string_key_len, &num_key, 0, &pos)) {
|
2000-05-31 01:03:56 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
|
|
|
if (recursive &&
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_find(dest, string_key, string_key_len,
|
2000-05-31 01:03:56 +08:00
|
|
|
(void **)&dest_entry) == SUCCESS) {
|
2002-09-11 02:34:16 +08:00
|
|
|
if (*src_entry == *dest_entry) {
|
2003-01-25 00:29:40 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "recursion detected");
|
2002-09-11 02:34:16 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2002-11-15 10:16:41 +08:00
|
|
|
SEPARATE_ZVAL(dest_entry);
|
2000-05-31 01:03:56 +08:00
|
|
|
convert_to_array_ex(dest_entry);
|
|
|
|
convert_to_array_ex(src_entry);
|
2002-09-11 02:34:16 +08:00
|
|
|
if (!php_array_merge(Z_ARRVAL_PP(dest_entry),
|
2002-09-11 06:36:43 +08:00
|
|
|
Z_ARRVAL_PP(src_entry), recursive TSRMLS_CC))
|
2002-09-11 02:34:16 +08:00
|
|
|
return 0;
|
2000-05-31 01:03:56 +08:00
|
|
|
} else {
|
|
|
|
(*src_entry)->refcount++;
|
|
|
|
|
|
|
|
zend_hash_update(dest, string_key, strlen(string_key)+1,
|
|
|
|
src_entry, sizeof(zval *), NULL);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HASH_KEY_IS_LONG:
|
|
|
|
(*src_entry)->refcount++;
|
|
|
|
zend_hash_next_index_insert(dest, src_entry, sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(src, &pos);
|
2000-05-31 01:03:56 +08:00
|
|
|
}
|
2002-09-11 02:34:16 +08:00
|
|
|
|
|
|
|
return 1;
|
2000-05-31 01:03:56 +08:00
|
|
|
}
|
|
|
|
|
2000-12-11 13:36:24 +08:00
|
|
|
static void php_array_merge_wrapper(INTERNAL_FUNCTION_PARAMETERS, int recursive)
|
2000-05-31 01:03:56 +08:00
|
|
|
{
|
|
|
|
zval ***args = NULL;
|
1999-11-14 04:31:54 +08:00
|
|
|
int argc,
|
|
|
|
i;
|
|
|
|
|
|
|
|
/* Get the argument count and check it */
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
2002-06-29 23:39:41 +08:00
|
|
|
if (argc < 1) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
1999-12-19 06:40:35 +08:00
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
for (i=0; i<argc; i++) {
|
2002-04-24 23:01:53 +08:00
|
|
|
SEPARATE_ZVAL(args[i]);
|
2000-05-31 01:03:56 +08:00
|
|
|
convert_to_array_ex(args[i]);
|
2002-09-11 06:36:43 +08:00
|
|
|
php_array_merge(Z_ARRVAL_P(return_value), Z_ARRVAL_PP(args[i]), recursive TSRMLS_CC);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
efree(args);
|
|
|
|
}
|
2000-05-31 01:03:56 +08:00
|
|
|
|
|
|
|
|
2000-07-05 04:31:54 +08:00
|
|
|
/* {{{ proto array array_merge(array arr1, array arr2 [, array ...])
|
2000-05-31 01:03:56 +08:00
|
|
|
Merges elements from passed arrays into one array */
|
|
|
|
PHP_FUNCTION(array_merge)
|
|
|
|
{
|
2000-12-11 13:36:24 +08:00
|
|
|
php_array_merge_wrapper(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
|
2000-05-31 01:03:56 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2000-07-05 04:31:54 +08:00
|
|
|
/* {{{ proto array array_merge_recursive(array arr1, array arr2 [, array ...])
|
2000-05-31 01:03:56 +08:00
|
|
|
Recursively merges elements from passed arrays into one array */
|
|
|
|
PHP_FUNCTION(array_merge_recursive)
|
|
|
|
{
|
2000-12-11 13:36:24 +08:00
|
|
|
php_array_merge_wrapper(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
|
2000-05-31 01:03:56 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_keys(array input [, mixed search_value])
|
2000-02-24 16:39:02 +08:00
|
|
|
Return just the keys from the input array, optionally only for the specified search_value */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(array_keys)
|
|
|
|
{
|
|
|
|
zval **input, /* Input array */
|
|
|
|
**search_value, /* Value to search for */
|
|
|
|
**entry, /* An entry in the input array */
|
|
|
|
res, /* Result of comparison */
|
|
|
|
*new_val; /* New value */
|
|
|
|
int add_key; /* Flag to indicate whether a key should be added */
|
|
|
|
char *string_key; /* String key */
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
1999-11-14 04:31:54 +08:00
|
|
|
ulong num_key; /* Numeric key */
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
search_value = NULL;
|
|
|
|
|
|
|
|
/* Get arguments and do error-checking */
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() < 1 || ZEND_NUM_ARGS() > 2 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &input, &search_value) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize return array */
|
|
|
|
array_init(return_value);
|
|
|
|
add_key = 1;
|
|
|
|
|
|
|
|
/* Go through input array and add keys to the return array */
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &pos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&entry, &pos) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
if (search_value != NULL) {
|
2002-06-18 21:16:33 +08:00
|
|
|
is_equal_function(&res, *search_value, *entry TSRMLS_CC);
|
1999-11-14 04:31:54 +08:00
|
|
|
add_key = zval_is_true(&res);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (add_key) {
|
|
|
|
MAKE_STD_ZVAL(new_val);
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(Z_ARRVAL_PP(input), &string_key, &string_key_len, &num_key, 1, &pos)) {
|
1999-11-14 04:31:54 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE_P(new_val) = IS_STRING;
|
|
|
|
Z_STRVAL_P(new_val) = string_key;
|
2001-03-17 03:29:23 +08:00
|
|
|
Z_STRLEN_P(new_val) = string_key_len-1;
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(return_value), &new_val,
|
1999-11-14 04:31:54 +08:00
|
|
|
sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HASH_KEY_IS_LONG:
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE_P(new_val) = IS_LONG;
|
|
|
|
Z_LVAL_P(new_val) = num_key;
|
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(return_value), &new_val,
|
1999-11-14 04:31:54 +08:00
|
|
|
sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_values(array input)
|
|
|
|
Return just the values from the input array */
|
|
|
|
PHP_FUNCTION(array_values)
|
|
|
|
{
|
|
|
|
zval **input, /* Input array */
|
|
|
|
**entry; /* An entry in the input array */
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Get arguments and do error-checking */
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(ZEND_NUM_ARGS(), &input) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize return array */
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
/* Go through input array and add values to the return array */
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &pos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&entry, &pos) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
(*entry)->refcount++;
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(return_value), entry,
|
1999-11-14 04:31:54 +08:00
|
|
|
sizeof(zval *), NULL);
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_count_values(array input)
|
2000-08-18 20:28:28 +08:00
|
|
|
Return the value as key and the frequency of that value in input as value */
|
1999-11-14 04:31:54 +08:00
|
|
|
PHP_FUNCTION(array_count_values)
|
|
|
|
{
|
|
|
|
zval **input, /* Input array */
|
|
|
|
**entry; /* An entry in the input array */
|
2002-06-18 21:16:33 +08:00
|
|
|
zval **tmp;
|
1999-11-14 04:31:54 +08:00
|
|
|
HashTable *myht;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Get arguments and do error-checking */
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &input) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize return array */
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
/* Go through input array and add values to the return array */
|
2000-11-27 21:31:21 +08:00
|
|
|
myht = Z_ARRVAL_PP(input);
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(myht, &pos);
|
|
|
|
while (zend_hash_get_current_data_ex(myht, (void **)&entry, &pos) == SUCCESS) {
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(entry) == IS_LONG) {
|
|
|
|
if (zend_hash_index_find(Z_ARRVAL_P(return_value),
|
|
|
|
Z_LVAL_PP(entry),
|
1999-11-14 04:31:54 +08:00
|
|
|
(void**)&tmp) == FAILURE) {
|
|
|
|
zval *data;
|
|
|
|
MAKE_STD_ZVAL(data);
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE_P(data) = IS_LONG;
|
|
|
|
Z_LVAL_P(data) = 1;
|
2001-08-12 01:03:37 +08:00
|
|
|
zend_hash_index_update(Z_ARRVAL_P(return_value), Z_LVAL_PP(entry), &data, sizeof(data), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_LVAL_PP(tmp)++;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2000-11-27 21:31:21 +08:00
|
|
|
} else if (Z_TYPE_PP(entry) == IS_STRING) {
|
|
|
|
if (zend_hash_find(Z_ARRVAL_P(return_value),
|
|
|
|
Z_STRVAL_PP(entry),
|
|
|
|
Z_STRLEN_PP(entry)+1,
|
1999-11-14 04:31:54 +08:00
|
|
|
(void**)&tmp) == FAILURE) {
|
|
|
|
zval *data;
|
|
|
|
MAKE_STD_ZVAL(data);
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE_P(data) = IS_LONG;
|
|
|
|
Z_LVAL_P(data) = 1;
|
2001-08-12 01:03:37 +08:00
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), Z_STRVAL_PP(entry), Z_STRLEN_PP(entry) + 1, &data, sizeof(data), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
} else {
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_LVAL_PP(tmp)++;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Can only count STRING and INTEGER values!");
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(myht, &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2000-09-13 03:04:50 +08:00
|
|
|
/* {{{ proto array array_reverse(array input [, bool preserve keys])
|
1999-11-14 04:31:54 +08:00
|
|
|
Return input as a new array with the order of the entries reversed */
|
|
|
|
PHP_FUNCTION(array_reverse)
|
|
|
|
{
|
2000-09-13 03:03:59 +08:00
|
|
|
zval **input, /* Input array */
|
|
|
|
**z_preserve_keys, /* Flag: whether to preserve keys */
|
|
|
|
**entry; /* An entry in the input array */
|
1999-11-14 04:31:54 +08:00
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
1999-11-14 04:31:54 +08:00
|
|
|
ulong num_key;
|
2000-09-13 03:03:59 +08:00
|
|
|
zend_bool preserve_keys = 0;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Get arguments and do error-checking */
|
2000-09-13 03:03:59 +08:00
|
|
|
if (ZEND_NUM_ARGS() < 1 || ZEND_NUM_ARGS() > 2 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &input, &z_preserve_keys) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-09-13 03:03:59 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
2000-09-13 03:03:59 +08:00
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() > 1) {
|
|
|
|
convert_to_boolean_ex(z_preserve_keys);
|
2000-09-20 02:19:02 +08:00
|
|
|
preserve_keys = Z_BVAL_PP(z_preserve_keys);
|
2000-09-13 03:03:59 +08:00
|
|
|
}
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Initialize return array */
|
|
|
|
array_init(return_value);
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_end_ex(Z_ARRVAL_PP(input), &pos);
|
2002-09-22 00:10:33 +08:00
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&entry, &pos) == SUCCESS) {
|
1999-11-14 04:31:54 +08:00
|
|
|
(*entry)->refcount++;
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(Z_ARRVAL_PP(input), &string_key, &string_key_len, &num_key, 0, &pos)) {
|
1999-11-14 04:31:54 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), string_key, string_key_len,
|
1999-11-14 04:31:54 +08:00
|
|
|
entry, sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HASH_KEY_IS_LONG:
|
2000-09-13 03:03:59 +08:00
|
|
|
if (preserve_keys)
|
|
|
|
zend_hash_index_update(Z_ARRVAL_P(return_value), num_key,
|
|
|
|
entry, sizeof(zval *), NULL);
|
|
|
|
else
|
|
|
|
zend_hash_next_index_insert(Z_ARRVAL_P(return_value),
|
|
|
|
entry, sizeof(zval *), NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_backwards_ex(Z_ARRVAL_PP(input), &pos);
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_pad(array input, int pad_size, mixed pad_value)
|
|
|
|
Returns a copy of input array padded with pad_value to size pad_size */
|
|
|
|
PHP_FUNCTION(array_pad)
|
|
|
|
{
|
|
|
|
zval **input; /* Input array */
|
|
|
|
zval **pad_size; /* Size to pad to */
|
|
|
|
zval **pad_value; /* Padding value obviously */
|
|
|
|
zval ***pads; /* Array to pass to splice */
|
|
|
|
HashTable *new_hash; /* Return value from splice */
|
|
|
|
int input_size; /* Size of the input array */
|
|
|
|
int pad_size_abs; /* Absolute value of pad_size */
|
|
|
|
int num_pads; /* How many pads do we need */
|
|
|
|
int do_pad; /* Whether we should do padding at all */
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Get arguments and do error-checking */
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 3 || zend_get_parameters_ex(3, &input, &pad_size, &pad_value) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure arguments are of the proper type */
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-11-14 04:31:54 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
convert_to_long_ex(pad_size);
|
|
|
|
|
|
|
|
/* Do some initial calculations */
|
2000-11-27 21:31:21 +08:00
|
|
|
input_size = zend_hash_num_elements(Z_ARRVAL_PP(input));
|
|
|
|
pad_size_abs = abs(Z_LVAL_PP(pad_size));
|
1999-11-14 04:31:54 +08:00
|
|
|
do_pad = (input_size >= pad_size_abs) ? 0 : 1;
|
|
|
|
|
|
|
|
/* Copy the original array */
|
|
|
|
*return_value = **input;
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
|
|
|
|
/* If no need to pad, no need to continue */
|
|
|
|
if (!do_pad)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Populate the pads array */
|
|
|
|
num_pads = pad_size_abs - input_size;
|
|
|
|
pads = (zval ***)emalloc(num_pads * sizeof(zval **));
|
|
|
|
for (i = 0; i < num_pads; i++)
|
|
|
|
pads[i] = pad_value;
|
|
|
|
|
|
|
|
/* Pad on the right or on the left */
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_LVAL_PP(pad_size) > 0)
|
|
|
|
new_hash = php_splice(Z_ARRVAL_P(return_value), input_size, 0, pads, num_pads, NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
else
|
2000-11-27 21:31:21 +08:00
|
|
|
new_hash = php_splice(Z_ARRVAL_P(return_value), 0, 0, pads, num_pads, NULL);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
|
|
|
|
/* Copy the result hash into return value */
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_destroy(Z_ARRVAL_P(return_value));
|
|
|
|
efree(Z_ARRVAL_P(return_value));
|
|
|
|
Z_ARRVAL_P(return_value) = new_hash;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
|
|
|
/* Clean up */
|
|
|
|
efree(pads);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
1999-11-21 20:37:53 +08:00
|
|
|
/* {{{ proto array array_flip(array input)
|
1999-11-21 22:06:30 +08:00
|
|
|
Return array with key <-> value flipped */
|
1999-11-21 20:37:53 +08:00
|
|
|
PHP_FUNCTION(array_flip)
|
|
|
|
{
|
|
|
|
zval **array, **entry, *data;
|
|
|
|
HashTable *target_hash;
|
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint str_key_len;
|
1999-11-21 20:37:53 +08:00
|
|
|
ulong num_key;
|
2001-03-12 16:07:00 +08:00
|
|
|
HashPosition pos;
|
|
|
|
|
2000-06-06 03:47:54 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
1999-11-21 20:37:53 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
1999-11-21 20:37:53 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
|
2001-03-12 16:07:00 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(target_hash, &pos);
|
|
|
|
while (zend_hash_get_current_data_ex(target_hash, (void **)&entry, &pos) == SUCCESS) {
|
1999-11-21 20:37:53 +08:00
|
|
|
MAKE_STD_ZVAL(data);
|
2001-03-12 16:07:00 +08:00
|
|
|
switch (zend_hash_get_current_key_ex(target_hash, &string_key, &str_key_len, &num_key, 1, &pos)) {
|
1999-11-21 20:37:53 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_STRVAL_P(data) = string_key;
|
2001-04-02 21:20:16 +08:00
|
|
|
Z_STRLEN_P(data) = str_key_len-1;
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE_P(data) = IS_STRING;
|
1999-11-21 20:37:53 +08:00
|
|
|
break;
|
|
|
|
case HASH_KEY_IS_LONG:
|
2000-11-27 21:31:21 +08:00
|
|
|
Z_TYPE_P(data) = IS_LONG;
|
|
|
|
Z_LVAL_P(data) = num_key;
|
1999-11-21 20:37:53 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(entry) == IS_LONG) {
|
2001-08-12 01:03:37 +08:00
|
|
|
zend_hash_index_update(Z_ARRVAL_P(return_value), Z_LVAL_PP(entry), &data, sizeof(data), NULL);
|
2000-11-27 21:31:21 +08:00
|
|
|
} else if (Z_TYPE_PP(entry) == IS_STRING) {
|
2001-08-12 01:03:37 +08:00
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), Z_STRVAL_PP(entry), Z_STRLEN_PP(entry) + 1, &data, sizeof(data), NULL);
|
1999-11-21 20:37:53 +08:00
|
|
|
} else {
|
2001-03-12 18:14:00 +08:00
|
|
|
zval_ptr_dtor(&data); /* will free also zval structure */
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Can only flip STRING and INTEGER values!");
|
1999-11-21 20:37:53 +08:00
|
|
|
}
|
|
|
|
|
2001-03-12 16:07:00 +08:00
|
|
|
zend_hash_move_forward_ex(target_hash, &pos);
|
1999-11-21 20:37:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2001-10-06 02:38:53 +08:00
|
|
|
/* {{{ proto array array_change_key_case(array input [, int case=CASE_LOWER])
|
|
|
|
Retuns an array with all string keys lowercased [or uppercased] */
|
|
|
|
PHP_FUNCTION(array_change_key_case)
|
|
|
|
{
|
|
|
|
zval **array, **entry, **to_upper;
|
|
|
|
char *string_key;
|
|
|
|
char *new_key;
|
|
|
|
uint str_key_len;
|
|
|
|
ulong num_key;
|
|
|
|
ulong change_to_upper=0;
|
|
|
|
|
|
|
|
HashPosition pos;
|
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() < 1 || ZEND_NUM_ARGS() > 2 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &array, &to_upper) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() > 1) {
|
|
|
|
convert_to_long_ex(to_upper);
|
|
|
|
change_to_upper = Z_LVAL_PP(to_upper);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Z_TYPE_PP(array) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
2001-10-06 02:38:53 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(array), &pos);
|
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(array), (void **)&entry, &pos) == SUCCESS) {
|
|
|
|
(*entry)->refcount++;
|
|
|
|
|
|
|
|
switch (zend_hash_get_current_key_ex(Z_ARRVAL_PP(array), &string_key, &str_key_len, &num_key, 0, &pos)) {
|
2002-06-18 21:16:33 +08:00
|
|
|
case HASH_KEY_IS_LONG:
|
2001-10-06 02:38:53 +08:00
|
|
|
zend_hash_index_update(Z_ARRVAL_P(return_value), num_key, entry, sizeof(entry), NULL);
|
|
|
|
break;
|
2002-06-18 21:16:33 +08:00
|
|
|
case HASH_KEY_IS_STRING:
|
2001-10-06 02:38:53 +08:00
|
|
|
new_key=estrndup(string_key,str_key_len);
|
|
|
|
if (change_to_upper)
|
|
|
|
php_strtoupper(new_key, str_key_len - 1);
|
|
|
|
else
|
|
|
|
php_strtolower(new_key, str_key_len - 1);
|
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), new_key, str_key_len, entry, sizeof(entry), NULL);
|
|
|
|
efree(new_key);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(array), &pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2000-06-11 02:58:45 +08:00
|
|
|
/* {{{ proto array array_unique(array input)
|
|
|
|
Removes duplicate values from array */
|
|
|
|
PHP_FUNCTION(array_unique)
|
|
|
|
{
|
|
|
|
zval **array;
|
|
|
|
HashTable *target_hash;
|
|
|
|
Bucket *p;
|
2002-06-19 03:37:59 +08:00
|
|
|
struct bucketindex {
|
|
|
|
Bucket *b;
|
|
|
|
unsigned int i;
|
|
|
|
};
|
|
|
|
struct bucketindex *arTmp, *cmpdata, *lastkept;
|
|
|
|
unsigned int i;
|
2000-06-11 02:58:45 +08:00
|
|
|
|
2001-09-09 19:03:31 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &array) == FAILURE) {
|
2000-06-11 02:58:45 +08:00
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
target_hash = HASH_OF(*array);
|
|
|
|
if (!target_hash) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
2000-06-11 02:58:45 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy the argument array */
|
|
|
|
*return_value = **array;
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
|
|
|
|
if (target_hash->nNumOfElements <= 1) /* nothing to do */
|
2002-06-18 21:16:33 +08:00
|
|
|
return;
|
2000-06-11 02:58:45 +08:00
|
|
|
|
|
|
|
/* create and sort array with pointers to the target_hash buckets */
|
2002-06-19 03:37:59 +08:00
|
|
|
arTmp = (struct bucketindex *) pemalloc((target_hash->nNumOfElements + 1) * sizeof(struct bucketindex), target_hash->persistent);
|
2000-06-11 02:58:45 +08:00
|
|
|
if (!arTmp)
|
|
|
|
RETURN_FALSE;
|
2002-06-19 03:37:59 +08:00
|
|
|
for (i = 0, p = target_hash->pListHead; p; i++, p = p->pListNext) {
|
|
|
|
arTmp[i].b = p;
|
|
|
|
arTmp[i].i = i;
|
|
|
|
}
|
|
|
|
arTmp[i].b = NULL;
|
2002-06-18 21:16:33 +08:00
|
|
|
set_compare_func(SORT_STRING TSRMLS_CC);
|
2002-06-19 03:37:59 +08:00
|
|
|
zend_qsort((void *) arTmp, i, sizeof(struct bucketindex), array_data_compare TSRMLS_CC);
|
2000-06-11 02:58:45 +08:00
|
|
|
|
|
|
|
/* go through the sorted array and delete duplicates from the copy */
|
|
|
|
lastkept = arTmp;
|
2002-06-19 03:37:59 +08:00
|
|
|
for (cmpdata = arTmp + 1; cmpdata->b; cmpdata++) {
|
2001-09-18 05:02:53 +08:00
|
|
|
if (array_data_compare(lastkept, cmpdata TSRMLS_CC)) {
|
2002-06-18 21:16:33 +08:00
|
|
|
lastkept = cmpdata;
|
2000-06-11 02:58:45 +08:00
|
|
|
} else {
|
2002-06-19 03:37:59 +08:00
|
|
|
if (lastkept->i > cmpdata->i) {
|
|
|
|
p = lastkept->b;
|
|
|
|
lastkept = cmpdata;
|
|
|
|
} else {
|
|
|
|
p = cmpdata->b;
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
if (p->nKeyLength)
|
2002-06-18 21:16:33 +08:00
|
|
|
zend_hash_del(Z_ARRVAL_P(return_value), p->arKey, p->nKeyLength);
|
2000-06-11 02:58:45 +08:00
|
|
|
else
|
2002-06-18 21:16:33 +08:00
|
|
|
zend_hash_index_del(Z_ARRVAL_P(return_value), p->h);
|
2000-06-11 02:58:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
pefree(arTmp, target_hash->persistent);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2002-09-21 22:50:04 +08:00
|
|
|
static void php_array_intersect(INTERNAL_FUNCTION_PARAMETERS, int behavior)
|
2000-06-11 02:58:45 +08:00
|
|
|
{
|
2000-10-22 01:48:11 +08:00
|
|
|
zval ***args = NULL;
|
2000-06-11 02:58:45 +08:00
|
|
|
HashTable *hash;
|
|
|
|
int argc, i, c = 0;
|
|
|
|
Bucket ***lists, **list, ***ptrs, *p;
|
|
|
|
|
|
|
|
/* Get the argument count and check it */
|
2003-01-21 22:56:40 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
2000-06-11 02:58:45 +08:00
|
|
|
if (argc < 2) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
/* for each argument, create and sort list with pointers to the hash buckets */
|
|
|
|
lists = (Bucket ***)emalloc(argc * sizeof(Bucket **));
|
|
|
|
ptrs = (Bucket ***)emalloc(argc * sizeof(Bucket **));
|
2002-06-18 21:16:33 +08:00
|
|
|
set_compare_func(SORT_STRING TSRMLS_CC);
|
2000-06-11 02:58:45 +08:00
|
|
|
for (i=0; i<argc; i++) {
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(args[i]) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d is not an array", i+1);
|
2000-06-11 02:58:45 +08:00
|
|
|
argc = i; /* only free up to i-1 */
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
hash = HASH_OF(*args[i]);
|
|
|
|
list = (Bucket **) pemalloc((hash->nNumOfElements + 1) * sizeof(Bucket *), hash->persistent);
|
|
|
|
if (!list)
|
2000-10-22 01:48:11 +08:00
|
|
|
RETURN_FALSE;
|
2000-06-11 02:58:45 +08:00
|
|
|
lists[i] = list;
|
|
|
|
ptrs[i] = list;
|
|
|
|
for (p = hash->pListHead; p; p = p->pListNext)
|
2000-10-22 01:48:11 +08:00
|
|
|
*list++ = p;
|
2000-06-11 02:58:45 +08:00
|
|
|
*list = NULL;
|
2002-09-21 22:50:04 +08:00
|
|
|
if (behavior == INTERSECT_NORMAL) {
|
|
|
|
zend_qsort((void *) lists[i], hash->nNumOfElements, sizeof(Bucket *), array_data_compare TSRMLS_CC);
|
|
|
|
} else if (behavior == INTERSECT_ASSOC) {
|
|
|
|
zend_qsort((void *) lists[i], hash->nNumOfElements, sizeof(Bucket *), array_key_compare TSRMLS_CC);
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
}
|
2000-10-22 01:48:11 +08:00
|
|
|
|
|
|
|
/* copy the argument array */
|
|
|
|
*return_value = **args[0];
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
|
2000-06-11 02:58:45 +08:00
|
|
|
/* go through the lists and look for common values */
|
|
|
|
while (*ptrs[0]) {
|
|
|
|
for (i=1; i<argc; i++) {
|
2002-09-21 22:50:04 +08:00
|
|
|
if (behavior == INTERSECT_NORMAL) {
|
|
|
|
while (*ptrs[i] && (0 < (c = array_data_compare(ptrs[0], ptrs[i] TSRMLS_CC))))
|
|
|
|
ptrs[i]++;
|
|
|
|
} else if (behavior == INTERSECT_ASSOC) {
|
|
|
|
while (*ptrs[i] && (0 < (c = array_key_compare(ptrs[0], ptrs[i] TSRMLS_CC))))
|
|
|
|
ptrs[i]++;
|
|
|
|
if (!c && *ptrs[i]) { /* this means that ptrs[i] is not NULL so we can compare */
|
|
|
|
/* and "c==0" is from last operation */
|
|
|
|
if (array_data_compare(ptrs[0], ptrs[i] TSRMLS_CC) != 0) {
|
|
|
|
c = 1;
|
|
|
|
/* we are going to the break */
|
|
|
|
} else {
|
|
|
|
/* continue looping */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2000-10-22 01:48:11 +08:00
|
|
|
if (!*ptrs[i]) {
|
|
|
|
/* delete any values corresponding to remains of ptrs[0] */
|
2002-09-21 22:50:04 +08:00
|
|
|
/* and exit because they do not present in at least one of */
|
|
|
|
/* the other arguments */
|
2000-10-22 01:48:11 +08:00
|
|
|
for (;;) {
|
|
|
|
p = *ptrs[0]++;
|
|
|
|
if (!p)
|
|
|
|
goto out;
|
|
|
|
if (p->nKeyLength)
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_del(Z_ARRVAL_P(return_value), p->arKey, p->nKeyLength);
|
2000-10-22 01:48:11 +08:00
|
|
|
else
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_index_del(Z_ARRVAL_P(return_value), p->h);
|
2000-10-22 01:48:11 +08:00
|
|
|
}
|
|
|
|
}
|
2002-09-21 22:50:04 +08:00
|
|
|
if (c) /* here we get if not all are equal */
|
2000-10-22 01:48:11 +08:00
|
|
|
break;
|
2000-06-11 02:58:45 +08:00
|
|
|
ptrs[i]++;
|
|
|
|
}
|
|
|
|
if (c) {
|
2000-10-22 01:48:11 +08:00
|
|
|
/* Value of ptrs[0] not in all arguments, delete all entries */
|
2002-06-18 21:16:33 +08:00
|
|
|
/* with value < value of ptrs[i] */
|
2000-10-22 01:48:11 +08:00
|
|
|
for (;;) {
|
|
|
|
p = *ptrs[0];
|
2000-06-11 02:58:45 +08:00
|
|
|
if (p->nKeyLength)
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_del(Z_ARRVAL_P(return_value), p->arKey, p->nKeyLength);
|
2000-06-11 02:58:45 +08:00
|
|
|
else
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_index_del(Z_ARRVAL_P(return_value), p->h);
|
2000-06-11 02:58:45 +08:00
|
|
|
if (!*++ptrs[0])
|
2000-10-22 01:48:11 +08:00
|
|
|
goto out;
|
2002-09-21 22:50:04 +08:00
|
|
|
if (behavior == INTERSECT_NORMAL) {
|
|
|
|
if (0 <= array_data_compare(ptrs[0], ptrs[i] TSRMLS_CC))
|
|
|
|
break;
|
|
|
|
} else if (behavior == INTERSECT_ASSOC) {
|
|
|
|
/* no need of looping because indexes are unique */
|
2000-10-22 01:48:11 +08:00
|
|
|
break;
|
2002-09-21 22:50:04 +08:00
|
|
|
}
|
2000-10-22 01:48:11 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* ptrs[0] is present in all the arguments */
|
|
|
|
/* Skip all entries with same value as ptrs[0] */
|
|
|
|
for (;;) {
|
|
|
|
if (!*++ptrs[0])
|
|
|
|
goto out;
|
2002-09-21 22:50:04 +08:00
|
|
|
if (behavior == INTERSECT_NORMAL) {
|
|
|
|
if (array_data_compare(ptrs[0]-1, ptrs[0] TSRMLS_CC))
|
|
|
|
break;
|
|
|
|
} else if (behavior == INTERSECT_ASSOC) {
|
|
|
|
/* no need of looping because indexes are unique */
|
2000-10-22 01:48:11 +08:00
|
|
|
break;
|
2002-09-21 22:50:04 +08:00
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2000-10-22 01:48:11 +08:00
|
|
|
|
2000-06-11 02:58:45 +08:00
|
|
|
out:
|
|
|
|
for (i=0; i<argc; i++) {
|
2000-12-29 07:50:42 +08:00
|
|
|
hash = HASH_OF(*args[i]);
|
2000-06-11 02:58:45 +08:00
|
|
|
pefree(lists[i], hash->persistent);
|
|
|
|
}
|
|
|
|
efree(ptrs);
|
|
|
|
efree(lists);
|
|
|
|
efree(args);
|
|
|
|
}
|
2002-09-21 22:50:04 +08:00
|
|
|
|
|
|
|
/* {{{ proto array array_intersect(array arr1, array arr2 [, array ...])
|
|
|
|
Returns the entries of arr1 that have values which are present in all the other arguments */
|
|
|
|
PHP_FUNCTION(array_intersect)
|
|
|
|
{
|
|
|
|
php_array_intersect(INTERNAL_FUNCTION_PARAM_PASSTHRU, INTERSECT_NORMAL);
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
/* }}} */
|
|
|
|
|
2002-09-21 22:50:04 +08:00
|
|
|
/* {{{ proto array array_intersect_assoc(array arr1, array arr2 [, array ...])
|
|
|
|
Returns the entries of arr1 that have values which are present in all the other arguments. Keys are used to do more restrctive check */
|
|
|
|
PHP_FUNCTION(array_intersect_assoc)
|
|
|
|
{
|
|
|
|
php_array_intersect(INTERNAL_FUNCTION_PARAM_PASSTHRU, INTERSECT_ASSOC);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2002-09-12 16:20:37 +08:00
|
|
|
static void php_array_diff(INTERNAL_FUNCTION_PARAMETERS, int behavior)
|
2000-06-11 02:58:45 +08:00
|
|
|
{
|
2000-12-29 07:50:42 +08:00
|
|
|
zval ***args = NULL;
|
2000-06-11 02:58:45 +08:00
|
|
|
HashTable *hash;
|
|
|
|
int argc, i, c;
|
|
|
|
Bucket ***lists, **list, ***ptrs, *p;
|
|
|
|
|
2002-09-12 02:40:56 +08:00
|
|
|
/* Get the argument count and check it */
|
2003-01-21 22:56:40 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
2000-06-11 02:58:45 +08:00
|
|
|
if (argc < 2) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
/* for each argument, create and sort list with pointers to the hash buckets */
|
|
|
|
lists = (Bucket ***)emalloc(argc * sizeof(Bucket **));
|
|
|
|
ptrs = (Bucket ***)emalloc(argc * sizeof(Bucket **));
|
2002-06-11 00:34:25 +08:00
|
|
|
set_compare_func(SORT_STRING TSRMLS_CC);
|
2002-09-12 02:40:56 +08:00
|
|
|
for (i = 0; i < argc; i++) {
|
2000-11-27 21:31:21 +08:00
|
|
|
if (Z_TYPE_PP(args[i]) != IS_ARRAY) {
|
2002-09-12 02:40:56 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d is not an array", i + 1);
|
2000-06-11 02:58:45 +08:00
|
|
|
argc = i; /* only free up to i-1 */
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
hash = HASH_OF(*args[i]);
|
|
|
|
list = (Bucket **) pemalloc((hash->nNumOfElements + 1) * sizeof(Bucket *), hash->persistent);
|
2002-09-12 02:40:56 +08:00
|
|
|
if (!list) {
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
lists[i] = list;
|
|
|
|
ptrs[i] = list;
|
2002-09-12 02:40:56 +08:00
|
|
|
for (p = hash->pListHead; p; p = p->pListNext) {
|
|
|
|
*list++ = p;
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
*list = NULL;
|
2002-09-12 16:04:42 +08:00
|
|
|
if (behavior == DIFF_NORMAL) {
|
2002-09-12 02:13:48 +08:00
|
|
|
zend_qsort((void *) lists[i], hash->nNumOfElements, sizeof(Bucket *), array_data_compare TSRMLS_CC);
|
2002-09-12 16:04:42 +08:00
|
|
|
} else if (behavior == DIFF_ASSOC) {
|
2002-09-12 02:13:48 +08:00
|
|
|
zend_qsort((void *) lists[i], hash->nNumOfElements, sizeof(Bucket *), array_key_compare TSRMLS_CC);
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
}
|
2000-10-22 01:48:11 +08:00
|
|
|
|
|
|
|
/* copy the argument array */
|
|
|
|
*return_value = **args[0];
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
|
2000-06-11 02:58:45 +08:00
|
|
|
/* go through the lists and look for values of ptr[0]
|
2002-06-18 21:16:33 +08:00
|
|
|
that are not in the others */
|
2000-06-11 02:58:45 +08:00
|
|
|
while (*ptrs[0]) {
|
2000-12-29 07:50:42 +08:00
|
|
|
c = 1;
|
2002-09-12 02:40:56 +08:00
|
|
|
for (i = 1; i < argc; i++) {
|
2002-09-12 16:04:42 +08:00
|
|
|
if (behavior == DIFF_NORMAL) {
|
2002-09-12 02:40:56 +08:00
|
|
|
while (*ptrs[i] && (0 < (c = array_data_compare(ptrs[0], ptrs[i] TSRMLS_CC)))) {
|
2000-12-29 07:50:42 +08:00
|
|
|
ptrs[i]++;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 16:04:42 +08:00
|
|
|
} else if (behavior == DIFF_ASSOC) {
|
2002-09-12 02:40:56 +08:00
|
|
|
while (*ptrs[i] && (0 < (c = array_key_compare(ptrs[0], ptrs[i] TSRMLS_CC)))) {
|
2002-09-12 02:13:48 +08:00
|
|
|
ptrs[i]++;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 02:13:48 +08:00
|
|
|
}
|
|
|
|
if (!c) {
|
2002-09-12 16:04:42 +08:00
|
|
|
if (behavior == DIFF_NORMAL) {
|
2002-09-12 02:40:56 +08:00
|
|
|
if (*ptrs[i]) {
|
2002-09-12 02:13:48 +08:00
|
|
|
ptrs[i]++;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 02:13:48 +08:00
|
|
|
break;
|
2002-09-12 16:04:42 +08:00
|
|
|
} else if (behavior == DIFF_ASSOC) {
|
2002-09-12 02:13:48 +08:00
|
|
|
if (*ptrs[i]) {
|
|
|
|
if (array_data_compare(ptrs[0], ptrs[i] TSRMLS_CC) != 0) {
|
|
|
|
c = -1;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!c) {
|
2000-10-22 01:48:11 +08:00
|
|
|
/* ptrs[0] in one of the other arguments */
|
|
|
|
/* delete all entries with value as ptrs[0] */
|
|
|
|
for (;;) {
|
|
|
|
p = *ptrs[0];
|
2002-09-12 02:40:56 +08:00
|
|
|
if (p->nKeyLength) {
|
2000-11-27 21:31:21 +08:00
|
|
|
zend_hash_del(Z_ARRVAL_P(return_value), p->arKey, p->nKeyLength);
|
2002-09-12 02:40:56 +08:00
|
|
|
} else {
|
|
|
|
zend_hash_index_del(Z_ARRVAL_P(return_value), p->h);
|
|
|
|
}
|
|
|
|
if (!*++ptrs[0]) {
|
2000-10-22 01:48:11 +08:00
|
|
|
goto out;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 16:04:42 +08:00
|
|
|
if (behavior == DIFF_NORMAL) {
|
2002-09-12 02:40:56 +08:00
|
|
|
if (array_data_compare(ptrs[0] - 1, ptrs[0] TSRMLS_CC)) {
|
2002-09-12 02:13:48 +08:00
|
|
|
break;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 16:04:42 +08:00
|
|
|
} else if (behavior == DIFF_ASSOC) {
|
2002-09-12 02:13:48 +08:00
|
|
|
/* in this case no array_key_compare is needed */
|
2000-10-22 01:48:11 +08:00
|
|
|
break;
|
2002-09-12 02:13:48 +08:00
|
|
|
}
|
2000-10-22 01:48:11 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* ptrs[0] in none of the other arguments */
|
|
|
|
/* skip all entries with value as ptrs[0] */
|
|
|
|
for (;;) {
|
2002-09-12 02:40:56 +08:00
|
|
|
if (!*++ptrs[0]) {
|
2000-10-22 01:48:11 +08:00
|
|
|
goto out;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 16:04:42 +08:00
|
|
|
if (behavior == DIFF_NORMAL) {
|
2002-09-12 02:40:56 +08:00
|
|
|
if (array_data_compare(ptrs[0]-1, ptrs[0] TSRMLS_CC)) {
|
2002-09-12 02:13:48 +08:00
|
|
|
break;
|
2002-09-12 02:40:56 +08:00
|
|
|
}
|
2002-09-12 16:04:42 +08:00
|
|
|
} else if (behavior == DIFF_ASSOC) {
|
2002-09-12 02:13:48 +08:00
|
|
|
/* in this case no array_key_compare is needed */
|
2000-10-22 01:48:11 +08:00
|
|
|
break;
|
2002-09-12 02:13:48 +08:00
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out:
|
2002-09-12 02:40:56 +08:00
|
|
|
for (i = 0; i < argc; i++) {
|
|
|
|
hash = HASH_OF(*args[i]);
|
2000-06-11 02:58:45 +08:00
|
|
|
pefree(lists[i], hash->persistent);
|
|
|
|
}
|
|
|
|
efree(ptrs);
|
|
|
|
efree(lists);
|
|
|
|
efree(args);
|
|
|
|
}
|
2002-09-12 02:13:48 +08:00
|
|
|
|
|
|
|
/* {{{ proto array array_diff(array arr1, array arr2 [, array ...])
|
|
|
|
Returns the entries of arr1 that have values which are not present in any of the others arguments */
|
|
|
|
PHP_FUNCTION(array_diff)
|
|
|
|
{
|
2002-09-12 16:20:37 +08:00
|
|
|
php_array_diff(INTERNAL_FUNCTION_PARAM_PASSTHRU, DIFF_NORMAL);
|
2002-09-12 02:13:48 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ proto array array_diff_assoc(array arr1, array arr2 [, array ...])
|
|
|
|
Returns the entries of arr1 that have values which are not present in any of the others arguments but do additional checks whether the keys are equal */
|
|
|
|
PHP_FUNCTION(array_diff_assoc)
|
|
|
|
{
|
2002-09-12 16:20:37 +08:00
|
|
|
php_array_diff(INTERNAL_FUNCTION_PARAM_PASSTHRU, DIFF_ASSOC);
|
2002-09-12 02:13:48 +08:00
|
|
|
}
|
2000-06-11 02:58:45 +08:00
|
|
|
/* }}} */
|
|
|
|
|
2002-09-12 02:13:48 +08:00
|
|
|
|
|
|
|
|
2000-07-12 00:48:03 +08:00
|
|
|
#define MULTISORT_ORDER 0
|
|
|
|
#define MULTISORT_TYPE 1
|
|
|
|
#define MULTISORT_LAST 2
|
|
|
|
|
2001-09-18 05:02:53 +08:00
|
|
|
int multisort_compare(const void *a, const void *b TSRMLS_DC)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
Bucket** ab = *(Bucket ***)a;
|
|
|
|
Bucket** bb = *(Bucket ***)b;
|
|
|
|
int r;
|
|
|
|
int result = 0;
|
|
|
|
zval temp;
|
1999-12-12 18:50:30 +08:00
|
|
|
|
1999-11-14 04:31:54 +08:00
|
|
|
r = 0;
|
|
|
|
do {
|
2001-08-06 11:50:52 +08:00
|
|
|
set_compare_func(ARRAYG(multisort_flags)[MULTISORT_TYPE][r] TSRMLS_CC);
|
2000-07-12 00:48:03 +08:00
|
|
|
|
2001-07-30 12:58:07 +08:00
|
|
|
ARRAYG(compare_func)(&temp, *((zval **)ab[r]->pData), *((zval **)bb[r]->pData) TSRMLS_CC);
|
2000-11-27 21:31:21 +08:00
|
|
|
result = ARRAYG(multisort_flags)[MULTISORT_ORDER][r] * Z_LVAL(temp);
|
1999-11-14 04:31:54 +08:00
|
|
|
if (result != 0)
|
|
|
|
return result;
|
|
|
|
r++;
|
|
|
|
} while (ab[r] != NULL);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
1999-12-14 03:42:26 +08:00
|
|
|
#define MULTISORT_ABORT \
|
2000-07-12 00:48:03 +08:00
|
|
|
for (k = 0; k < MULTISORT_LAST; k++) \
|
|
|
|
efree(ARRAYG(multisort_flags)[k]); \
|
1999-12-14 03:42:26 +08:00
|
|
|
efree(arrays); \
|
|
|
|
efree(args); \
|
|
|
|
RETURN_FALSE;
|
|
|
|
|
2000-07-17 21:37:26 +08:00
|
|
|
/* {{{ proto bool array_multisort(array ar1 [, SORT_ASC|SORT_DESC [, SORT_REGULAR|SORT_NUMERIC|SORT_STRING]] [, array ar2 [, SORT_ASC|SORT_DESC [, SORT_REGULAR|SORT_NUMERIC|SORT_STRING]], ...])
|
1999-12-14 03:42:26 +08:00
|
|
|
Sort multiple arrays at once similar to how ORDER BY clause works in SQL */
|
|
|
|
PHP_FUNCTION(array_multisort)
|
1999-11-14 04:31:54 +08:00
|
|
|
{
|
|
|
|
zval*** args;
|
1999-12-14 03:42:26 +08:00
|
|
|
zval*** arrays;
|
1999-11-14 04:31:54 +08:00
|
|
|
Bucket*** indirect;
|
|
|
|
Bucket* p;
|
|
|
|
HashTable* hash;
|
|
|
|
int argc;
|
|
|
|
int array_size;
|
1999-12-14 03:42:26 +08:00
|
|
|
int num_arrays = 0;
|
2000-07-12 00:48:03 +08:00
|
|
|
int parse_state[MULTISORT_LAST]; /* 0 - flag not allowed
|
2002-06-18 21:16:33 +08:00
|
|
|
1 - flag allowed */
|
1999-12-14 03:42:26 +08:00
|
|
|
int sort_order = SORT_ASC;
|
2000-07-12 00:48:03 +08:00
|
|
|
int sort_type = SORT_REGULAR;
|
1999-11-14 04:31:54 +08:00
|
|
|
int i, k;
|
|
|
|
|
|
|
|
/* Get the argument count and check it */
|
2000-06-06 03:47:54 +08:00
|
|
|
argc = ZEND_NUM_ARGS();
|
1999-11-14 04:31:54 +08:00
|
|
|
if (argc < 1) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate arguments array and get the arguments, checking for errors. */
|
|
|
|
args = (zval ***)emalloc(argc * sizeof(zval **));
|
1999-12-19 06:40:35 +08:00
|
|
|
if (zend_get_parameters_array_ex(argc, args) == FAILURE) {
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-07-12 00:48:03 +08:00
|
|
|
/* Allocate space for storing pointers to input arrays and sort flags. */
|
1999-12-14 03:42:26 +08:00
|
|
|
arrays = (zval ***)ecalloc(argc, sizeof(zval **));
|
2000-07-12 00:48:03 +08:00
|
|
|
for (i = 0; i < MULTISORT_LAST; i++) {
|
|
|
|
parse_state[i] = 0;
|
|
|
|
ARRAYG(multisort_flags)[i] = (int *)ecalloc(argc, sizeof(int));
|
|
|
|
}
|
1999-12-14 03:42:26 +08:00
|
|
|
|
|
|
|
/* Here we go through the input arguments and parse them. Each one can
|
2000-07-12 00:51:04 +08:00
|
|
|
be either an array or a sort flag which follows an array. If not
|
|
|
|
specified, the sort flags defaults to SORT_ASC and SORT_REGULAR
|
|
|
|
accordingly. There can't be two sort flags of the same type after an
|
|
|
|
array, and the very first argument has to be an array.
|
1999-12-14 03:42:26 +08:00
|
|
|
*/
|
1999-11-14 04:31:54 +08:00
|
|
|
for (i = 0; i < argc; i++) {
|
2000-07-12 00:48:03 +08:00
|
|
|
if (Z_TYPE_PP(args[i]) == IS_ARRAY) {
|
2000-07-12 00:51:04 +08:00
|
|
|
/* We see the next array, so we update the sort flags of
|
|
|
|
the previous array and reset the sort flags. */
|
1999-12-14 03:42:26 +08:00
|
|
|
if (i > 0) {
|
2000-07-12 00:48:03 +08:00
|
|
|
ARRAYG(multisort_flags)[MULTISORT_ORDER][num_arrays-1] = sort_order;
|
|
|
|
ARRAYG(multisort_flags)[MULTISORT_TYPE][num_arrays-1] = sort_type;
|
1999-12-14 03:42:26 +08:00
|
|
|
sort_order = SORT_ASC;
|
2000-07-12 00:48:03 +08:00
|
|
|
sort_type = SORT_REGULAR;
|
1999-12-14 03:42:26 +08:00
|
|
|
}
|
|
|
|
arrays[num_arrays++] = args[i];
|
|
|
|
|
2000-07-12 00:51:04 +08:00
|
|
|
/* Next one may be an array or a list of sort flags. */
|
2000-07-12 00:48:03 +08:00
|
|
|
for (k = 0; k < MULTISORT_LAST; k++)
|
|
|
|
parse_state[k] = 1;
|
|
|
|
} else if (Z_TYPE_PP(args[i]) == IS_LONG) {
|
|
|
|
switch (Z_LVAL_PP(args[i])) {
|
|
|
|
case SORT_ASC:
|
|
|
|
case SORT_DESC:
|
|
|
|
/* flag allowed here */
|
|
|
|
if (parse_state[MULTISORT_ORDER] == 1) {
|
|
|
|
/* Save the flag and make sure then next arg is not the current flag. */
|
|
|
|
sort_order = Z_LVAL_PP(args[i]) == SORT_DESC ? -1 : 1;
|
|
|
|
parse_state[MULTISORT_ORDER] = 0;
|
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d is expected to be an array or sorting flag that has not already been specified", i+1);
|
2000-07-12 00:48:03 +08:00
|
|
|
MULTISORT_ABORT;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SORT_REGULAR:
|
|
|
|
case SORT_NUMERIC:
|
|
|
|
case SORT_STRING:
|
|
|
|
/* flag allowed here */
|
|
|
|
if (parse_state[MULTISORT_TYPE] == 1) {
|
|
|
|
/* Save the flag and make sure then next arg is not the current flag. */
|
|
|
|
sort_type = Z_LVAL_PP(args[i]);
|
|
|
|
parse_state[MULTISORT_TYPE] = 0;
|
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d is expected to be an array or sorting flag that has not already been specified", i + 1);
|
2000-07-12 00:48:03 +08:00
|
|
|
MULTISORT_ABORT;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d is an unknown sort flag", i + 1);
|
1999-12-14 03:42:26 +08:00
|
|
|
MULTISORT_ABORT;
|
2000-07-12 00:48:03 +08:00
|
|
|
break;
|
|
|
|
|
1999-12-14 03:42:26 +08:00
|
|
|
}
|
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d is expected to be an array or a sort flag", i + 1);
|
1999-12-14 03:42:26 +08:00
|
|
|
MULTISORT_ABORT;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
2000-07-12 00:48:03 +08:00
|
|
|
/* Take care of the last array sort flags. */
|
|
|
|
ARRAYG(multisort_flags)[MULTISORT_ORDER][num_arrays-1] = sort_order;
|
|
|
|
ARRAYG(multisort_flags)[MULTISORT_TYPE][num_arrays-1] = sort_type;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-07-12 00:51:04 +08:00
|
|
|
/* Make sure the arrays are of the same size. */
|
2000-07-12 00:48:03 +08:00
|
|
|
array_size = zend_hash_num_elements(Z_ARRVAL_PP(arrays[0]));
|
1999-12-14 03:42:26 +08:00
|
|
|
for (i = 0; i < num_arrays; i++) {
|
2000-07-12 00:48:03 +08:00
|
|
|
if (zend_hash_num_elements(Z_ARRVAL_PP(arrays[i])) != array_size) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Array sizes are inconsistent");
|
1999-12-14 03:42:26 +08:00
|
|
|
MULTISORT_ABORT;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-04-28 22:01:08 +08:00
|
|
|
/* If all arrays are empty or have only one entry,
|
|
|
|
we don't need to do anything. */
|
2000-05-18 20:29:45 +08:00
|
|
|
if (array_size <= 1) {
|
2000-07-12 00:48:03 +08:00
|
|
|
for (k = 0; k < MULTISORT_LAST; k++)
|
|
|
|
efree(ARRAYG(multisort_flags)[k]);
|
2000-04-28 22:01:08 +08:00
|
|
|
efree(arrays);
|
|
|
|
efree(args);
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
|
1999-12-14 03:42:26 +08:00
|
|
|
/* Create the indirection array. This array is of size MxN, where
|
|
|
|
M is the number of entries in each input array and N is the number
|
|
|
|
of the input arrays + 1. The last column is NULL to indicate the end
|
|
|
|
of the row.
|
|
|
|
*/
|
1999-11-14 04:31:54 +08:00
|
|
|
indirect = (Bucket ***)emalloc(array_size * sizeof(Bucket **));
|
|
|
|
for (i = 0; i < array_size; i++)
|
1999-12-14 03:42:26 +08:00
|
|
|
indirect[i] = (Bucket **)emalloc((num_arrays+1) * sizeof(Bucket *));
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-12-14 03:42:26 +08:00
|
|
|
for (i = 0; i < num_arrays; i++) {
|
1999-11-14 04:31:54 +08:00
|
|
|
k = 0;
|
2000-07-12 00:48:03 +08:00
|
|
|
for (p = Z_ARRVAL_PP(arrays[i])->pListHead; p; p = p->pListNext, k++) {
|
1999-11-14 04:31:54 +08:00
|
|
|
indirect[k][i] = p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (k = 0; k < array_size; k++)
|
1999-12-14 03:42:26 +08:00
|
|
|
indirect[k][num_arrays] = NULL;
|
1999-11-14 04:31:54 +08:00
|
|
|
|
2000-07-12 00:48:03 +08:00
|
|
|
/* Do the actual sort magic - bada-bim, bada-boom. */
|
2001-09-18 05:02:53 +08:00
|
|
|
zend_qsort(indirect, array_size, sizeof(Bucket **), multisort_compare TSRMLS_CC);
|
1999-11-14 04:31:54 +08:00
|
|
|
|
1999-12-14 03:42:26 +08:00
|
|
|
/* Restructure the arrays based on sorted indirect - this is mostly
|
2000-07-12 00:51:04 +08:00
|
|
|
taken from zend_hash_sort() function. */
|
1999-11-14 04:31:54 +08:00
|
|
|
HANDLE_BLOCK_INTERRUPTIONS();
|
1999-12-14 03:42:26 +08:00
|
|
|
for (i = 0; i < num_arrays; i++) {
|
2000-11-27 21:31:21 +08:00
|
|
|
hash = Z_ARRVAL_PP(arrays[i]);
|
1999-11-14 04:31:54 +08:00
|
|
|
hash->pListHead = indirect[0][i];;
|
|
|
|
hash->pListTail = NULL;
|
|
|
|
hash->pInternalPointer = hash->pListHead;
|
|
|
|
|
|
|
|
for (k = 0; k < array_size; k++) {
|
|
|
|
if (hash->pListTail) {
|
|
|
|
hash->pListTail->pListNext = indirect[k][i];
|
|
|
|
}
|
|
|
|
indirect[k][i]->pListLast = hash->pListTail;
|
|
|
|
indirect[k][i]->pListNext = NULL;
|
|
|
|
hash->pListTail = indirect[k][i];
|
|
|
|
}
|
|
|
|
|
|
|
|
p = hash->pListHead;
|
|
|
|
k = 0;
|
|
|
|
while (p != NULL) {
|
|
|
|
if (p->nKeyLength == 0)
|
|
|
|
p->h = k++;
|
|
|
|
p = p->pListNext;
|
|
|
|
}
|
|
|
|
hash->nNextFreeElement = array_size;
|
|
|
|
zend_hash_rehash(hash);
|
|
|
|
}
|
|
|
|
HANDLE_UNBLOCK_INTERRUPTIONS();
|
|
|
|
|
2000-07-12 00:51:04 +08:00
|
|
|
/* Clean up. */
|
1999-11-14 04:31:54 +08:00
|
|
|
for (i = 0; i < array_size; i++)
|
|
|
|
efree(indirect[i]);
|
|
|
|
efree(indirect);
|
2000-07-12 00:48:03 +08:00
|
|
|
for (k = 0; k < MULTISORT_LAST; k++)
|
|
|
|
efree(ARRAYG(multisort_flags)[k]);
|
1999-12-14 03:42:26 +08:00
|
|
|
efree(arrays);
|
1999-11-14 04:31:54 +08:00
|
|
|
efree(args);
|
1999-12-14 03:42:26 +08:00
|
|
|
RETURN_TRUE;
|
1999-11-14 04:31:54 +08:00
|
|
|
}
|
2000-03-29 19:19:01 +08:00
|
|
|
/* }}} */
|
2000-02-26 05:27:03 +08:00
|
|
|
|
2000-05-10 03:27:00 +08:00
|
|
|
|
2000-06-25 21:21:37 +08:00
|
|
|
/* {{{ proto mixed array_rand(array input [, int num_req])
|
2000-05-10 03:27:00 +08:00
|
|
|
Return key/keys for random entry/entries in the array */
|
|
|
|
PHP_FUNCTION(array_rand)
|
|
|
|
{
|
|
|
|
zval **input, **num_req;
|
2001-09-06 03:47:55 +08:00
|
|
|
long randval;
|
2000-05-10 03:27:00 +08:00
|
|
|
int num_req_val, num_avail, key_type;
|
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
2000-05-10 03:27:00 +08:00
|
|
|
ulong num_key;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
2000-05-10 03:27:00 +08:00
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() < 1 || ZEND_NUM_ARGS() > 2 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &input, &num_req) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "First argument has to be an array");
|
2000-05-10 03:27:00 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
num_avail = zend_hash_num_elements(Z_ARRVAL_PP(input));
|
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() > 1) {
|
|
|
|
convert_to_long_ex(num_req);
|
|
|
|
num_req_val = Z_LVAL_PP(num_req);
|
|
|
|
if (num_req_val <= 0 || num_req_val > num_avail) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Second argument has to be between 1 and the number of elements in the array");
|
2000-05-10 03:27:00 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
num_req_val = 1;
|
|
|
|
|
|
|
|
/* Make the return value an array only if we need to pass back more than one
|
|
|
|
result. */
|
|
|
|
if (num_req_val > 1)
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
/* We can't use zend_hash_index_find() because the array may have string keys or gaps. */
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &pos);
|
|
|
|
while (num_req_val && (key_type = zend_hash_get_current_key_ex(Z_ARRVAL_PP(input), &string_key, &string_key_len, &num_key, 0, &pos)) != HASH_KEY_NON_EXISTANT) {
|
2000-05-10 03:27:00 +08:00
|
|
|
|
2002-06-04 01:49:48 +08:00
|
|
|
randval = php_rand(TSRMLS_C);
|
2000-05-10 03:27:00 +08:00
|
|
|
|
2001-09-06 03:47:55 +08:00
|
|
|
if ((double)(randval/(PHP_RAND_MAX+1.0)) < (double)num_req_val/(double)num_avail) {
|
2000-05-10 03:27:00 +08:00
|
|
|
/* If we are returning a single result, just do it. */
|
|
|
|
if (Z_TYPE_P(return_value) != IS_ARRAY) {
|
|
|
|
if (key_type == HASH_KEY_IS_STRING) {
|
2001-03-17 03:29:23 +08:00
|
|
|
RETURN_STRINGL(string_key, string_key_len-1, 1);
|
2000-05-10 03:27:00 +08:00
|
|
|
} else {
|
|
|
|
RETURN_LONG(num_key);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Append the result to the return value. */
|
|
|
|
if (key_type == HASH_KEY_IS_STRING)
|
2001-03-17 03:29:23 +08:00
|
|
|
add_next_index_stringl(return_value, string_key, string_key_len-1, 1);
|
2000-05-10 03:27:00 +08:00
|
|
|
else
|
|
|
|
add_next_index_long(return_value, num_key);
|
|
|
|
}
|
|
|
|
num_req_val--;
|
2000-12-22 20:57:09 +08:00
|
|
|
}
|
2000-05-10 03:27:00 +08:00
|
|
|
num_avail--;
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &pos);
|
2000-10-27 22:08:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (num_req_val == num_avail) {
|
2002-06-10 20:21:58 +08:00
|
|
|
array_data_shuffle(return_value TSRMLS_CC);
|
2000-05-10 03:27:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2000-10-26 01:40:11 +08:00
|
|
|
/* {{{ proto mixed array_sum(array input)
|
|
|
|
Returns the sum of the array entries */
|
|
|
|
|
|
|
|
PHP_FUNCTION(array_sum)
|
|
|
|
{
|
|
|
|
zval **input,
|
2002-04-24 23:23:43 +08:00
|
|
|
**entry,
|
|
|
|
entry_n;
|
2000-10-26 01:40:11 +08:00
|
|
|
int argc = ZEND_NUM_ARGS();
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
2001-08-04 03:18:51 +08:00
|
|
|
double dval;
|
2000-10-26 01:40:11 +08:00
|
|
|
|
|
|
|
if (argc != 1 || zend_get_parameters_ex(argc, &input) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2000-12-23 00:31:42 +08:00
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
|
2000-12-23 00:31:42 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2000-10-26 01:40:11 +08:00
|
|
|
ZVAL_LONG(return_value, 0);
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
for (zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &pos);
|
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&entry, &pos) == SUCCESS;
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &pos)) {
|
2000-10-26 01:40:11 +08:00
|
|
|
|
|
|
|
if (Z_TYPE_PP(entry) == IS_ARRAY || Z_TYPE_PP(entry) == IS_OBJECT)
|
|
|
|
continue;
|
|
|
|
|
2002-04-24 23:23:43 +08:00
|
|
|
entry_n = **entry;
|
|
|
|
zval_copy_ctor(&entry_n);
|
|
|
|
convert_scalar_to_number(&entry_n TSRMLS_CC);
|
2000-10-26 01:40:11 +08:00
|
|
|
|
2002-04-24 23:23:43 +08:00
|
|
|
if (Z_TYPE(entry_n) == IS_LONG && Z_TYPE_P(return_value) == IS_LONG) {
|
|
|
|
dval = (double)Z_LVAL_P(return_value) + (double)Z_LVAL(entry_n);
|
2001-08-04 03:18:51 +08:00
|
|
|
if ( (double)LONG_MIN <= dval && dval <= (double)LONG_MAX ) {
|
2002-04-24 23:23:43 +08:00
|
|
|
Z_LVAL_P(return_value) += Z_LVAL(entry_n);
|
2001-08-04 03:18:51 +08:00
|
|
|
continue;
|
|
|
|
}
|
2000-10-26 01:40:11 +08:00
|
|
|
}
|
2001-08-04 03:18:51 +08:00
|
|
|
convert_to_double(return_value);
|
2002-04-24 23:23:43 +08:00
|
|
|
convert_to_double(&entry_n);
|
|
|
|
Z_DVAL_P(return_value) += Z_DVAL(entry_n);
|
2000-10-26 01:40:11 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* }}} */
|
|
|
|
|
2001-03-12 11:06:53 +08:00
|
|
|
/* {{{ proto mixed array_reduce(array input, mixed callback [, int initial])
|
2001-03-12 11:10:15 +08:00
|
|
|
Iteratively reduce the array to a single value via the callback. */
|
2001-03-12 11:06:53 +08:00
|
|
|
PHP_FUNCTION(array_reduce)
|
|
|
|
{
|
|
|
|
zval **input, **callback, **initial;
|
|
|
|
zval **args[2];
|
|
|
|
zval **operand;
|
|
|
|
zval *result = NULL;
|
|
|
|
zval *retval;
|
|
|
|
char *callback_name;
|
2001-03-17 03:29:23 +08:00
|
|
|
HashPosition pos;
|
2001-03-12 11:06:53 +08:00
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() < 2 || ZEND_NUM_ARGS() > 3 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &input, &callback, &initial) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be an array");
|
2001-03-12 11:06:53 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!zend_is_callable(*callback, 0, &callback_name)) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The second argument, '%s', should be a valid callback", callback_name);
|
2001-03-12 11:06:53 +08:00
|
|
|
efree(callback_name);
|
|
|
|
return;
|
|
|
|
}
|
2001-05-11 00:30:12 +08:00
|
|
|
efree(callback_name);
|
2001-03-12 11:06:53 +08:00
|
|
|
|
2001-08-08 00:41:33 +08:00
|
|
|
if (ZEND_NUM_ARGS() > 2) {
|
2001-03-12 11:06:53 +08:00
|
|
|
result = *initial;
|
2001-08-08 00:41:33 +08:00
|
|
|
zval_add_ref(&result);
|
|
|
|
}
|
2001-03-12 11:06:53 +08:00
|
|
|
|
|
|
|
if (zend_hash_num_elements(Z_ARRVAL_PP(input)) == 0) {
|
|
|
|
if (result) {
|
|
|
|
*return_value = *result;
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &pos);
|
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&operand, &pos) == SUCCESS) {
|
2001-03-12 11:06:53 +08:00
|
|
|
if (result) {
|
|
|
|
args[0] = &result;
|
|
|
|
args[1] = operand;
|
2001-07-30 16:24:42 +08:00
|
|
|
if (call_user_function_ex(EG(function_table), NULL, *callback, &retval, 2, args, 0, NULL TSRMLS_CC) == SUCCESS && retval) {
|
2001-08-08 00:41:33 +08:00
|
|
|
zval_ptr_dtor(&result);
|
2001-03-12 11:06:53 +08:00
|
|
|
result = retval;
|
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "An error occurred while invoking the reduction callback");
|
2001-03-12 11:06:53 +08:00
|
|
|
return;
|
|
|
|
}
|
2001-08-08 00:41:33 +08:00
|
|
|
} else {
|
2001-03-12 11:06:53 +08:00
|
|
|
result = *operand;
|
2001-08-08 00:41:33 +08:00
|
|
|
zval_add_ref(&result);
|
|
|
|
}
|
2001-03-12 11:06:53 +08:00
|
|
|
|
2001-03-17 03:29:23 +08:00
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &pos);
|
2001-03-12 11:06:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
*return_value = *result;
|
2001-08-08 00:41:33 +08:00
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
zval_ptr_dtor(&result);
|
2001-03-12 11:06:53 +08:00
|
|
|
}
|
2001-03-17 04:46:33 +08:00
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
|
|
/* {{{ proto array array_filter(array input [, mixed callback])
|
|
|
|
Filters elements from the array via the callback. */
|
|
|
|
PHP_FUNCTION(array_filter)
|
|
|
|
{
|
|
|
|
zval **input, **callback = NULL;
|
|
|
|
zval **operand;
|
|
|
|
zval **args[1];
|
|
|
|
zval *retval = NULL;
|
|
|
|
char *callback_name;
|
|
|
|
char *string_key;
|
2001-08-21 20:57:53 +08:00
|
|
|
uint string_key_len;
|
2001-03-17 04:46:33 +08:00
|
|
|
ulong num_key;
|
|
|
|
HashPosition pos;
|
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() < 1 || ZEND_NUM_ARGS() > 2 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &input, &callback) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Z_TYPE_PP(input) != IS_ARRAY) {
|
2002-11-13 21:31:33 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be an array");
|
2001-03-17 04:46:33 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() > 1) {
|
|
|
|
if (!zend_is_callable(*callback, 0, &callback_name)) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The second argument, '%s', should be a valid callback", callback_name);
|
2001-03-17 04:46:33 +08:00
|
|
|
efree(callback_name);
|
|
|
|
return;
|
|
|
|
}
|
2001-05-11 00:30:12 +08:00
|
|
|
efree(callback_name);
|
2001-03-17 04:46:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
if (zend_hash_num_elements(Z_ARRVAL_PP(input)) == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(input), &pos);
|
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_PP(input), (void **)&operand, &pos) == SUCCESS;
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(input), &pos)) {
|
|
|
|
|
|
|
|
if (callback) {
|
|
|
|
args[0] = operand;
|
2001-07-30 16:24:42 +08:00
|
|
|
if (call_user_function_ex(EG(function_table), NULL, *callback, &retval, 1, args, 0, NULL TSRMLS_CC) == SUCCESS && retval) {
|
2001-03-17 04:46:33 +08:00
|
|
|
if (!zend_is_true(retval)) {
|
|
|
|
zval_ptr_dtor(&retval);
|
|
|
|
continue;
|
|
|
|
} else
|
|
|
|
zval_ptr_dtor(&retval);
|
|
|
|
} else {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "An error occurred while invoking the filter callback");
|
2001-03-17 04:46:33 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (!zend_is_true(*operand))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
zval_add_ref(operand);
|
|
|
|
switch (zend_hash_get_current_key_ex(Z_ARRVAL_PP(input), &string_key, &string_key_len, &num_key, 0, &pos)) {
|
|
|
|
case HASH_KEY_IS_STRING:
|
|
|
|
zend_hash_update(Z_ARRVAL_P(return_value), string_key,
|
|
|
|
string_key_len, operand, sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HASH_KEY_IS_LONG:
|
|
|
|
zend_hash_index_update(Z_ARRVAL_P(return_value), num_key,
|
|
|
|
operand, sizeof(zval *), NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2001-03-12 11:06:53 +08:00
|
|
|
|
2001-03-20 05:20:02 +08:00
|
|
|
/* {{{ proto array array_map(mixed callback, array input1 [, array input2 ,...])
|
|
|
|
Applies the callback to the elements in given arrays. */
|
|
|
|
PHP_FUNCTION(array_map)
|
|
|
|
{
|
|
|
|
zval ***args = NULL;
|
|
|
|
zval ***params;
|
|
|
|
zval *callback;
|
|
|
|
zval *result, *null;
|
2001-08-24 10:58:23 +08:00
|
|
|
HashPosition *array_pos;
|
2001-03-20 05:20:02 +08:00
|
|
|
char *callback_name;
|
|
|
|
int i, k, maxlen = 0;
|
|
|
|
int *array_len;
|
|
|
|
|
|
|
|
if (ZEND_NUM_ARGS() < 2) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
|
|
|
args = (zval ***)emalloc(ZEND_NUM_ARGS() * sizeof(zval **));
|
|
|
|
if (zend_get_parameters_array_ex(ZEND_NUM_ARGS(), args) == FAILURE) {
|
|
|
|
efree(args);
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
|
|
|
|
2003-01-03 13:05:12 +08:00
|
|
|
RETVAL_NULL();
|
|
|
|
|
2001-03-20 05:20:02 +08:00
|
|
|
callback = *args[0];
|
2001-08-05 00:54:20 +08:00
|
|
|
if (Z_TYPE_P(callback) != IS_NULL) {
|
|
|
|
if (!zend_is_callable(callback, 0, &callback_name)) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument, '%s', should be either NULL or a valid callback", callback_name);
|
2001-08-05 00:54:20 +08:00
|
|
|
efree(callback_name);
|
|
|
|
efree(args);
|
|
|
|
return;
|
|
|
|
}
|
2001-03-20 05:20:02 +08:00
|
|
|
efree(callback_name);
|
|
|
|
}
|
|
|
|
|
2001-08-24 10:58:23 +08:00
|
|
|
/* Allocate array sizes and iterators. */
|
2001-03-20 05:20:02 +08:00
|
|
|
array_len = (int*)emalloc((ZEND_NUM_ARGS()-1) * sizeof(int));
|
2001-08-24 10:58:23 +08:00
|
|
|
array_pos = (HashPosition*)emalloc((ZEND_NUM_ARGS()-1) * sizeof(HashPosition));
|
2001-03-20 05:20:02 +08:00
|
|
|
|
|
|
|
/* Check that arrays are indeed arrays and calculate maximum size. */
|
|
|
|
for (i = 0; i < ZEND_NUM_ARGS()-1; i++) {
|
|
|
|
if (Z_TYPE_PP(args[i+1]) != IS_ARRAY) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Argument #%d should be an array", i + 2);
|
2001-03-20 05:20:02 +08:00
|
|
|
efree(array_len);
|
2003-01-03 13:05:12 +08:00
|
|
|
efree(array_pos);
|
2001-03-20 05:20:02 +08:00
|
|
|
efree(args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
array_len[i] = zend_hash_num_elements(Z_ARRVAL_PP(args[i+1]));
|
|
|
|
if (array_len[i] > maxlen)
|
|
|
|
maxlen = array_len[i];
|
2001-08-24 10:58:23 +08:00
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_PP(args[i+1]), &array_pos[i]);
|
2001-03-20 05:20:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Short-circuit: if no callback and only one array, just return it. */
|
|
|
|
if (Z_TYPE_P(callback) == IS_NULL && ZEND_NUM_ARGS() == 2) {
|
|
|
|
*return_value = **args[1];
|
|
|
|
zval_copy_ctor(return_value);
|
|
|
|
efree(array_len);
|
2001-08-24 10:58:23 +08:00
|
|
|
efree(array_pos);
|
2001-03-20 05:20:02 +08:00
|
|
|
efree(args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
params = (zval ***)emalloc((ZEND_NUM_ARGS()-1) * sizeof(zval **));
|
|
|
|
MAKE_STD_ZVAL(null);
|
|
|
|
ZVAL_NULL(null);
|
|
|
|
|
|
|
|
/* We iterate through all the arrays at once. */
|
|
|
|
for (k = 0; k < maxlen; k++) {
|
2001-08-29 05:50:37 +08:00
|
|
|
uint str_key_len;
|
|
|
|
ulong num_key;
|
2001-08-24 10:58:23 +08:00
|
|
|
char *str_key;
|
2002-07-08 15:33:22 +08:00
|
|
|
int key_type = 0;
|
2001-08-24 10:58:23 +08:00
|
|
|
|
2001-03-20 05:20:02 +08:00
|
|
|
/*
|
|
|
|
* If no callback, the result will be an array, consisting of current
|
|
|
|
* entries from all arrays.
|
|
|
|
*/
|
|
|
|
if (Z_TYPE_P(callback) == IS_NULL) {
|
|
|
|
MAKE_STD_ZVAL(result);
|
|
|
|
array_init(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < ZEND_NUM_ARGS()-1; i++) {
|
|
|
|
/*
|
|
|
|
* If this array still hash elements, add the current one to the
|
|
|
|
* parameter list, otherwise use null value.
|
|
|
|
*/
|
|
|
|
if (k < array_len[i]) {
|
2001-08-24 10:58:23 +08:00
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_PP(args[i+1]), (void **)¶ms[i], &array_pos[i]);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* It is safe to store only last value of key type, because
|
|
|
|
* this loop will run just once if there is only 1 array.
|
|
|
|
*/
|
|
|
|
if (ZEND_NUM_ARGS() == 2) {
|
|
|
|
key_type = zend_hash_get_current_key_ex(Z_ARRVAL_PP(args[1]), &str_key, &str_key_len, &num_key, 0, &array_pos[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(args[i+1]), &array_pos[i]);
|
2001-03-20 05:20:02 +08:00
|
|
|
} else {
|
|
|
|
if (Z_TYPE_P(callback) == IS_NULL)
|
|
|
|
zval_add_ref(&null);
|
|
|
|
params[i] = &null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Z_TYPE_P(callback) == IS_NULL)
|
|
|
|
add_next_index_zval(result, *params[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Z_TYPE_P(callback) != IS_NULL) {
|
2001-07-30 16:24:42 +08:00
|
|
|
if (!call_user_function_ex(EG(function_table), NULL, callback, &result, ZEND_NUM_ARGS()-1, params, 0, NULL TSRMLS_CC) == SUCCESS && result) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "An error occurred while invoking the map callback");
|
2001-03-20 05:20:02 +08:00
|
|
|
efree(array_len);
|
|
|
|
efree(args);
|
2001-08-24 10:58:23 +08:00
|
|
|
efree(array_pos);
|
2001-03-20 05:20:02 +08:00
|
|
|
zval_dtor(return_value);
|
|
|
|
RETURN_NULL();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-08-24 10:58:23 +08:00
|
|
|
if (ZEND_NUM_ARGS() > 2) {
|
|
|
|
add_next_index_zval(return_value, result);
|
|
|
|
} else {
|
|
|
|
if (key_type == HASH_KEY_IS_STRING)
|
|
|
|
add_assoc_zval_ex(return_value, str_key, str_key_len, result);
|
|
|
|
else
|
|
|
|
add_index_zval(return_value, num_key, result);
|
|
|
|
}
|
2001-03-20 05:20:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
zval_ptr_dtor(&null);
|
|
|
|
efree(params);
|
|
|
|
efree(array_len);
|
2001-08-24 10:58:23 +08:00
|
|
|
efree(array_pos);
|
2001-03-20 05:20:02 +08:00
|
|
|
efree(args);
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
2001-11-12 02:28:43 +08:00
|
|
|
/* {{{ proto bool array_key_exists(mixed key, array search)
|
2001-04-30 12:06:09 +08:00
|
|
|
Checks if the given key or index exists in the array */
|
2001-11-12 02:28:43 +08:00
|
|
|
PHP_FUNCTION(array_key_exists)
|
2001-04-30 12:06:09 +08:00
|
|
|
{
|
|
|
|
zval **key, /* key to check for */
|
|
|
|
**array; /* array to check in */
|
2001-11-12 02:28:43 +08:00
|
|
|
|
2001-04-30 12:06:09 +08:00
|
|
|
if (ZEND_NUM_ARGS() != 2 ||
|
|
|
|
zend_get_parameters_ex(ZEND_NUM_ARGS(), &key, &array) == FAILURE) {
|
|
|
|
WRONG_PARAM_COUNT;
|
|
|
|
}
|
2001-11-12 02:28:43 +08:00
|
|
|
|
2001-04-30 12:06:09 +08:00
|
|
|
if (Z_TYPE_PP(array) != IS_ARRAY && Z_TYPE_PP(array) != IS_OBJECT) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The second argument should be either an array or an object");
|
2001-04-30 12:06:09 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (Z_TYPE_PP(key)) {
|
|
|
|
case IS_STRING:
|
|
|
|
if (zend_hash_exists(HASH_OF(*array), Z_STRVAL_PP(key), Z_STRLEN_PP(key)+1)) {
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
RETURN_FALSE;
|
|
|
|
|
|
|
|
case IS_LONG:
|
|
|
|
if (zend_hash_index_exists(HASH_OF(*array), Z_LVAL_PP(key))) {
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
RETURN_FALSE;
|
2002-12-07 01:35:52 +08:00
|
|
|
case IS_NULL:
|
|
|
|
if (zend_hash_exists(HASH_OF(*array), "", 1)) {
|
|
|
|
RETURN_TRUE;
|
|
|
|
}
|
|
|
|
RETURN_FALSE;
|
2001-04-30 12:06:09 +08:00
|
|
|
|
|
|
|
default:
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The first argument should be either a string or an integer");
|
2001-04-30 12:06:09 +08:00
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
2001-11-12 02:28:43 +08:00
|
|
|
|
2001-04-30 12:06:09 +08:00
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2001-10-27 13:11:15 +08:00
|
|
|
|
|
|
|
/* {{{ proto array array_chunk(array input, int size [, bool preserve_keys])
|
|
|
|
Split array into chunks */
|
|
|
|
PHP_FUNCTION(array_chunk)
|
|
|
|
{
|
|
|
|
int argc = ZEND_NUM_ARGS(), key_type;
|
|
|
|
long size, current = 0;
|
|
|
|
char *str_key;
|
|
|
|
uint str_key_len;
|
|
|
|
ulong num_key;
|
|
|
|
zend_bool preserve_keys = 0;
|
|
|
|
zval *input = NULL;
|
|
|
|
zval *chunk = NULL;
|
|
|
|
zval **entry;
|
|
|
|
HashPosition pos;
|
|
|
|
|
|
|
|
if (zend_parse_parameters(argc TSRMLS_CC, "al|b", &input, &size,
|
|
|
|
&preserve_keys) == FAILURE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Do bounds checking for size parameter. */
|
|
|
|
if (size < 1) {
|
2002-08-24 09:19:28 +08:00
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Size parameter expected to be greater than 0");
|
2001-10-27 13:11:15 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(input), &pos);
|
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_P(input), (void**)&entry, &pos) == SUCCESS) {
|
|
|
|
/* If new chunk, create and initialize it. */
|
|
|
|
if (!chunk) {
|
|
|
|
MAKE_STD_ZVAL(chunk);
|
|
|
|
array_init(chunk);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add entry to the chunk, preserving keys if necessary. */
|
|
|
|
zval_add_ref(entry);
|
|
|
|
|
|
|
|
if (preserve_keys) {
|
|
|
|
key_type = zend_hash_get_current_key_ex(Z_ARRVAL_P(input), &str_key,
|
|
|
|
&str_key_len, &num_key, 0, &pos);
|
|
|
|
if (key_type == HASH_KEY_IS_STRING) {
|
|
|
|
add_assoc_zval_ex(chunk, str_key, str_key_len, *entry);
|
|
|
|
} else {
|
|
|
|
add_index_zval(chunk, num_key, *entry);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
add_next_index_zval(chunk, *entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If reached the chunk size, add it to the result array, and reset the
|
|
|
|
* pointer.
|
|
|
|
*/
|
|
|
|
if (!(++current % size)) {
|
|
|
|
add_next_index_zval(return_value, chunk);
|
|
|
|
chunk = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_P(input), &pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add the final chunk if there is one. */
|
|
|
|
if (chunk) {
|
|
|
|
add_next_index_zval(return_value, chunk);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2003-01-14 02:12:23 +08:00
|
|
|
/* {{{ proto array array_combine(array keys, array values)
|
|
|
|
Creates an array by using the elements of the first parameter as keys and the elements of the second as correspoding keys */
|
|
|
|
PHP_FUNCTION(array_combine)
|
|
|
|
{
|
|
|
|
zval *values, *keys;
|
|
|
|
HashPosition pos_values, pos_keys;
|
|
|
|
zval **entry_keys, **entry_values;
|
|
|
|
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "aa", &keys, &values) == FAILURE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (zend_hash_num_elements(Z_ARRVAL_P(keys)) == 0 || zend_hash_num_elements(Z_ARRVAL_P(values)) == 0) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Both parameters should have number of elements at least 0");
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (zend_hash_num_elements(Z_ARRVAL_P(keys)) != zend_hash_num_elements(Z_ARRVAL_P(values))) {
|
|
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Both parameters should have equal number of elements");
|
|
|
|
RETURN_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_init(return_value);
|
|
|
|
|
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(keys), &pos_keys);
|
|
|
|
zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(values), &pos_values);
|
|
|
|
while (zend_hash_get_current_data_ex(Z_ARRVAL_P(keys), (void **)&entry_keys, &pos_keys) == SUCCESS &&
|
|
|
|
zend_hash_get_current_data_ex(Z_ARRVAL_P(values), (void **)&entry_values, &pos_values) == SUCCESS) {
|
|
|
|
if (Z_TYPE_PP(entry_keys) == IS_STRING) {
|
|
|
|
zval_add_ref(entry_values);
|
|
|
|
add_assoc_zval(return_value, Z_STRVAL_PP(entry_keys), *entry_values);
|
|
|
|
} else if (Z_TYPE_PP(entry_keys) == IS_LONG) {
|
|
|
|
zval_add_ref(entry_values);
|
|
|
|
add_index_zval(return_value, Z_LVAL_PP(entry_keys), *entry_values);
|
|
|
|
}
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(entry_keys), &pos_keys);
|
|
|
|
zend_hash_move_forward_ex(Z_ARRVAL_PP(entry_values), &pos_values);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
2000-02-26 05:27:03 +08:00
|
|
|
/*
|
|
|
|
* Local variables:
|
|
|
|
* tab-width: 4
|
|
|
|
* c-basic-offset: 4
|
|
|
|
* End:
|
2001-09-17 04:49:57 +08:00
|
|
|
* vim600: noet sw=4 ts=4 fdm=marker
|
|
|
|
* vim<600: noet sw=4 ts=4
|
2000-02-26 05:27:03 +08:00
|
|
|
*/
|