LCOV - code coverage report
Current view: top level - src/backend/access/nbtree - nbtcompare.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 113 123 91.9 %
Date: 2025-01-18 05:15:39 Functions: 18 19 94.7 %
Legend: Lines: hit not hit

          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             : }

Generated by: LCOV version 1.14