LCOV - code coverage report
Current view: top level - src/test/modules/test_int128 - test_int128.c (source / functions) Hit Total Coverage
Test: PostgreSQL 19devel Lines: 77 123 62.6 %
Date: 2025-08-09 08:18:06 Functions: 2 2 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * test_int128.c
       4             :  *    Testbed for roll-our-own 128-bit integer arithmetic.
       5             :  *
       6             :  * This is a standalone test program that compares the behavior of an
       7             :  * implementation in int128.h to an (assumed correct) int128 native type.
       8             :  *
       9             :  * Copyright (c) 2017-2025, PostgreSQL Global Development Group
      10             :  *
      11             :  *
      12             :  * IDENTIFICATION
      13             :  *    src/test/modules/test_int128/test_int128.c
      14             :  *
      15             :  *-------------------------------------------------------------------------
      16             :  */
      17             : 
      18             : #include "postgres_fe.h"
      19             : 
      20             : #include <time.h>
      21             : 
      22             : /* Require a native int128 type */
      23             : #ifdef HAVE_INT128
      24             : 
      25             : /*
      26             :  * By default, we test the non-native implementation in int128.h; but
      27             :  * by predefining USE_NATIVE_INT128 to 1, you can test the native
      28             :  * implementation, just to be sure.
      29             :  */
      30             : #ifndef USE_NATIVE_INT128
      31             : #define USE_NATIVE_INT128 0
      32             : #endif
      33             : 
      34             : #include "common/int128.h"
      35             : #include "common/pg_prng.h"
      36             : 
      37             : /*
      38             :  * We assume the parts of this union are laid out compatibly.
      39             :  */
      40             : typedef union
      41             : {
      42             :     int128      i128;
      43             :     INT128      I128;
      44             :     struct
      45             :     {
      46             : #ifdef WORDS_BIGENDIAN
      47             :         int64       hi;
      48             :         uint64      lo;
      49             : #else
      50             :         uint64      lo;
      51             :         int64       hi;
      52             : #endif
      53             :     }           hl;
      54             : }           test128;
      55             : 
      56             : #define INT128_HEX_FORMAT   "%016" PRIx64 "%016" PRIx64
      57             : 
      58             : /*
      59             :  * Control version of comparator.
      60             :  */
      61             : static inline int
      62     4000000 : my_int128_compare(int128 x, int128 y)
      63             : {
      64     4000000 :     if (x < y)
      65     1998480 :         return -1;
      66     2001520 :     if (x > y)
      67     2001520 :         return 1;
      68           0 :     return 0;
      69             : }
      70             : 
      71             : /*
      72             :  * Main program.
      73             :  *
      74             :  * Generates a lot of random numbers and tests the implementation for each.
      75             :  * The results should be reproducible, since we use a fixed PRNG seed.
      76             :  *
      77             :  * You can give a loop count if you don't like the default 1B iterations.
      78             :  */
      79             : int
      80           2 : main(int argc, char **argv)
      81             : {
      82             :     long        count;
      83             : 
      84           2 :     pg_prng_seed(&pg_global_prng_state, (uint64) time(NULL));
      85             : 
      86           2 :     if (argc >= 2)
      87           2 :         count = strtol(argv[1], NULL, 0);
      88             :     else
      89           0 :         count = 1000000000;
      90             : 
      91     2000002 :     while (count-- > 0)
      92             :     {
      93     2000000 :         int64       x = pg_prng_uint64(&pg_global_prng_state);
      94     2000000 :         int64       y = pg_prng_uint64(&pg_global_prng_state);
      95     2000000 :         int64       z = pg_prng_uint64(&pg_global_prng_state);
      96     2000000 :         int64       w = pg_prng_uint64(&pg_global_prng_state);
      97     2000000 :         int32       z32 = (int32) z;
      98             :         test128     t1;
      99             :         test128     t2;
     100             :         test128     t3;
     101             :         int32       r1;
     102             :         int32       r2;
     103             : 
     104             :         /* check unsigned addition */
     105     2000000 :         t1.hl.hi = x;
     106     2000000 :         t1.hl.lo = y;
     107     2000000 :         t2 = t1;
     108     2000000 :         t1.i128 += (int128) (uint64) z;
     109     2000000 :         int128_add_uint64(&t2.I128, (uint64) z);
     110             : 
     111     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     112             :         {
     113           0 :             printf(INT128_HEX_FORMAT " + unsigned %016" PRIx64 "\n", x, y, z);
     114           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     115           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     116           0 :             return 1;
     117             :         }
     118             : 
     119             :         /* check signed addition */
     120     2000000 :         t1.hl.hi = x;
     121     2000000 :         t1.hl.lo = y;
     122     2000000 :         t2 = t1;
     123     2000000 :         t1.i128 += (int128) z;
     124     2000000 :         int128_add_int64(&t2.I128, z);
     125             : 
     126     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     127             :         {
     128           0 :             printf(INT128_HEX_FORMAT " + signed %016" PRIx64 "\n", x, y, z);
     129           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     130           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     131           0 :             return 1;
     132             :         }
     133             : 
     134             :         /* check 128-bit signed addition */
     135     2000000 :         t1.hl.hi = x;
     136     2000000 :         t1.hl.lo = y;
     137     2000000 :         t2 = t1;
     138     2000000 :         t3.hl.hi = z;
     139     2000000 :         t3.hl.lo = w;
     140     2000000 :         t1.i128 += t3.i128;
     141     2000000 :         int128_add_int128(&t2.I128, t3.I128);
     142             : 
     143     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     144             :         {
     145           0 :             printf(INT128_HEX_FORMAT " + " INT128_HEX_FORMAT "\n", x, y, z, w);
     146           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     147           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     148           0 :             return 1;
     149             :         }
     150             : 
     151             :         /* check unsigned subtraction */
     152     2000000 :         t1.hl.hi = x;
     153     2000000 :         t1.hl.lo = y;
     154     2000000 :         t2 = t1;
     155     2000000 :         t1.i128 -= (int128) (uint64) z;
     156     2000000 :         int128_sub_uint64(&t2.I128, (uint64) z);
     157             : 
     158     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     159             :         {
     160           0 :             printf(INT128_HEX_FORMAT " - unsigned %016" PRIx64 "\n", x, y, z);
     161           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     162           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     163           0 :             return 1;
     164             :         }
     165             : 
     166             :         /* check signed subtraction */
     167     2000000 :         t1.hl.hi = x;
     168     2000000 :         t1.hl.lo = y;
     169     2000000 :         t2 = t1;
     170     2000000 :         t1.i128 -= (int128) z;
     171     2000000 :         int128_sub_int64(&t2.I128, z);
     172             : 
     173     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     174             :         {
     175           0 :             printf(INT128_HEX_FORMAT " - signed %016" PRIx64 "\n", x, y, z);
     176           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     177           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     178           0 :             return 1;
     179             :         }
     180             : 
     181             :         /* check 64x64-bit multiply-add */
     182     2000000 :         t1.hl.hi = x;
     183     2000000 :         t1.hl.lo = y;
     184     2000000 :         t2 = t1;
     185     2000000 :         t1.i128 += (int128) z * (int128) w;
     186     2000000 :         int128_add_int64_mul_int64(&t2.I128, z, w);
     187             : 
     188     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     189             :         {
     190           0 :             printf(INT128_HEX_FORMAT " + %016" PRIx64 " * %016" PRIx64 "\n", x, y, z, w);
     191           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     192           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     193           0 :             return 1;
     194             :         }
     195             : 
     196             :         /* check 64x64-bit multiply-subtract */
     197     2000000 :         t1.hl.hi = x;
     198     2000000 :         t1.hl.lo = y;
     199     2000000 :         t2 = t1;
     200     2000000 :         t1.i128 -= (int128) z * (int128) w;
     201     2000000 :         int128_sub_int64_mul_int64(&t2.I128, z, w);
     202             : 
     203     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     204             :         {
     205           0 :             printf(INT128_HEX_FORMAT " - %016" PRIx64 " * %016" PRIx64 "\n", x, y, z, w);
     206           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     207           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     208           0 :             return 1;
     209             :         }
     210             : 
     211             :         /* check 128/32-bit division */
     212     2000000 :         t3.hl.hi = x;
     213     2000000 :         t3.hl.lo = y;
     214     2000000 :         t1.i128 = t3.i128 / z32;
     215     2000000 :         r1 = (int32) (t3.i128 % z32);
     216     2000000 :         t2 = t3;
     217     2000000 :         int128_div_mod_int32(&t2.I128, z32, &r2);
     218             : 
     219     2000000 :         if (t1.hl.hi != t2.hl.hi || t1.hl.lo != t2.hl.lo)
     220             :         {
     221           0 :             printf(INT128_HEX_FORMAT " / signed %08X\n", t3.hl.hi, t3.hl.lo, z32);
     222           0 :             printf("native = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     223           0 :             printf("result = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     224           0 :             return 1;
     225             :         }
     226     2000000 :         if (r1 != r2)
     227             :         {
     228           0 :             printf(INT128_HEX_FORMAT " %% signed %08X\n", t3.hl.hi, t3.hl.lo, z32);
     229           0 :             printf("native = %08X\n", r1);
     230           0 :             printf("result = %08X\n", r2);
     231           0 :             return 1;
     232             :         }
     233             : 
     234             :         /* check comparison */
     235     2000000 :         t1.hl.hi = x;
     236     2000000 :         t1.hl.lo = y;
     237     2000000 :         t2.hl.hi = z;
     238     2000000 :         t2.hl.lo = w;
     239             : 
     240     4000000 :         if (my_int128_compare(t1.i128, t2.i128) !=
     241     2000000 :             int128_compare(t1.I128, t2.I128))
     242             :         {
     243           0 :             printf("comparison failure: %d vs %d\n",
     244             :                    my_int128_compare(t1.i128, t2.i128),
     245             :                    int128_compare(t1.I128, t2.I128));
     246           0 :             printf("arg1 = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     247           0 :             printf("arg2 = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     248           0 :             return 1;
     249             :         }
     250             : 
     251             :         /* check case with identical hi parts; above will hardly ever hit it */
     252     2000000 :         t2.hl.hi = x;
     253             : 
     254     4000000 :         if (my_int128_compare(t1.i128, t2.i128) !=
     255     2000000 :             int128_compare(t1.I128, t2.I128))
     256             :         {
     257           0 :             printf("comparison failure: %d vs %d\n",
     258             :                    my_int128_compare(t1.i128, t2.i128),
     259             :                    int128_compare(t1.I128, t2.I128));
     260           0 :             printf("arg1 = " INT128_HEX_FORMAT "\n", t1.hl.hi, t1.hl.lo);
     261           0 :             printf("arg2 = " INT128_HEX_FORMAT "\n", t2.hl.hi, t2.hl.lo);
     262           0 :             return 1;
     263             :         }
     264             :     }
     265             : 
     266           2 :     return 0;
     267             : }
     268             : 
     269             : #else                           /* ! HAVE_INT128 */
     270             : 
     271             : /*
     272             :  * For now, do nothing if we don't have a native int128 type.
     273             :  */
     274             : int
     275             : main(int argc, char **argv)
     276             : {
     277             :     printf("skipping tests: no native int128 type\n");
     278             :     return 0;
     279             : }
     280             : 
     281             : #endif

Generated by: LCOV version 1.16