Line data Source code
1 : /*------------------------------------------------------------------------- 2 : * 3 : * nbtcompare.c 4 : * Comparison functions for btree access method. 5 : * 6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group 7 : * Portions Copyright (c) 1994, Regents of the University of California 8 : * 9 : * 10 : * IDENTIFICATION 11 : * src/backend/access/nbtree/nbtcompare.c 12 : * 13 : * NOTES 14 : * 15 : * These functions are stored in pg_amproc. For each operator class 16 : * defined on btrees, they compute 17 : * 18 : * compare(a, b): 19 : * < 0 if a < b, 20 : * = 0 if a == b, 21 : * > 0 if a > b. 22 : * 23 : * The result is always an int32 regardless of the input datatype. 24 : * 25 : * Although any negative int32 is acceptable for reporting "<", 26 : * and any positive int32 is acceptable for reporting ">", routines 27 : * that work on 32-bit or wider datatypes can't just return "a - b". 28 : * That could overflow and give the wrong answer. 29 : * 30 : * NOTE: it is critical that the comparison function impose a total order 31 : * on all non-NULL values of the data type, and that the datatype's 32 : * boolean comparison operators (= < >= etc) yield results consistent 33 : * with the comparison routine. Otherwise bad behavior may ensue. 34 : * (For example, the comparison operators must NOT punt when faced with 35 : * NAN or other funny values; you must devise some collation sequence for 36 : * all such values.) If the datatype is not trivial, this is most 37 : * reliably done by having the boolean operators invoke the same 38 : * three-way comparison code that the btree function does. Therefore, 39 : * this file contains only btree support for "trivial" datatypes --- 40 : * all others are in the /utils/adt/ files that implement their datatypes. 41 : * 42 : * NOTE: these routines must not leak memory, since memory allocated 43 : * during an index access won't be recovered till end of query. This 44 : * primarily affects comparison routines for toastable datatypes; 45 : * they have to be careful to free any detoasted copy of an input datum. 46 : * 47 : * NOTE: we used to forbid comparison functions from returning INT_MIN, 48 : * but that proves to be too error-prone because some platforms' versions 49 : * of memcmp() etc can return INT_MIN. As a means of stress-testing 50 : * callers, this file can be compiled with STRESS_SORT_INT_MIN defined 51 : * to cause many of these functions to return INT_MIN or INT_MAX instead of 52 : * their customary -1/+1. For production, though, that's not a good idea 53 : * since users or third-party code might expect the traditional results. 54 : *------------------------------------------------------------------------- 55 : */ 56 : #include "postgres.h" 57 : 58 : #include <limits.h> 59 : 60 : #include "utils/fmgrprotos.h" 61 : #include "utils/sortsupport.h" 62 : 63 : #ifdef STRESS_SORT_INT_MIN 64 : #define A_LESS_THAN_B INT_MIN 65 : #define A_GREATER_THAN_B INT_MAX 66 : #else 67 : #define A_LESS_THAN_B (-1) 68 : #define A_GREATER_THAN_B 1 69 : #endif 70 : 71 : 72 : Datum 73 48665552 : btboolcmp(PG_FUNCTION_ARGS) 74 : { 75 48665552 : bool a = PG_GETARG_BOOL(0); 76 48665552 : bool b = PG_GETARG_BOOL(1); 77 : 78 48665552 : PG_RETURN_INT32((int32) a - (int32) b); 79 : } 80 : 81 : Datum 82 10399796 : btint2cmp(PG_FUNCTION_ARGS) 83 : { 84 10399796 : int16 a = PG_GETARG_INT16(0); 85 10399796 : int16 b = PG_GETARG_INT16(1); 86 : 87 10399796 : PG_RETURN_INT32((int32) a - (int32) b); 88 : } 89 : 90 : static int 91 33903364 : btint2fastcmp(Datum x, Datum y, SortSupport ssup) 92 : { 93 33903364 : int16 a = DatumGetInt16(x); 94 33903364 : int16 b = DatumGetInt16(y); 95 : 96 33903364 : return (int) a - (int) b; 97 : } 98 : 99 : Datum 100 11372 : btint2sortsupport(PG_FUNCTION_ARGS) 101 : { 102 11372 : SortSupport ssup = (SortSupport) PG_GETARG_POINTER(0); 103 : 104 11372 : ssup->comparator = btint2fastcmp; 105 11372 : PG_RETURN_VOID(); 106 : } 107 : 108 : Datum 109 91529936 : btint4cmp(PG_FUNCTION_ARGS) 110 : { 111 91529936 : int32 a = PG_GETARG_INT32(0); 112 91529936 : int32 b = PG_GETARG_INT32(1); 113 : 114 91529936 : if (a > b) 115 35426140 : PG_RETURN_INT32(A_GREATER_THAN_B); 116 56103796 : else if (a == b) 117 19404394 : PG_RETURN_INT32(0); 118 : else 119 36699402 : PG_RETURN_INT32(A_LESS_THAN_B); 120 : } 121 : 122 : Datum 123 166582 : btint4sortsupport(PG_FUNCTION_ARGS) 124 : { 125 166582 : SortSupport ssup = (SortSupport) PG_GETARG_POINTER(0); 126 : 127 166582 : ssup->comparator = ssup_datum_int32_cmp; 128 166582 : PG_RETURN_VOID(); 129 : } 130 : 131 : Datum 132 16636144 : btint8cmp(PG_FUNCTION_ARGS) 133 : { 134 16636144 : int64 a = PG_GETARG_INT64(0); 135 16636144 : int64 b = PG_GETARG_INT64(1); 136 : 137 16636144 : if (a > b) 138 9721354 : PG_RETURN_INT32(A_GREATER_THAN_B); 139 6914790 : else if (a == b) 140 844844 : PG_RETURN_INT32(0); 141 : else 142 6069946 : PG_RETURN_INT32(A_LESS_THAN_B); 143 : } 144 : 145 : #if SIZEOF_DATUM < 8 146 : static int 147 : btint8fastcmp(Datum x, Datum y, SortSupport ssup) 148 : { 149 : int64 a = DatumGetInt64(x); 150 : int64 b = DatumGetInt64(y); 151 : 152 : if (a > b) 153 : return A_GREATER_THAN_B; 154 : else if (a == b) 155 : return 0; 156 : else 157 : return A_LESS_THAN_B; 158 : } 159 : #endif 160 : 161 : Datum 162 3124 : btint8sortsupport(PG_FUNCTION_ARGS) 163 : { 164 3124 : SortSupport ssup = (SortSupport) PG_GETARG_POINTER(0); 165 : 166 : #if SIZEOF_DATUM >= 8 167 3124 : ssup->comparator = ssup_datum_signed_cmp; 168 : #else 169 : ssup->comparator = btint8fastcmp; 170 : #endif 171 3124 : PG_RETURN_VOID(); 172 : } 173 : 174 : Datum 175 1452 : btint48cmp(PG_FUNCTION_ARGS) 176 : { 177 1452 : int32 a = PG_GETARG_INT32(0); 178 1452 : int64 b = PG_GETARG_INT64(1); 179 : 180 1452 : if (a > b) 181 498 : PG_RETURN_INT32(A_GREATER_THAN_B); 182 954 : else if (a == b) 183 66 : PG_RETURN_INT32(0); 184 : else 185 888 : PG_RETURN_INT32(A_LESS_THAN_B); 186 : } 187 : 188 : Datum 189 102 : btint84cmp(PG_FUNCTION_ARGS) 190 : { 191 102 : int64 a = PG_GETARG_INT64(0); 192 102 : int32 b = PG_GETARG_INT32(1); 193 : 194 102 : if (a > b) 195 30 : PG_RETURN_INT32(A_GREATER_THAN_B); 196 72 : else if (a == b) 197 30 : PG_RETURN_INT32(0); 198 : else 199 42 : PG_RETURN_INT32(A_LESS_THAN_B); 200 : } 201 : 202 : Datum 203 38462 : btint24cmp(PG_FUNCTION_ARGS) 204 : { 205 38462 : int16 a = PG_GETARG_INT16(0); 206 38462 : int32 b = PG_GETARG_INT32(1); 207 : 208 38462 : if (a > b) 209 22004 : PG_RETURN_INT32(A_GREATER_THAN_B); 210 16458 : else if (a == b) 211 4664 : PG_RETURN_INT32(0); 212 : else 213 11794 : PG_RETURN_INT32(A_LESS_THAN_B); 214 : } 215 : 216 : Datum 217 1366 : btint42cmp(PG_FUNCTION_ARGS) 218 : { 219 1366 : int32 a = PG_GETARG_INT32(0); 220 1366 : int16 b = PG_GETARG_INT16(1); 221 : 222 1366 : if (a > b) 223 76 : PG_RETURN_INT32(A_GREATER_THAN_B); 224 1290 : else if (a == b) 225 274 : PG_RETURN_INT32(0); 226 : else 227 1016 : PG_RETURN_INT32(A_LESS_THAN_B); 228 : } 229 : 230 : Datum 231 36 : btint28cmp(PG_FUNCTION_ARGS) 232 : { 233 36 : int16 a = PG_GETARG_INT16(0); 234 36 : int64 b = PG_GETARG_INT64(1); 235 : 236 36 : if (a > b) 237 0 : PG_RETURN_INT32(A_GREATER_THAN_B); 238 36 : else if (a == b) 239 0 : PG_RETURN_INT32(0); 240 : else 241 36 : PG_RETURN_INT32(A_LESS_THAN_B); 242 : } 243 : 244 : Datum 245 0 : btint82cmp(PG_FUNCTION_ARGS) 246 : { 247 0 : int64 a = PG_GETARG_INT64(0); 248 0 : int16 b = PG_GETARG_INT16(1); 249 : 250 0 : if (a > b) 251 0 : PG_RETURN_INT32(A_GREATER_THAN_B); 252 0 : else if (a == b) 253 0 : PG_RETURN_INT32(0); 254 : else 255 0 : PG_RETURN_INT32(A_LESS_THAN_B); 256 : } 257 : 258 : Datum 259 194779560 : btoidcmp(PG_FUNCTION_ARGS) 260 : { 261 194779560 : Oid a = PG_GETARG_OID(0); 262 194779560 : Oid b = PG_GETARG_OID(1); 263 : 264 194779560 : if (a > b) 265 51182876 : PG_RETURN_INT32(A_GREATER_THAN_B); 266 143596684 : else if (a == b) 267 46675088 : PG_RETURN_INT32(0); 268 : else 269 96921596 : PG_RETURN_INT32(A_LESS_THAN_B); 270 : } 271 : 272 : static int 273 210521388 : btoidfastcmp(Datum x, Datum y, SortSupport ssup) 274 : { 275 210521388 : Oid a = DatumGetObjectId(x); 276 210521388 : Oid b = DatumGetObjectId(y); 277 : 278 210521388 : if (a > b) 279 51823722 : return A_GREATER_THAN_B; 280 158697666 : else if (a == b) 281 106622584 : return 0; 282 : else 283 52075082 : return A_LESS_THAN_B; 284 : } 285 : 286 : Datum 287 109942 : btoidsortsupport(PG_FUNCTION_ARGS) 288 : { 289 109942 : SortSupport ssup = (SortSupport) PG_GETARG_POINTER(0); 290 : 291 109942 : ssup->comparator = btoidfastcmp; 292 109942 : PG_RETURN_VOID(); 293 : } 294 : 295 : Datum 296 6208648 : btoidvectorcmp(PG_FUNCTION_ARGS) 297 : { 298 6208648 : oidvector *a = (oidvector *) PG_GETARG_POINTER(0); 299 6208648 : oidvector *b = (oidvector *) PG_GETARG_POINTER(1); 300 : int i; 301 : 302 : /* We arbitrarily choose to sort first by vector length */ 303 6208648 : if (a->dim1 != b->dim1) 304 942618 : PG_RETURN_INT32(a->dim1 - b->dim1); 305 : 306 9193808 : for (i = 0; i < a->dim1; i++) 307 : { 308 7042360 : if (a->values[i] != b->values[i]) 309 : { 310 3114582 : if (a->values[i] > b->values[i]) 311 1573420 : PG_RETURN_INT32(A_GREATER_THAN_B); 312 : else 313 1541162 : PG_RETURN_INT32(A_LESS_THAN_B); 314 : } 315 : } 316 2151448 : PG_RETURN_INT32(0); 317 : } 318 : 319 : Datum 320 47562250 : btcharcmp(PG_FUNCTION_ARGS) 321 : { 322 47562250 : char a = PG_GETARG_CHAR(0); 323 47562250 : char b = PG_GETARG_CHAR(1); 324 : 325 : /* Be careful to compare chars as unsigned */ 326 47562250 : PG_RETURN_INT32((int32) ((uint8) a) - (int32) ((uint8) b)); 327 : }