LCOV - code coverage report
Current view: top level - src/test/modules/test_bloomfilter - test_bloomfilter.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 38 41 92.7 %
Date: 2025-01-18 04:15:08 Functions: 6 6 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*--------------------------------------------------------------------------
       2             :  *
       3             :  * test_bloomfilter.c
       4             :  *      Test false positive rate of Bloom filter.
       5             :  *
       6             :  * Copyright (c) 2018-2025, PostgreSQL Global Development Group
       7             :  *
       8             :  * IDENTIFICATION
       9             :  *      src/test/modules/test_bloomfilter/test_bloomfilter.c
      10             :  *
      11             :  * -------------------------------------------------------------------------
      12             :  */
      13             : #include "postgres.h"
      14             : 
      15             : #include "common/pg_prng.h"
      16             : #include "fmgr.h"
      17             : #include "lib/bloomfilter.h"
      18             : #include "miscadmin.h"
      19             : 
      20           2 : PG_MODULE_MAGIC;
      21             : 
      22             : /* Fits decimal representation of PG_INT64_MIN + 2 bytes: */
      23             : #define MAX_ELEMENT_BYTES       21
      24             : /* False positive rate WARNING threshold (1%): */
      25             : #define FPOSITIVE_THRESHOLD     0.01
      26             : 
      27             : 
      28             : /*
      29             :  * Populate an empty Bloom filter with "nelements" dummy strings.
      30             :  */
      31             : static void
      32           2 : populate_with_dummy_strings(bloom_filter *filter, int64 nelements)
      33             : {
      34             :     char        element[MAX_ELEMENT_BYTES];
      35             :     int64       i;
      36             : 
      37     1677724 :     for (i = 0; i < nelements; i++)
      38             :     {
      39     1677722 :         CHECK_FOR_INTERRUPTS();
      40             : 
      41     1677722 :         snprintf(element, sizeof(element), "i" INT64_FORMAT, i);
      42     1677722 :         bloom_add_element(filter, (unsigned char *) element, strlen(element));
      43             :     }
      44           2 : }
      45             : 
      46             : /*
      47             :  * Returns number of strings that are indicated as probably appearing in Bloom
      48             :  * filter that were in fact never added by populate_with_dummy_strings().
      49             :  * These are false positives.
      50             :  */
      51             : static int64
      52           2 : nfalsepos_for_missing_strings(bloom_filter *filter, int64 nelements)
      53             : {
      54             :     char        element[MAX_ELEMENT_BYTES];
      55           2 :     int64       nfalsepos = 0;
      56             :     int64       i;
      57             : 
      58     1677724 :     for (i = 0; i < nelements; i++)
      59             :     {
      60     1677722 :         CHECK_FOR_INTERRUPTS();
      61             : 
      62     1677722 :         snprintf(element, sizeof(element), "M" INT64_FORMAT, i);
      63     1677722 :         if (!bloom_lacks_element(filter, (unsigned char *) element,
      64             :                                  strlen(element)))
      65       13562 :             nfalsepos++;
      66             :     }
      67             : 
      68           2 :     return nfalsepos;
      69             : }
      70             : 
      71             : static void
      72           2 : create_and_test_bloom(int power, int64 nelements, int callerseed)
      73             : {
      74             :     int         bloom_work_mem;
      75             :     uint64      seed;
      76             :     int64       nfalsepos;
      77             :     bloom_filter *filter;
      78             : 
      79           2 :     bloom_work_mem = (1L << power) / 8L / 1024L;
      80             : 
      81           2 :     elog(DEBUG1, "bloom_work_mem (KB): %d", bloom_work_mem);
      82             : 
      83             :     /*
      84             :      * Generate random seed, or use caller's.  Seed should always be a
      85             :      * positive value less than or equal to PG_INT32_MAX, to ensure that any
      86             :      * random seed can be recreated through callerseed if the need arises.
      87             :      */
      88           2 :     seed = callerseed < 0 ? pg_prng_int32p(&pg_global_prng_state) : callerseed;
      89             : 
      90             :     /* Create Bloom filter, populate it, and report on false positive rate */
      91           2 :     filter = bloom_create(nelements, bloom_work_mem, seed);
      92           2 :     populate_with_dummy_strings(filter, nelements);
      93           2 :     nfalsepos = nfalsepos_for_missing_strings(filter, nelements);
      94             : 
      95           2 :     ereport((nfalsepos > nelements * FPOSITIVE_THRESHOLD) ? WARNING : DEBUG1,
      96             :             (errmsg_internal("seed: " UINT64_FORMAT " false positives: " INT64_FORMAT " (%.6f%%) bitset %.2f%% set",
      97             :                              seed, nfalsepos, (double) nfalsepos / nelements,
      98             :                              100.0 * bloom_prop_bits_set(filter))));
      99             : 
     100           2 :     bloom_free(filter);
     101           2 : }
     102             : 
     103           4 : PG_FUNCTION_INFO_V1(test_bloomfilter);
     104             : 
     105             : /*
     106             :  * SQL-callable entry point to perform all tests.
     107             :  *
     108             :  * If a 1% false positive threshold is not met, emits WARNINGs.
     109             :  *
     110             :  * See README for details of arguments.
     111             :  */
     112             : Datum
     113           2 : test_bloomfilter(PG_FUNCTION_ARGS)
     114             : {
     115           2 :     int         power = PG_GETARG_INT32(0);
     116           2 :     int64       nelements = PG_GETARG_INT64(1);
     117           2 :     int         seed = PG_GETARG_INT32(2);
     118           2 :     int         tests = PG_GETARG_INT32(3);
     119             :     int         i;
     120             : 
     121           2 :     if (power < 23 || power > 32)
     122           0 :         elog(ERROR, "power argument must be between 23 and 32 inclusive");
     123             : 
     124           2 :     if (tests <= 0)
     125           0 :         elog(ERROR, "invalid number of tests: %d", tests);
     126             : 
     127           2 :     if (nelements < 0)
     128           0 :         elog(ERROR, "invalid number of elements: %d", tests);
     129             : 
     130           4 :     for (i = 0; i < tests; i++)
     131             :     {
     132           2 :         elog(DEBUG1, "beginning test #%d...", i + 1);
     133             : 
     134           2 :         create_and_test_bloom(power, nelements, seed);
     135             :     }
     136             : 
     137           2 :     PG_RETURN_VOID();
     138             : }

Generated by: LCOV version 1.14