diff --git a/NEWS b/NEWS index 21d5d334a4e4c..5e0f75578366e 100644 --- a/NEWS +++ b/NEWS @@ -46,8 +46,6 @@ PHP NEWS - Standard: . Fixed bug GH-9017 (php_stream_sock_open_from_socket could return NULL). (Heiko Weber) - . Fixed bug GH-9296 (`ksort` behaves incorrectly on arrays with mixed keys). - (Denis Vaksman) - Streams: . Fixed bug GH-8472 (The resource returned by stream_socket_accept may have diff --git a/ext/standard/array.c b/ext/standard/array.c index d145661e3782e..32ad1b0eaa6ed 100644 --- a/ext/standard/array.c +++ b/ext/standard/array.c @@ -166,25 +166,40 @@ static zend_never_inline ZEND_COLD int stable_sort_fallback(Bucket *a, Bucket *b static zend_always_inline int php_array_key_compare_unstable_i(Bucket *f, Bucket *s) /* {{{ */ { - zval first; - zval second; - - if (f->key == NULL && s->key == NULL) { - return (zend_long)f->h > (zend_long)s->h ? 1 : -1; - } else if (f->key && s->key) { - return zendi_smart_strcmp(f->key, s->key); - } - if (f->key) { - ZVAL_STR(&first, f->key); - } else { - ZVAL_LONG(&first, f->h); - } - if (s->key) { - ZVAL_STR(&second, s->key); - } else { - ZVAL_LONG(&second, s->h); - } - return zend_compare(&first, &second); + zend_uchar t; + zend_long l1, l2; + double d; + + if (f->key == NULL) { + if (s->key == NULL) { + return (zend_long)f->h > (zend_long)s->h ? 1 : -1; + } else { + l1 = (zend_long)f->h; + t = is_numeric_string(s->key->val, s->key->len, &l2, &d, 1); + if (t == IS_LONG) { + /* pass */ + } else if (t == IS_DOUBLE) { + return ZEND_NORMALIZE_BOOL((double)l1 - d); + } else { + l2 = 0; + } + } + } else { + if (s->key) { + return zendi_smart_strcmp(f->key, s->key); + } else { + l2 = (zend_long)s->h; + t = is_numeric_string(f->key->val, f->key->len, &l1, &d, 1); + if (t == IS_LONG) { + /* pass */ + } else if (t == IS_DOUBLE) { + return ZEND_NORMALIZE_BOOL(d - (double)l2); + } else { + l1 = 0; + } + } + } + return ZEND_NORMALIZE_BOOL(l1 - l2); } /* }}} */ diff --git a/ext/standard/tests/array/002.phpt b/ext/standard/tests/array/002.phpt index 5cf377e2a3538..fdb838deb19dd 100644 --- a/ext/standard/tests/array/002.phpt +++ b/ext/standard/tests/array/002.phpt @@ -276,8 +276,6 @@ array(8) { -- Testing krsort() -- No second argument: array(8) { - ["test"]=> - int(27) [16777216]=> float(-0.3333333333333333) [1001]=> @@ -290,6 +288,8 @@ array(8) { string(4) "Test" [0]=> string(3) "PHP" + ["test"]=> + int(27) [-1000]=> array(2) { [0]=> @@ -300,8 +300,6 @@ array(8) { } Using SORT_REGULAR: array(8) { - ["test"]=> - int(27) [16777216]=> float(-0.3333333333333333) [1001]=> @@ -314,6 +312,8 @@ array(8) { string(4) "Test" [0]=> string(3) "PHP" + ["test"]=> + int(27) [-1000]=> array(2) { [0]=> @@ -334,10 +334,10 @@ array(8) { string(27) "PHP: Hypertext Preprocessor" [5]=> string(4) "Test" - ["test"]=> - int(27) [0]=> string(3) "PHP" + ["test"]=> + int(27) [-1000]=> array(2) { [0]=> @@ -383,6 +383,8 @@ array(8) { } [0]=> string(3) "PHP" + ["test"]=> + int(27) [5]=> string(4) "Test" [17]=> @@ -393,8 +395,6 @@ array(8) { string(6) "monkey" [16777216]=> float(-0.3333333333333333) - ["test"]=> - int(27) } Using SORT_REGULAR: array(8) { @@ -407,6 +407,8 @@ array(8) { } [0]=> string(3) "PHP" + ["test"]=> + int(27) [5]=> string(4) "Test" [17]=> @@ -417,8 +419,6 @@ array(8) { string(6) "monkey" [16777216]=> float(-0.3333333333333333) - ["test"]=> - int(27) } Using SORT_NUMERIC: array(8) { diff --git a/ext/standard/tests/array/gh9296.phpt b/ext/standard/tests/array/gh9296.phpt deleted file mode 100644 index cfbc5cbbb09ec..0000000000000 --- a/ext/standard/tests/array/gh9296.phpt +++ /dev/null @@ -1,17 +0,0 @@ ---TEST-- -GH-9296: incorrect ksort(..., SORT_REGULAR) behaviour on arrays with numeric and string keys ---FILE-- - 0, 600 => 1]; -ksort($array, SORT_REGULAR); -var_dump($array); -var_dump(array_key_first($array) <=> array_key_last($array)); -?> ---EXPECT-- -array(2) { - [600]=> - int(1) - ["aaa"]=> - int(0) -} -int(-1) diff --git a/ext/standard/tests/array/krsort_variation8.phpt b/ext/standard/tests/array/krsort_variation8.phpt index 39e72e2eb2221..1483e6fd1bbc1 100644 Binary files a/ext/standard/tests/array/krsort_variation8.phpt and b/ext/standard/tests/array/krsort_variation8.phpt differ diff --git a/ext/standard/tests/array/krsort_variation9.phpt b/ext/standard/tests/array/krsort_variation9.phpt index b0309f373d362..4364911b62127 100644 --- a/ext/standard/tests/array/krsort_variation9.phpt +++ b/ext/standard/tests/array/krsort_variation9.phpt @@ -82,22 +82,22 @@ array(5) { - With default sort flag - bool(true) array(3) { - ["c"]=> - string(5) "apple" ["a"]=> string(6) "orange" [0]=> string(6) "banana" + ["c"]=> + string(5) "apple" } - Sort flag = SORT_REGULAR - bool(true) array(3) { - ["c"]=> - string(5) "apple" ["a"]=> string(6) "orange" [0]=> string(6) "banana" + ["c"]=> + string(5) "apple" } -- Iteration 3 -- diff --git a/ext/standard/tests/array/ksort_variation8.phpt b/ext/standard/tests/array/ksort_variation8.phpt index 9a15c4dc3e75d..7b7e1c003e587 100644 Binary files a/ext/standard/tests/array/ksort_variation8.phpt and b/ext/standard/tests/array/ksort_variation8.phpt differ diff --git a/ext/standard/tests/array/ksort_variation9.phpt b/ext/standard/tests/array/ksort_variation9.phpt index 0023a9e0602ae..9322af70f57fa 100644 --- a/ext/standard/tests/array/ksort_variation9.phpt +++ b/ext/standard/tests/array/ksort_variation9.phpt @@ -81,20 +81,20 @@ array(5) { - With default sort flag - bool(true) array(3) { - [0]=> - string(6) "banana" ["a"]=> string(6) "orange" + [0]=> + string(6) "banana" ["c"]=> string(5) "apple" } - Sort flag = SORT_REGULAR - bool(true) array(3) { - [0]=> - string(6) "banana" ["a"]=> string(6) "orange" + [0]=> + string(6) "banana" ["c"]=> string(5) "apple" }