LCOV - code coverage report
Current view: top level - src/backend/optimizer/geqo - geqo_main.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 40 45 88.9 %
Date: 2025-01-18 05:15:39 Functions: 3 3 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*------------------------------------------------------------------------
       2             :  *
       3             :  * geqo_main.c
       4             :  *    solution to the query optimization problem
       5             :  *    by means of a Genetic Algorithm (GA)
       6             :  *
       7             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       8             :  * Portions Copyright (c) 1994, Regents of the University of California
       9             :  *
      10             :  * src/backend/optimizer/geqo/geqo_main.c
      11             :  *
      12             :  *-------------------------------------------------------------------------
      13             :  */
      14             : 
      15             : /* contributed by:
      16             :    =*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
      17             :    *  Martin Utesch              * Institute of Automatic Control      *
      18             :    =                             = University of Mining and Technology =
      19             :    *  utesch@aut.tu-freiberg.de  * Freiberg, Germany                   *
      20             :    =*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=
      21             :  */
      22             : 
      23             : /* -- parts of this are adapted from D. Whitley's Genitor algorithm -- */
      24             : 
      25             : #include "postgres.h"
      26             : 
      27             : #include <math.h>
      28             : 
      29             : #include "optimizer/geqo.h"
      30             : 
      31             : #include "optimizer/geqo_misc.h"
      32             : #if defined(CX)
      33             : #include "optimizer/geqo_mutation.h"
      34             : #endif
      35             : #include "optimizer/geqo_pool.h"
      36             : #include "optimizer/geqo_random.h"
      37             : #include "optimizer/geqo_recombination.h"
      38             : #include "optimizer/geqo_selection.h"
      39             : 
      40             : 
      41             : /*
      42             :  * Configuration options
      43             :  */
      44             : int         Geqo_effort;
      45             : int         Geqo_pool_size;
      46             : int         Geqo_generations;
      47             : double      Geqo_selection_bias;
      48             : double      Geqo_seed;
      49             : 
      50             : 
      51             : static int  gimme_pool_size(int nr_rel);
      52             : static int  gimme_number_generations(int pool_size);
      53             : 
      54             : /* complain if no recombination mechanism is #define'd */
      55             : #if !defined(ERX) && \
      56             :     !defined(PMX) && \
      57             :     !defined(CX)  && \
      58             :     !defined(PX)  && \
      59             :     !defined(OX1) && \
      60             :     !defined(OX2)
      61             : #error "must choose one GEQO recombination mechanism in geqo.h"
      62             : #endif
      63             : 
      64             : 
      65             : /*
      66             :  * geqo
      67             :  *    solution of the query optimization problem
      68             :  *    similar to a constrained Traveling Salesman Problem (TSP)
      69             :  */
      70             : 
      71             : RelOptInfo *
      72           6 : geqo(PlannerInfo *root, int number_of_rels, List *initial_rels)
      73             : {
      74             :     GeqoPrivateData private;
      75             :     int         generation;
      76             :     Chromosome *momma;
      77             :     Chromosome *daddy;
      78             :     Chromosome *kid;
      79             :     Pool       *pool;
      80             :     int         pool_size,
      81             :                 number_generations;
      82             : 
      83             : #ifdef GEQO_DEBUG
      84             :     int         status_interval;
      85             : #endif
      86             :     Gene       *best_tour;
      87             :     RelOptInfo *best_rel;
      88             : 
      89             : #if defined(ERX)
      90             :     Edge       *edge_table;     /* list of edges */
      91           6 :     int         edge_failures = 0;
      92             : #endif
      93             : #if defined(CX) || defined(PX) || defined(OX1) || defined(OX2)
      94             :     City       *city_table;     /* list of cities */
      95             : #endif
      96             : #if defined(CX)
      97             :     int         cycle_diffs = 0;
      98             :     int         mutations = 0;
      99             : #endif
     100             : 
     101             : /* set up private information */
     102           6 :     root->join_search_private = &private;
     103           6 :     private.initial_rels = initial_rels;
     104             : 
     105             : /* initialize private number generator */
     106           6 :     geqo_set_seed(root, Geqo_seed);
     107             : 
     108             : /* set GA parameters */
     109           6 :     pool_size = gimme_pool_size(number_of_rels);
     110           6 :     number_generations = gimme_number_generations(pool_size);
     111             : #ifdef GEQO_DEBUG
     112             :     status_interval = 10;
     113             : #endif
     114             : 
     115             : /* allocate genetic pool memory */
     116           6 :     pool = alloc_pool(root, pool_size, number_of_rels);
     117             : 
     118             : /* random initialization of the pool */
     119           6 :     random_init_pool(root, pool);
     120             : 
     121             : /* sort the pool according to cheapest path as fitness */
     122           6 :     sort_pool(root, pool);      /* we have to do it only one time, since all
     123             :                                  * kids replace the worst individuals in
     124             :                                  * future (-> geqo_pool.c:spread_chromo ) */
     125             : 
     126             : #ifdef GEQO_DEBUG
     127             :     elog(DEBUG1, "GEQO selected %d pool entries, best %.2f, worst %.2f",
     128             :          pool_size,
     129             :          pool->data[0].worth,
     130             :          pool->data[pool_size - 1].worth);
     131             : #endif
     132             : 
     133             : /* allocate chromosome momma and daddy memory */
     134           6 :     momma = alloc_chromo(root, pool->string_length);
     135           6 :     daddy = alloc_chromo(root, pool->string_length);
     136             : 
     137             : #if defined (ERX)
     138             : #ifdef GEQO_DEBUG
     139             :     elog(DEBUG2, "using edge recombination crossover [ERX]");
     140             : #endif
     141             : /* allocate edge table memory */
     142           6 :     edge_table = alloc_edge_table(root, pool->string_length);
     143             : #elif defined(PMX)
     144             : #ifdef GEQO_DEBUG
     145             :     elog(DEBUG2, "using partially matched crossover [PMX]");
     146             : #endif
     147             : /* allocate chromosome kid memory */
     148             :     kid = alloc_chromo(root, pool->string_length);
     149             : #elif defined(CX)
     150             : #ifdef GEQO_DEBUG
     151             :     elog(DEBUG2, "using cycle crossover [CX]");
     152             : #endif
     153             : /* allocate city table memory */
     154             :     kid = alloc_chromo(root, pool->string_length);
     155             :     city_table = alloc_city_table(root, pool->string_length);
     156             : #elif defined(PX)
     157             : #ifdef GEQO_DEBUG
     158             :     elog(DEBUG2, "using position crossover [PX]");
     159             : #endif
     160             : /* allocate city table memory */
     161             :     kid = alloc_chromo(root, pool->string_length);
     162             :     city_table = alloc_city_table(root, pool->string_length);
     163             : #elif defined(OX1)
     164             : #ifdef GEQO_DEBUG
     165             :     elog(DEBUG2, "using order crossover [OX1]");
     166             : #endif
     167             : /* allocate city table memory */
     168             :     kid = alloc_chromo(root, pool->string_length);
     169             :     city_table = alloc_city_table(root, pool->string_length);
     170             : #elif defined(OX2)
     171             : #ifdef GEQO_DEBUG
     172             :     elog(DEBUG2, "using order crossover [OX2]");
     173             : #endif
     174             : /* allocate city table memory */
     175             :     kid = alloc_chromo(root, pool->string_length);
     176             :     city_table = alloc_city_table(root, pool->string_length);
     177             : #endif
     178             : 
     179             : 
     180             : /* my pain main part: */
     181             : /* iterative optimization */
     182             : 
     183         390 :     for (generation = 0; generation < number_generations; generation++)
     184             :     {
     185             :         /* SELECTION: using linear bias function */
     186         384 :         geqo_selection(root, momma, daddy, pool, Geqo_selection_bias);
     187             : 
     188             : #if defined (ERX)
     189             :         /* EDGE RECOMBINATION CROSSOVER */
     190         384 :         gimme_edge_table(root, momma->string, daddy->string, pool->string_length, edge_table);
     191             : 
     192         384 :         kid = momma;
     193             : 
     194             :         /* are there any edge failures ? */
     195         384 :         edge_failures += gimme_tour(root, edge_table, kid->string, pool->string_length);
     196             : #elif defined(PMX)
     197             :         /* PARTIALLY MATCHED CROSSOVER */
     198             :         pmx(root, momma->string, daddy->string, kid->string, pool->string_length);
     199             : #elif defined(CX)
     200             :         /* CYCLE CROSSOVER */
     201             :         cycle_diffs = cx(root, momma->string, daddy->string, kid->string, pool->string_length, city_table);
     202             :         /* mutate the child */
     203             :         if (cycle_diffs == 0)
     204             :         {
     205             :             mutations++;
     206             :             geqo_mutation(root, kid->string, pool->string_length);
     207             :         }
     208             : #elif defined(PX)
     209             :         /* POSITION CROSSOVER */
     210             :         px(root, momma->string, daddy->string, kid->string, pool->string_length, city_table);
     211             : #elif defined(OX1)
     212             :         /* ORDER CROSSOVER */
     213             :         ox1(root, momma->string, daddy->string, kid->string, pool->string_length, city_table);
     214             : #elif defined(OX2)
     215             :         /* ORDER CROSSOVER */
     216             :         ox2(root, momma->string, daddy->string, kid->string, pool->string_length, city_table);
     217             : #endif
     218             : 
     219             : 
     220             :         /* EVALUATE FITNESS */
     221         384 :         kid->worth = geqo_eval(root, kid->string, pool->string_length);
     222             : 
     223             :         /* push the kid into the wilderness of life according to its worth */
     224         384 :         spread_chromo(root, kid, pool);
     225             : 
     226             : 
     227             : #ifdef GEQO_DEBUG
     228             :         if (status_interval && !(generation % status_interval))
     229             :             print_gen(stdout, pool, generation);
     230             : #endif
     231             : 
     232             :     }
     233             : 
     234             : 
     235             : #if defined(ERX)
     236             : #if defined(GEQO_DEBUG)
     237             :     if (edge_failures != 0)
     238             :         elog(LOG, "[GEQO] failures: %d, average: %d",
     239             :              edge_failures, (int) number_generations / edge_failures);
     240             :     else
     241             :         elog(LOG, "[GEQO] no edge failures detected");
     242             : #else
     243             :     /* suppress variable-set-but-not-used warnings from some compilers */
     244             :     (void) edge_failures;
     245             : #endif
     246             : #endif
     247             : 
     248             : #if defined(CX) && defined(GEQO_DEBUG)
     249             :     if (mutations != 0)
     250             :         elog(LOG, "[GEQO] mutations: %d, generations: %d",
     251             :              mutations, number_generations);
     252             :     else
     253             :         elog(LOG, "[GEQO] no mutations processed");
     254             : #endif
     255             : 
     256             : #ifdef GEQO_DEBUG
     257             :     print_pool(stdout, pool, 0, pool_size - 1);
     258             : #endif
     259             : 
     260             : #ifdef GEQO_DEBUG
     261             :     elog(DEBUG1, "GEQO best is %.2f after %d generations",
     262             :          pool->data[0].worth, number_generations);
     263             : #endif
     264             : 
     265             : 
     266             :     /*
     267             :      * got the cheapest query tree processed by geqo; first element of the
     268             :      * population indicates the best query tree
     269             :      */
     270           6 :     best_tour = (Gene *) pool->data[0].string;
     271             : 
     272           6 :     best_rel = gimme_tree(root, best_tour, pool->string_length);
     273             : 
     274           6 :     if (best_rel == NULL)
     275           0 :         elog(ERROR, "geqo failed to make a valid plan");
     276             : 
     277             :     /* DBG: show the query plan */
     278             : #ifdef NOT_USED
     279             :     print_plan(best_plan, root);
     280             : #endif
     281             : 
     282             :     /* ... free memory stuff */
     283           6 :     free_chromo(root, momma);
     284           6 :     free_chromo(root, daddy);
     285             : 
     286             : #if defined (ERX)
     287           6 :     free_edge_table(root, edge_table);
     288             : #elif defined(PMX)
     289             :     free_chromo(root, kid);
     290             : #elif defined(CX)
     291             :     free_chromo(root, kid);
     292             :     free_city_table(root, city_table);
     293             : #elif defined(PX)
     294             :     free_chromo(root, kid);
     295             :     free_city_table(root, city_table);
     296             : #elif defined(OX1)
     297             :     free_chromo(root, kid);
     298             :     free_city_table(root, city_table);
     299             : #elif defined(OX2)
     300             :     free_chromo(root, kid);
     301             :     free_city_table(root, city_table);
     302             : #endif
     303             : 
     304           6 :     free_pool(root, pool);
     305             : 
     306             :     /* ... clear root pointer to our private storage */
     307           6 :     root->join_search_private = NULL;
     308             : 
     309           6 :     return best_rel;
     310             : }
     311             : 
     312             : 
     313             : /*
     314             :  * Return either configured pool size or a good default
     315             :  *
     316             :  * The default is based on query size (no. of relations) = 2^(QS+1),
     317             :  * but constrained to a range based on the effort value.
     318             :  */
     319             : static int
     320           6 : gimme_pool_size(int nr_rel)
     321             : {
     322             :     double      size;
     323             :     int         minsize;
     324             :     int         maxsize;
     325             : 
     326             :     /* Legal pool size *must* be at least 2, so ignore attempt to select 1 */
     327           6 :     if (Geqo_pool_size >= 2)
     328           0 :         return Geqo_pool_size;
     329             : 
     330           6 :     size = pow(2.0, nr_rel + 1.0);
     331             : 
     332           6 :     maxsize = 50 * Geqo_effort; /* 50 to 500 individuals */
     333           6 :     if (size > maxsize)
     334           0 :         return maxsize;
     335             : 
     336           6 :     minsize = 10 * Geqo_effort; /* 10 to 100 individuals */
     337           6 :     if (size < minsize)
     338           0 :         return minsize;
     339             : 
     340           6 :     return (int) ceil(size);
     341             : }
     342             : 
     343             : 
     344             : /*
     345             :  * Return either configured number of generations or a good default
     346             :  *
     347             :  * The default is the same as the pool size, which allows us to be
     348             :  * sure that less-fit individuals get pushed out of the breeding
     349             :  * population before the run finishes.
     350             :  */
     351             : static int
     352           6 : gimme_number_generations(int pool_size)
     353             : {
     354           6 :     if (Geqo_generations > 0)
     355           0 :         return Geqo_generations;
     356             : 
     357           6 :     return pool_size;
     358             : }

Generated by: LCOV version 1.14