Sort an array by values using a user-defined comparison function */
PHP_FUNCTION(usort)
{
- zval **array;
+ zval **array_ptr, *array;
int refcount;
HashTable *target_hash;
+ zval *func_name;
PHP_ARRAY_CMP_FUNC_VARS;
PHP_ARRAY_CMP_FUNC_BACKUP();
- if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array, &BG(user_compare_func_name)) == FAILURE) {
+ if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array_ptr, &BG(user_compare_func_name)) == FAILURE) {
PHP_ARRAY_CMP_FUNC_RESTORE();
WRONG_PARAM_COUNT;
}
+ array = *array_ptr;
- target_hash = HASH_OF(*array);
+ target_hash = HASH_OF(array);
if (!target_hash) {
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
PHP_ARRAY_CMP_FUNC_RESTORE();
}
PHP_ARRAY_CMP_FUNC_CHECK(BG(user_compare_func_name))
+ func_name = *BG(user_compare_func_name);
+ BG(user_compare_func_name) = &func_name;
BG(user_compare_fci_cache).initialized = 0;
/* Clear the is_ref flag, so the attemts to modify the array in user
* comparison. The result of sorting in such case is undefined and the
* function returns FALSE.
*/
- (*array)->is_ref = 0;
- refcount = (*array)->refcount;
+ array->is_ref = 0;
+ refcount = array->refcount;
if (zend_hash_sort(target_hash, zend_qsort, array_user_compare, 1 TSRMLS_CC) == FAILURE) {
RETVAL_FALSE;
} else {
- if (refcount > (*array)->refcount) {
+ if (refcount > array->refcount) {
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Array was modified by the user comparison function");
RETVAL_FALSE;
} else {
}
}
- if ((*array)->refcount > 1) {
- (*array)->is_ref = 1;
+ if (array->refcount > 1) {
+ array->is_ref = 1;
}
PHP_ARRAY_CMP_FUNC_RESTORE();
Sort an array with a user-defined comparison function and maintain index association */
PHP_FUNCTION(uasort)
{
- zval **array;
+ zval **array_ptr, *array;
int refcount;
HashTable *target_hash;
+ zval *func_name;
PHP_ARRAY_CMP_FUNC_VARS;
PHP_ARRAY_CMP_FUNC_BACKUP();
- if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array, &BG(user_compare_func_name)) == FAILURE) {
+ if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array_ptr, &BG(user_compare_func_name)) == FAILURE) {
PHP_ARRAY_CMP_FUNC_RESTORE();
WRONG_PARAM_COUNT;
}
- target_hash = HASH_OF(*array);
+ array = *array_ptr;
+ target_hash = HASH_OF(array);
if (!target_hash) {
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
PHP_ARRAY_CMP_FUNC_RESTORE();
}
PHP_ARRAY_CMP_FUNC_CHECK(BG(user_compare_func_name))
+ func_name = *BG(user_compare_func_name);
+ BG(user_compare_func_name) = &func_name;
BG(user_compare_fci_cache).initialized = 0;
/* Clear the is_ref flag, so the attemts to modify the array in user
* comparison. The result of sorting in such case is undefined and the
* function returns FALSE.
*/
- (*array)->is_ref = 0;
- refcount = (*array)->refcount;
+ array->is_ref = 0;
+ refcount = array->refcount;
if (zend_hash_sort(target_hash, zend_qsort, array_user_compare, 0 TSRMLS_CC) == FAILURE) {
RETVAL_FALSE;
} else {
- if (refcount > (*array)->refcount) {
+ if (refcount > array->refcount) {
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Array was modified by the user comparison function");
RETVAL_FALSE;
} else {
}
}
- if ((*array)->refcount > 1) {
- (*array)->is_ref = 1;
+ if (array->refcount > 1) {
+ array->is_ref = 1;
}
PHP_ARRAY_CMP_FUNC_RESTORE();
Sort an array by keys using a user-defined comparison function */
PHP_FUNCTION(uksort)
{
- zval **array;
+ zval **array_ptr, *array;
+ int refcount;
HashTable *target_hash;
+ zval *func_name;
PHP_ARRAY_CMP_FUNC_VARS;
PHP_ARRAY_CMP_FUNC_BACKUP();
- if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array, &BG(user_compare_func_name)) == FAILURE) {
+ if (ZEND_NUM_ARGS() != 2 || zend_get_parameters_ex(2, &array_ptr, &BG(user_compare_func_name)) == FAILURE) {
PHP_ARRAY_CMP_FUNC_RESTORE();
WRONG_PARAM_COUNT;
}
- target_hash = HASH_OF(*array);
+ array = *array_ptr;
+ target_hash = HASH_OF(array);
if (!target_hash) {
php_error_docref(NULL TSRMLS_CC, E_WARNING, "The argument should be an array");
PHP_ARRAY_CMP_FUNC_RESTORE();
}
PHP_ARRAY_CMP_FUNC_CHECK(BG(user_compare_func_name))
+ func_name = *BG(user_compare_func_name);
+ BG(user_compare_func_name) = &func_name;
BG(user_compare_fci_cache).initialized = 0;
+ /* Clear the is_ref flag, so the attemts to modify the array in user
+ * comaprison function will create a copy of array and won't affect the
+ * original array. The fact of modification is detected using refcount
+ * comparison. The result of sorting in such case is undefined and the
+ * function returns FALSE.
+ */
+ array->is_ref = 0;
+ refcount = array->refcount;
+
if (zend_hash_sort(target_hash, zend_qsort, array_user_key_compare, 0 TSRMLS_CC) == FAILURE) {
- PHP_ARRAY_CMP_FUNC_RESTORE();
+ RETVAL_FALSE;
+ } else {
+ if (refcount > array->refcount) {
+ php_error_docref(NULL TSRMLS_CC, E_WARNING, "Array was modified by the user comparison function");
+ RETVAL_FALSE;
+ } else {
+ RETVAL_TRUE;
+ }
+ }
- RETURN_FALSE;
+ if (array->refcount > 1) {
+ array->is_ref = 1;
}
PHP_ARRAY_CMP_FUNC_RESTORE();
- RETURN_TRUE;
}
/* }}} */