LCOV - code coverage report
Current view: top level - src/backend/nodes - equalfuncs.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 51 67 76.1 %
Date: 2025-01-18 04:15:08 Functions: 5 6 83.3 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * equalfuncs.c
       4             :  *    Equality functions to compare node trees.
       5             :  *
       6             :  * NOTE: it is intentional that parse location fields (in nodes that have
       7             :  * one) are not compared.  This is because we want, for example, a variable
       8             :  * "x" to be considered equal() to another reference to "x" in the query.
       9             :  *
      10             :  *
      11             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
      12             :  * Portions Copyright (c) 1994, Regents of the University of California
      13             :  *
      14             :  * IDENTIFICATION
      15             :  *    src/backend/nodes/equalfuncs.c
      16             :  *
      17             :  *-------------------------------------------------------------------------
      18             :  */
      19             : 
      20             : #include "postgres.h"
      21             : 
      22             : #include "miscadmin.h"
      23             : #include "utils/datum.h"
      24             : 
      25             : 
      26             : /*
      27             :  * Macros to simplify comparison of different kinds of fields.  Use these
      28             :  * wherever possible to reduce the chance for silly typos.  Note that these
      29             :  * hard-wire the convention that the local variables in an Equal routine are
      30             :  * named 'a' and 'b'.
      31             :  */
      32             : 
      33             : /* Compare a simple scalar field (int, float, bool, enum, etc) */
      34             : #define COMPARE_SCALAR_FIELD(fldname) \
      35             :     do { \
      36             :         if (a->fldname != b->fldname) \
      37             :             return false; \
      38             :     } while (0)
      39             : 
      40             : /* Compare a field that is a pointer to some kind of Node or Node tree */
      41             : #define COMPARE_NODE_FIELD(fldname) \
      42             :     do { \
      43             :         if (!equal(a->fldname, b->fldname)) \
      44             :             return false; \
      45             :     } while (0)
      46             : 
      47             : /* Compare a field that is a pointer to a Bitmapset */
      48             : #define COMPARE_BITMAPSET_FIELD(fldname) \
      49             :     do { \
      50             :         if (!bms_equal(a->fldname, b->fldname)) \
      51             :             return false; \
      52             :     } while (0)
      53             : 
      54             : /* Compare a field that is a pointer to a C string, or perhaps NULL */
      55             : #define COMPARE_STRING_FIELD(fldname) \
      56             :     do { \
      57             :         if (!equalstr(a->fldname, b->fldname)) \
      58             :             return false; \
      59             :     } while (0)
      60             : 
      61             : /* Macro for comparing string fields that might be NULL */
      62             : #define equalstr(a, b)  \
      63             :     (((a) != NULL && (b) != NULL) ? (strcmp(a, b) == 0) : (a) == (b))
      64             : 
      65             : /* Compare a field that is an inline array */
      66             : #define COMPARE_ARRAY_FIELD(fldname) \
      67             :     do { \
      68             :         if (memcmp(a->fldname, b->fldname, sizeof(a->fldname)) != 0) \
      69             :             return false; \
      70             :     } while (0)
      71             : 
      72             : /* Compare a field that is a pointer to a simple palloc'd object of size sz */
      73             : #define COMPARE_POINTER_FIELD(fldname, sz) \
      74             :     do { \
      75             :         if (memcmp(a->fldname, b->fldname, (sz)) != 0) \
      76             :             return false; \
      77             :     } while (0)
      78             : 
      79             : /* Compare a parse location field (this is a no-op, per note above) */
      80             : #define COMPARE_LOCATION_FIELD(fldname) \
      81             :     ((void) 0)
      82             : 
      83             : /* Compare a CoercionForm field (also a no-op, per comment in primnodes.h) */
      84             : #define COMPARE_COERCIONFORM_FIELD(fldname) \
      85             :     ((void) 0)
      86             : 
      87             : 
      88             : #include "equalfuncs.funcs.c"
      89             : 
      90             : 
      91             : /*
      92             :  * Support functions for nodes with custom_copy_equal attribute
      93             :  */
      94             : 
      95             : static bool
      96     2271336 : _equalConst(const Const *a, const Const *b)
      97             : {
      98     2271336 :     COMPARE_SCALAR_FIELD(consttype);
      99     2242302 :     COMPARE_SCALAR_FIELD(consttypmod);
     100     2242282 :     COMPARE_SCALAR_FIELD(constcollid);
     101     2236450 :     COMPARE_SCALAR_FIELD(constlen);
     102     2236450 :     COMPARE_SCALAR_FIELD(constisnull);
     103     2236296 :     COMPARE_SCALAR_FIELD(constbyval);
     104             :     COMPARE_LOCATION_FIELD(location);
     105             : 
     106             :     /*
     107             :      * We treat all NULL constants of the same type as equal. Someday this
     108             :      * might need to change?  But datumIsEqual doesn't work on nulls, so...
     109             :      */
     110     2236296 :     if (a->constisnull)
     111      131792 :         return true;
     112     2104504 :     return datumIsEqual(a->constvalue, b->constvalue,
     113     2104504 :                         a->constbyval, a->constlen);
     114             : }
     115             : 
     116             : static bool
     117           0 : _equalExtensibleNode(const ExtensibleNode *a, const ExtensibleNode *b)
     118             : {
     119             :     const ExtensibleNodeMethods *methods;
     120             : 
     121           0 :     COMPARE_STRING_FIELD(extnodename);
     122             : 
     123             :     /* At this point, we know extnodename is the same for both nodes. */
     124           0 :     methods = GetExtensibleNodeMethods(a->extnodename, false);
     125             : 
     126             :     /* compare the private fields */
     127           0 :     if (!methods->nodeEqual(a, b))
     128           0 :         return false;
     129             : 
     130           0 :     return true;
     131             : }
     132             : 
     133             : static bool
     134     2660866 : _equalA_Const(const A_Const *a, const A_Const *b)
     135             : {
     136     2660866 :     COMPARE_SCALAR_FIELD(isnull);
     137             :     /* Hack for in-line val field.  Also val is not valid if isnull is true */
     138     2660866 :     if (!a->isnull &&
     139     2502394 :         !equal(&a->val, &b->val))
     140           0 :         return false;
     141             :     COMPARE_LOCATION_FIELD(location);
     142             : 
     143     2660866 :     return true;
     144             : }
     145             : 
     146             : static bool
     147       43248 : _equalBitmapset(const Bitmapset *a, const Bitmapset *b)
     148             : {
     149       43248 :     return bms_equal(a, b);
     150             : }
     151             : 
     152             : /*
     153             :  * Lists are handled specially
     154             :  */
     155             : static bool
     156    28136002 : _equalList(const List *a, const List *b)
     157             : {
     158             :     const ListCell *item_a;
     159             :     const ListCell *item_b;
     160             : 
     161             :     /*
     162             :      * Try to reject by simple scalar checks before grovelling through all the
     163             :      * list elements...
     164             :      */
     165    28136002 :     COMPARE_SCALAR_FIELD(type);
     166    28136002 :     COMPARE_SCALAR_FIELD(length);
     167             : 
     168             :     /*
     169             :      * We place the switch outside the loop for the sake of efficiency; this
     170             :      * may not be worth doing...
     171             :      */
     172    28015430 :     switch (a->type)
     173             :     {
     174    23294468 :         case T_List:
     175    79218980 :             forboth(item_a, a, item_b, b)
     176             :             {
     177    56110686 :                 if (!equal(lfirst(item_a), lfirst(item_b)))
     178      186174 :                     return false;
     179             :             }
     180    23108294 :             break;
     181      374664 :         case T_IntList:
     182     5920846 :             forboth(item_a, a, item_b, b)
     183             :             {
     184     5546198 :                 if (lfirst_int(item_a) != lfirst_int(item_b))
     185          16 :                     return false;
     186             :             }
     187      374648 :             break;
     188     4346298 :         case T_OidList:
     189     8134766 :             forboth(item_a, a, item_b, b)
     190             :             {
     191     4689646 :                 if (lfirst_oid(item_a) != lfirst_oid(item_b))
     192      901178 :                     return false;
     193             :             }
     194     3445120 :             break;
     195           0 :         case T_XidList:
     196           0 :             forboth(item_a, a, item_b, b)
     197             :             {
     198           0 :                 if (lfirst_xid(item_a) != lfirst_xid(item_b))
     199           0 :                     return false;
     200             :             }
     201           0 :             break;
     202           0 :         default:
     203           0 :             elog(ERROR, "unrecognized list node type: %d",
     204             :                  (int) a->type);
     205             :             return false;       /* keep compiler quiet */
     206             :     }
     207             : 
     208             :     /*
     209             :      * If we got here, we should have run out of elements of both lists
     210             :      */
     211             :     Assert(item_a == NULL);
     212             :     Assert(item_b == NULL);
     213             : 
     214    26928062 :     return true;
     215             : }
     216             : 
     217             : 
     218             : /*
     219             :  * equal
     220             :  *    returns whether two nodes are equal
     221             :  */
     222             : bool
     223   202390144 : equal(const void *a, const void *b)
     224             : {
     225             :     bool        retval;
     226             : 
     227   202390144 :     if (a == b)
     228    86934226 :         return true;
     229             : 
     230             :     /*
     231             :      * note that a!=b, so only one of them can be NULL
     232             :      */
     233   115455918 :     if (a == NULL || b == NULL)
     234      198276 :         return false;
     235             : 
     236             :     /*
     237             :      * are they the same type of nodes?
     238             :      */
     239   115257642 :     if (nodeTag(a) != nodeTag(b))
     240     1847794 :         return false;
     241             : 
     242             :     /* Guard against stack overflow due to overly complex expressions */
     243   113409848 :     check_stack_depth();
     244             : 
     245   113409848 :     switch (nodeTag(a))
     246             :     {
     247             : #include "equalfuncs.switch.c"
     248             : 
     249    28136002 :         case T_List:
     250             :         case T_IntList:
     251             :         case T_OidList:
     252             :         case T_XidList:
     253    28136002 :             retval = _equalList(a, b);
     254    28136002 :             break;
     255             : 
     256           0 :         default:
     257           0 :             elog(ERROR, "unrecognized node type: %d",
     258             :                  (int) nodeTag(a));
     259             :             retval = false;     /* keep compiler quiet */
     260             :             break;
     261             :     }
     262             : 
     263   113409848 :     return retval;
     264             : }

Generated by: LCOV version 1.14