]> granicus.if.org Git - postgresql/blob - src/port/qsort_arg.c
Use Min() instead of min() in qsort, for consistency and to avoid
[postgresql] / src / port / qsort_arg.c
1 /*
2  *      qsort_arg.c: qsort with a passthrough "void *" argument
3  *
4  *      Modifications from vanilla NetBSD source:
5  *        Add do ... while() macro fix
6  *        Remove __inline, _DIAGASSERTs, __P
7  *        Remove ill-considered "swap_cnt" switch to insertion sort,
8  *        in favor of a simple check for presorted input.
9  *
10  *      CAUTION: if you change this file, see also qsort.c
11  *
12  *      $PostgreSQL: pgsql/src/port/qsort_arg.c,v 1.3 2006/10/12 15:04:55 tgl Exp $
13  */
14
15 /*      $NetBSD: qsort.c,v 1.13 2003/08/07 16:43:42 agc Exp $   */
16
17 /*-
18  * Copyright (c) 1992, 1993
19  *      The Regents of the University of California.  All rights reserved.
20  *
21  * Redistribution and use in source and binary forms, with or without
22  * modification, are permitted provided that the following conditions
23  * are met:
24  * 1. Redistributions of source code must retain the above copyright
25  *        notice, this list of conditions and the following disclaimer.
26  * 2. Redistributions in binary form must reproduce the above copyright
27  *        notice, this list of conditions and the following disclaimer in the
28  *        documentation and/or other materials provided with the distribution.
29  * 3. Neither the name of the University nor the names of its contributors
30  *        may be used to endorse or promote products derived from this software
31  *        without specific prior written permission.
32  *
33  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
34  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
35  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
36  * ARE DISCLAIMED.      IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
37  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
38  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
39  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
40  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
41  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
42  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
43  * SUCH DAMAGE.
44  */
45
46 #include "c.h"
47
48
49 static char *med3(char *a, char *b, char *c,
50          qsort_arg_comparator cmp, void *arg);
51 static void swapfunc(char *, char *, size_t, int);
52
53 /*
54  * Qsort routine based on J. L. Bentley and M. D. McIlroy,
55  * "Engineering a sort function",
56  * Software--Practice and Experience 23 (1993) 1249-1265.
57  * We have modified their original by adding a check for already-sorted input,
58  * which seems to be a win per discussions on pgsql-hackers around 2006-03-21.
59  */
60 #define swapcode(TYPE, parmi, parmj, n) \
61 do {            \
62         size_t i = (n) / sizeof (TYPE);                 \
63         TYPE *pi = (TYPE *)(void *)(parmi);                     \
64         TYPE *pj = (TYPE *)(void *)(parmj);                     \
65         do {                                            \
66                 TYPE    t = *pi;                        \
67                 *pi++ = *pj;                            \
68                 *pj++ = t;                              \
69                 } while (--i > 0);                              \
70 } while (0)
71
72 #define SWAPINIT(a, es) swaptype = ((char *)(a) - (char *)0) % sizeof(long) || \
73         (es) % sizeof(long) ? 2 : (es) == sizeof(long)? 0 : 1;
74
75 static void
76 swapfunc(a, b, n, swaptype)
77 char       *a,
78                    *b;
79 size_t          n;
80 int                     swaptype;
81 {
82         if (swaptype <= 1)
83                 swapcode(long, a, b, n);
84         else
85                 swapcode(char, a, b, n);
86 }
87
88 #define swap(a, b)                                              \
89         if (swaptype == 0) {                                    \
90                 long t = *(long *)(void *)(a);                  \
91                 *(long *)(void *)(a) = *(long *)(void *)(b);    \
92                 *(long *)(void *)(b) = t;                       \
93         } else                                                  \
94                 swapfunc(a, b, es, swaptype)
95
96 #define vecswap(a, b, n) if ((n) > 0) swapfunc((a), (b), (size_t)(n), swaptype)
97
98 static char *
99 med3(char *a, char *b, char *c, qsort_arg_comparator cmp, void *arg)
100 {
101         return cmp(a, b, arg) < 0 ?
102                 (cmp(b, c, arg) < 0 ? b : (cmp(a, c, arg) < 0 ? c : a))
103                 : (cmp(b, c, arg) > 0 ? b : (cmp(a, c, arg) < 0 ? a : c));
104 }
105
106 void
107 qsort_arg(void *a, size_t n, size_t es, qsort_arg_comparator cmp, void *arg)
108 {
109         char       *pa,
110                            *pb,
111                            *pc,
112                            *pd,
113                            *pl,
114                            *pm,
115                            *pn;
116         int                     d,
117                                 r,
118                                 swaptype,
119                                 presorted;
120
121 loop:SWAPINIT(a, es);
122         if (n < 7)
123         {
124                 for (pm = (char *) a + es; pm < (char *) a + n * es; pm += es)
125                         for (pl = pm; pl > (char *) a && cmp(pl - es, pl, arg) > 0;
126                                  pl -= es)
127                                 swap(pl, pl - es);
128                 return;
129         }
130         presorted = 1;
131         for (pm = (char *) a + es; pm < (char *) a + n * es; pm += es)
132         {
133                 if (cmp(pm - es, pm, arg) > 0)
134                 {
135                         presorted = 0;
136                         break;
137                 }
138         }
139         if (presorted)
140                 return;
141         pm = (char *) a + (n / 2) * es;
142         if (n > 7)
143         {
144                 pl = (char *) a;
145                 pn = (char *) a + (n - 1) * es;
146                 if (n > 40)
147                 {
148                         d = (n / 8) * es;
149                         pl = med3(pl, pl + d, pl + 2 * d, cmp, arg);
150                         pm = med3(pm - d, pm, pm + d, cmp, arg);
151                         pn = med3(pn - 2 * d, pn - d, pn, cmp, arg);
152                 }
153                 pm = med3(pl, pm, pn, cmp, arg);
154         }
155         swap(a, pm);
156         pa = pb = (char *) a + es;
157         pc = pd = (char *) a + (n - 1) * es;
158         for (;;)
159         {
160                 while (pb <= pc && (r = cmp(pb, a, arg)) <= 0)
161                 {
162                         if (r == 0)
163                         {
164                                 swap(pa, pb);
165                                 pa += es;
166                         }
167                         pb += es;
168                 }
169                 while (pb <= pc && (r = cmp(pc, a, arg)) >= 0)
170                 {
171                         if (r == 0)
172                         {
173                                 swap(pc, pd);
174                                 pd -= es;
175                         }
176                         pc -= es;
177                 }
178                 if (pb > pc)
179                         break;
180                 swap(pb, pc);
181                 pb += es;
182                 pc -= es;
183         }
184         pn = (char *) a + n * es;
185         r = Min(pa - (char *) a, pb - pa);
186         vecswap(a, pb - r, r);
187         r = Min(pd - pc, pn - pd - es);
188         vecswap(pb, pn - r, r);
189         if ((r = pb - pa) > es)
190                 qsort_arg(a, r / es, es, cmp, arg);
191         if ((r = pd - pc) > es)
192         {
193                 /* Iterate rather than recurse to save stack space */
194                 a = pn - r;
195                 n = r / es;
196                 goto loop;
197         }
198 /*              qsort_arg(pn - r, r / es, es, cmp, arg);*/
199 }