Coverage Report

Created: 2026-01-18 06:49

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/php-src/Zend/zend_sort.c
Line
Count
Source
1
/*
2
   +----------------------------------------------------------------------+
3
   | Zend Engine                                                          |
4
   +----------------------------------------------------------------------+
5
   | Copyright (c) Zend Technologies Ltd. (http://www.zend.com)           |
6
   +----------------------------------------------------------------------+
7
   | This source file is subject to version 2.00 of the Zend license,     |
8
   | that is bundled with this package in the file LICENSE, and is        |
9
   | available through the world-wide-web at the following url:           |
10
   | http://www.zend.com/license/2_00.txt.                                |
11
   | If you did not receive a copy of the Zend license and are unable to  |
12
   | obtain it through the world-wide-web, please send a note to          |
13
   | license@zend.com so we can mail you a copy immediately.              |
14
   +----------------------------------------------------------------------+
15
   | Authors: Xinchen Hui <laruence@php.net>                              |
16
   |          Sterling Hughes <sterling@php.net>                          |
17
   +----------------------------------------------------------------------+
18
*/
19
20
#include "zend.h"
21
#include "zend_sort.h"
22
#include <limits.h>
23
24
4.59k
static inline void zend_sort_2(void *a, void *b, compare_func_t cmp, swap_func_t swp) /* {{{ */ {
25
4.59k
  if (cmp(a, b) > 0) {
26
2.96k
    swp(a, b);
27
2.96k
  }
28
4.59k
}
29
/* }}} */
30
31
154k
static inline void zend_sort_3(void *a, void *b, void *c, compare_func_t cmp, swap_func_t swp) /* {{{ */ {
32
154k
  if (!(cmp(a, b) > 0)) {
33
51.2k
    if (!(cmp(b, c) > 0)) {
34
28.7k
      return;
35
28.7k
    }
36
22.5k
    swp(b, c);
37
22.5k
    if (cmp(a, b) > 0) {
38
8.16k
      swp(a, b);
39
8.16k
    }
40
22.5k
    return;
41
51.2k
  }
42
103k
  if (!(cmp(c, b) > 0)) {
43
11.5k
    swp(a, c);
44
11.5k
    return;
45
11.5k
  }
46
91.7k
  swp(a, b);
47
91.7k
  if (cmp(b, c) > 0) {
48
45.7k
    swp(b, c);
49
45.7k
  }
50
91.7k
}
51
/* }}} */
52
53
17.1k
static void zend_sort_4(void *a, void *b, void *c, void *d, compare_func_t cmp, swap_func_t swp) /* {{{ */ {
54
17.1k
  zend_sort_3(a, b, c, cmp, swp);
55
17.1k
  if (cmp(c, d) > 0) {
56
13.5k
    swp(c, d);
57
13.5k
    if (cmp(b, c) > 0) {
58
6.88k
      swp(b, c);
59
6.88k
      if (cmp(a, b) > 0) {
60
3.69k
        swp(a, b);
61
3.69k
      }
62
6.88k
    }
63
13.5k
  }
64
17.1k
}
65
/* }}} */
66
67
6.87k
static void zend_sort_5(void *a, void *b, void *c, void *d, void *e, compare_func_t cmp, swap_func_t swp) /* {{{ */ {
68
6.87k
  zend_sort_4(a, b, c, d, cmp, swp);
69
6.87k
  if (cmp(d, e) > 0) {
70
5.25k
    swp(d, e);
71
5.25k
    if (cmp(c, d) > 0) {
72
4.21k
      swp(c, d);
73
4.21k
      if (cmp(b, c) > 0) {
74
1.36k
        swp(b, c);
75
1.36k
        if (cmp(a, b) > 0) {
76
689
          swp(a, b);
77
689
        }
78
1.36k
      }
79
4.21k
    }
80
5.25k
  }
81
6.87k
}
82
/* }}} */
83
84
126k
ZEND_API void zend_insert_sort(void *base, size_t nmemb, size_t siz, compare_func_t cmp, swap_func_t swp) /* {{{ */{
85
126k
  switch (nmemb) {
86
0
    case 0:
87
3.82k
    case 1:
88
3.82k
      break;
89
4.59k
    case 2:
90
4.59k
      zend_sort_2(base, (char *)base + siz, cmp, swp);
91
4.59k
      break;
92
14.8k
    case 3:
93
14.8k
      zend_sort_3(base, (char *)base + siz, (char *)base + siz + siz, cmp, swp);
94
14.8k
      break;
95
10.2k
    case 4:
96
10.2k
      {
97
10.2k
        size_t siz2 = siz + siz;
98
10.2k
        zend_sort_4(base, (char *)base + siz, (char *)base + siz2, (char *)base + siz + siz2, cmp, swp);
99
10.2k
      }
100
10.2k
      break;
101
5.88k
    case 5:
102
5.88k
      {
103
5.88k
        size_t siz2 = siz + siz;
104
5.88k
        zend_sort_5(base, (char *)base + siz, (char *)base + siz2, (char *)base + siz + siz2, (char *)base + siz2 + siz2, cmp, swp);
105
5.88k
      }
106
5.88k
      break;
107
87.5k
    default:
108
87.5k
      {
109
87.5k
        char *i, *j, *k;
110
87.5k
        char *start = (char *)base;
111
87.5k
        char *end = start + (nmemb * siz);
112
87.5k
        size_t siz2= siz + siz;
113
87.5k
        char *sentry = start + (6 * siz);
114
525k
        for (i = start + siz; i < sentry; i += siz) {
115
437k
          j = i - siz;
116
437k
          if (!(cmp(j, i) > 0)) {
117
168k
            continue;
118
168k
          }
119
580k
          while (j != start) {
120
483k
            j -= siz;
121
483k
            if (!(cmp(j, i) > 0)) {
122
172k
              j += siz;
123
172k
              break;
124
172k
            }
125
483k
          }
126
848k
          for (k = i; k > j; k -= siz) {
127
580k
            swp(k, k - siz);
128
580k
          }
129
268k
        }
130
548k
        for (i = sentry; i < end; i += siz) {
131
461k
          j = i - siz;
132
461k
          if (!(cmp(j, i) > 0)) {
133
83.7k
            continue;
134
83.7k
          }
135
819k
          do {
136
819k
            j -= siz2;
137
819k
            if (!(cmp(j, i) > 0)) {
138
332k
              j += siz;
139
332k
              if (!(cmp(j, i) > 0)) {
140
188k
                j += siz;
141
188k
              }
142
332k
              break;
143
332k
            }
144
487k
            if (j == start) {
145
12.6k
              break;
146
12.6k
            }
147
474k
            if (j == start + siz) {
148
32.8k
              j -= siz;
149
32.8k
              if (cmp(i, j) > 0) {
150
14.9k
                j += siz;
151
14.9k
              }
152
32.8k
              break;
153
32.8k
            }
154
474k
          } while (1);
155
1.89M
          for (k = i; k > j; k -= siz) {
156
1.51M
            swp(k, k - siz);
157
1.51M
          }
158
377k
        }
159
87.5k
      }
160
87.5k
      break;
161
126k
  }
162
126k
}
163
/* }}} */
164
165
/* {{{ ZEND_API void zend_sort(void *base, size_t nmemb, size_t siz, compare_func_t cmp, swap_func_t swp)
166
 *
167
 * Derived from LLVM's libc++ implementation of std::sort.
168
 *
169
 * ===========================================================================
170
 * libc++ License
171
 * ===========================================================================
172
 *
173
 * The libc++ library is dual licensed under both the University of Illinois
174
 * "BSD-Like" license and the MIT license. As a user of this code you may
175
 * choose to use it under either license. As a contributor, you agree to allow
176
 * your code to be used under both.
177
 *
178
 * Full text of the relevant licenses is included below.
179
 *
180
 * ===========================================================================
181
 *
182
 * University of Illinois/NCSA
183
 * Open Source License
184
 *
185
 * Copyright (c) 2009-2012 by the contributors listed at
186
 * http://llvm.org/svn/llvm-project/libcxx/trunk/CREDITS.TXT
187
 *
188
 * All rights reserved.
189
 *
190
 * Developed by:
191
 *
192
 *     LLVM Team
193
 *
194
 *     University of Illinois at Urbana-Champaign
195
 *
196
 *     http://llvm.org
197
 *
198
 * Permission is hereby granted, free of charge, to any person obtaining a copy
199
 * of this software and associated documentation files (the "Software"), to
200
 * deal with the Software without restriction, including without limitation the
201
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
202
 * sell copies of the Software, and to permit persons to whom the Software is
203
 * furnished to do so, subject to the following conditions:
204
 *
205
 *     * Redistributions of source code must retain the above copyright notice,
206
 *       this list of conditions and the following disclaimers.
207
 *
208
 *     * Redistributions in binary form must reproduce the above copyright
209
 *       notice, this list of conditions and the following disclaimers in the
210
 *       documentation and/or other materials provided with the distribution.
211
 *
212
 *     * Neither the names of the LLVM Team, University of Illinois at
213
 *       Urbana-Champaign, nor the names of its contributors may be used to
214
 *       endorse or promote products derived from this Software without
215
 *       specific prior written permission.
216
 *
217
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
218
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
219
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
220
 * CONTRIBUTORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
221
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
222
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
223
 * WITH THE SOFTWARE.
224
 *
225
 * ===========================================================================
226
 *
227
 * Copyright (c) 2009-2012 by the contributors listed at
228
 * http://llvm.org/svn/llvm-project/libcxx/trunk/CREDITS.TXT
229
 *
230
 * Permission is hereby granted, free of charge, to any person obtaining a copy
231
 * of this software and associated documentation files (the "Software"), to
232
 * deal in the Software without restriction, including without limitation the
233
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
234
 * sell copies of the Software, and to permit persons to whom the Software is
235
 * furnished to do so, subject to the following conditions:
236
 *
237
 * The above copyright notice and this permission notice shall be included in
238
 * all copies or substantial portions of the Software.
239
 *
240
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
241
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
242
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
243
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
244
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
245
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
246
 * IN THE SOFTWARE.
247
 */
