Blender V5.0
sort.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2013 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
8
9#include <cstdlib>
10
11#include "BLI_sort.h"
12
13#ifdef min /* For MSVC. */
14# undef min
15#endif
16
17/* Intentional null pointer subtraction warns with: CLANG 19.1.7. */
18#ifdef __clang__
19# pragma clang diagnostic push
20# pragma clang diagnostic ignored "-Wnull-pointer-subtraction"
21#endif
22
23/* Maintained by FreeBSD. */
24/* clang-format off */
25
33inline char *med3(char *a, char *b, char *c, BLI_sort_cmp_t cmp, void *thunk);
34inline void swapfunc(char *a, char *b, int n, int swaptype);
35
36#define min(a, b) (a) < (b) ? (a) : (b)
37#define swapcode(TYPE, parmi, parmj, n) \
38{ \
39 long i = (n) / sizeof(TYPE); \
40 TYPE *pi = (TYPE *) (parmi); \
41 TYPE *pj = (TYPE *) (parmj); \
42 do { \
43 TYPE t = *pi; \
44 *pi++ = *pj; \
45 *pj++ = t; \
46 } while (--i > 0); \
47}
48#define SWAPINIT(a, es) swaptype = ((char *)a - (char *)0) % sizeof(long) || \
49 es % sizeof(long) ? 2 : es == sizeof(long)? 0 : 1;
50
51inline void swapfunc(char *a, char *b, int n, int swaptype)
52{
53 if (swaptype <= 1)
54 swapcode(long, a, b, n)
55 else
56 swapcode(char, a, b, n)
57}
58
59#define swap(a, b) \
60 if (swaptype == 0) { \
61 long t = *(long *)(a); \
62 *(long *)(a) = *(long *)(b);\
63 *(long *)(b) = t; \
64 } else \
65 swapfunc(a, b, es, swaptype)
66
67#define vecswap(a, b, n) if ((n) > 0) swapfunc(a, b, n, swaptype)
68#define CMP(t, x, y) (cmp((x), (y), (t)))
69
70inline char *med3(char *a, char *b, char *c, BLI_sort_cmp_t cmp, void *thunk)
71{
72 return CMP(thunk, a, b) < 0 ?
73 (CMP(thunk, b, c) < 0 ? b : (CMP(thunk, a, c) < 0 ? c : a )) :
74 (CMP(thunk, b, c) > 0 ? b : (CMP(thunk, a, c) < 0 ? a : c ));
75}
76
77void BLI_qsort_r(void *a, size_t n, size_t es, BLI_sort_cmp_t cmp, void *thunk)
78{
79 char *pa, *pb, *pc, *pd, *pl, *pm, *pn;
80 int d, r, swaptype, swap_cnt;
81
82loop:
83 SWAPINIT(a, es);
84 swap_cnt = 0;
85 if (n < 7) {
86 for (pm = (char *)a + es; pm < (char *)a + n * es; pm += es) {
87 for (pl = pm;
88 pl > (char *)a && CMP(thunk, pl - es, pl) > 0;
89 pl -= es)
90 {
91 swap(pl, pl - es);
92 }
93 }
94 return;
95 }
96 pm = (char *)a + (n / 2) * es;
97 if (n > 7) {
98 pl = (char *)a;
99 pn = (char *)a + (n - 1) * es;
100 if (n > 40) {
101 d = (n / 8) * es;
102 pl = med3(pl, pl + d, pl + 2 * d, cmp, thunk);
103 pm = med3(pm - d, pm, pm + d, cmp, thunk);
104 pn = med3(pn - 2 * d, pn - d, pn, cmp, thunk);
105 }
106 pm = med3(pl, pm, pn, cmp, thunk);
107 }
108 swap((char *)a, pm);
109 pa = pb = (char *)a + es;
110
111 pc = pd = (char *)a + (n - 1) * es;
112 for (;;) {
113 while (pb <= pc && (r = CMP(thunk, pb, a)) <= 0) {
114 if (r == 0) {
115 swap_cnt = 1;
116 swap(pa, pb);
117 pa += es;
118 }
119 pb += es;
120 }
121 while (pb <= pc && (r = CMP(thunk, pc, a)) >= 0) {
122 if (r == 0) {
123 swap_cnt = 1;
124 swap(pc, pd);
125 pd -= es;
126 }
127 pc -= es;
128 }
129 if (pb > pc) {
130 break;
131 }
132 swap(pb, pc);
133 swap_cnt = 1;
134 pb += es;
135 pc -= es;
136 }
137 if (swap_cnt == 0) { /* Switch to insertion sort */
138 for (pm = (char *)a + es; pm < (char *)a + n * es; pm += es) {
139 for (pl = pm;
140 pl > (char *)a && CMP(thunk, pl - es, pl) > 0;
141 pl -= es)
142 {
143 swap(pl, pl - es);
144 }
145 }
146 return;
147 }
148
149 pn = (char *)a + n * es;
150 r = min(pa - (char *)a, pb - pa);
151 vecswap((char *)a, pb - r, r);
152 r = min(pd - pc, pn - pd - es);
153 vecswap(pb, pn - r, r);
154 if ((r = pb - pa) > es) {
155 BLI_qsort_r(a, r / es, es, cmp, thunk);
156 }
157 if ((r = pd - pc) > es) {
158 /* Iterate rather than recurse to save stack space */
159 a = pn - r;
160 n = r / es;
161 goto loop;
162 }
163}
164
165/* clang-format on */
166
167#ifdef __clang__
168# pragma clang diagnostic pop
169#endif
int(* BLI_sort_cmp_t)(const void *a, const void *b, void *ctx)
Definition BLI_sort.h:14
char * med3(char *a, char *b, char *c, BLI_sort_cmp_t cmp, void *thunk)
Definition sort.cc:70
void swapfunc(char *a, char *b, int n, int swaptype)
Definition sort.cc:51
#define swap(a, b)
Definition sort.cc:59
#define vecswap(a, b, n)
Definition sort.cc:67
#define CMP(t, x, y)
Definition sort.cc:68
#define swapcode(TYPE, parmi, parmj, n)
Definition sort.cc:37
#define min(a, b)
Definition sort.cc:36
#define SWAPINIT(a, es)
Definition sort.cc:48
void BLI_qsort_r(void *a, size_t n, size_t es, BLI_sort_cmp_t cmp, void *thunk)
Definition sort.cc:77