248
ZEND_API void zend_sort(void *base, size_t nmemb, size_t siz, compare_func_t cmp, swap_func_t swp)
249
126k
{
250
250k
  while (1) {
251
250k
    if (nmemb <= 16) {
252
126k
      zend_insert_sort(base, nmemb, siz, cmp, swp);
253
126k
      return;
254
126k
    } else {
255
123k
      char *i, *j;
256
123k
      char *start = (char *)base;
257
123k
      char *end = start + (nmemb * siz);
258
123k
      size_t offset = (nmemb >> Z_L(1));
259
123k
      char *pivot = start + (offset * siz);
260
261
123k
      if ((nmemb >> Z_L(10))) {
262
994
        size_t delta = (offset >> Z_L(1)) * siz;
263
994
        zend_sort_5(start, start + delta, pivot, pivot + delta, end - siz, cmp, swp);
264
122k
      } else {
265
122k
        zend_sort_3(start, pivot, end - siz, cmp, swp);
266
122k
      }
267
123k
      swp(start + siz, pivot);
268
123k
      pivot = start + siz;
269
123k
      i = pivot + siz;
270
123k
      j = end - siz;
271
862k
      while (1) {
272
6.49M
        while (cmp(pivot, i) > 0) {
273
5.69M
          i += siz;
274
5.69M
          if (UNEXPECTED(i == j)) {
275
68.7k
            goto done;
276
68.7k
          }
277
5.69M
        }
278
793k
        j -= siz;
279
793k
        if (UNEXPECTED(j == i)) {
280
20.7k
          goto done;
281
20.7k
        }
282
3.89M
        while (cmp(j, pivot) > 0) {
283
3.14M
          j -= siz;
284
3.14M
          if (UNEXPECTED(j == i)) {
285
20.9k
            goto done;
286
20.9k
          }
287
3.14M
        }
288
751k
        swp(i, j);
289
751k
        i += siz;
290
751k
        if (UNEXPECTED(i == j)) {
291
13.1k
          goto done;
292
13.1k
        }
293
751k
      }
294
123k
done:
295
123k
      swp(pivot, i - siz);
296
123k
      if ((i - siz) - start < end - i) {
297
51.1k
        zend_sort(start, (i - start)/siz - 1, siz, cmp, swp);
298
51.1k
        base = i;
299
51.1k
        nmemb = (end - i)/siz;
300
72.4k
      } else {
301
72.4k
        zend_sort(i, (end - i)/siz, siz, cmp, swp);
302
72.4k
        nmemb = (i - start)/siz - 1;
303
72.4k
      }
304
123k
    }
305
250k
  }
306
126k
}
307
/* }}} */