LCOV - code coverage report
Current view: top level - monetdb5/optimizer - opt_pushselect.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 272 525 51.8 %
Date: 2024-11-13 22:44:48 Functions: 5 5 100.0 %

          Line data    Source code
       1             : /*
       2             :  * SPDX-License-Identifier: MPL-2.0
       3             :  *
       4             :  * This Source Code Form is subject to the terms of the Mozilla Public
       5             :  * License, v. 2.0.  If a copy of the MPL was not distributed with this
       6             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       7             :  *
       8             :  * Copyright 2024 MonetDB Foundation;
       9             :  * Copyright August 2008 - 2023 MonetDB B.V.;
      10             :  * Copyright 1997 - July 2008 CWI.
      11             :  */
      12             : 
      13             : #include "monetdb_config.h"
      14             : #include "opt_pushselect.h"
      15             : #include "mal_interpreter.h"  /* for showErrors() */
      16             : 
      17             : static InstrPtr
      18       66075 : PushArgument(MalBlkPtr mb, InstrPtr p, int arg, int pos)
      19             : {
      20       66075 :         int i;
      21             : 
      22       66075 :         p = pushArgument(mb, p, arg);   /* push at end */
      23       66075 :         if (mb->errors == NULL) {
      24      264300 :                 for (i = p->argc - 1; i > pos; i--)
      25      198225 :                         getArg(p, i) = getArg(p, i - 1);
      26       66075 :                 getArg(p, pos) = arg;
      27             :         }
      28       66075 :         return p;
      29             : }
      30             : 
      31             : static InstrPtr
      32       25968 : ReplaceWithNil(MalBlkPtr mb, InstrPtr p, int pos)
      33             : {
      34       25968 :         p = pushNilBat(mb, p);  /* push at end */
      35       25968 :         getArg(p, pos) = getArg(p, p->argc - 1);
      36       25968 :         p->argc--;
      37       25968 :         return p;
      38             : }
      39             : 
      40             : 
      41             : #define MAX_TABLES 64
      42             : 
      43             : typedef struct subselect_t {
      44             :         int nr;
      45             :         int tid[MAX_TABLES];
      46             :         int subselect[MAX_TABLES];
      47             : } subselect_t;
      48             : 
      49             : static int
      50        6706 : subselect_add(subselect_t *subselects, int tid, int subselect)
      51             : {
      52        6706 :         int i;
      53             : 
      54        6938 :         for (i = 0; i < subselects->nr; i++) {
      55         543 :                 if (subselects->tid[i] == tid) {
      56         311 :                         if (subselects->subselect[i] == subselect)
      57             :                                 return i;
      58             :                         else
      59         311 :                                 return -1;
      60             :                 }
      61             :         }
      62        6395 :         if (i >= MAX_TABLES)
      63             :                 return -1;
      64        6395 :         subselects->nr++;
      65        6395 :         subselects->tid[i] = tid;
      66        6395 :         subselects->subselect[i] = subselect;
      67        6395 :         return i;
      68             : }
      69             : 
      70             : static int
      71    31968098 : lastbat_arg(MalBlkPtr mb, InstrPtr p)
      72             : {
      73    31968098 :         int i = 0;
      74    39687875 :         for (i = p->retc; i < p->argc; i++) {
      75    17861977 :                 int type = getArgType(mb, p, i);
      76    17861977 :                 if (!isaBatType(type))
      77             :                         break;
      78             :         }
      79    31968098 :         if (i < p->argc)
      80    10142226 :                 return i - 1;
      81             :         return 0;
      82             : }
      83             : 
      84             : /* check for updates in between assignment to variables newv and oldv */
      85             : static int
      86             : no_updates(InstrPtr *old, int *vars, int oldv, int newv)
      87             : {
      88             :         while (newv > oldv) {
      89             :                 InstrPtr q = old[vars[newv]];
      90             : 
      91             :                 if (isUpdateInstruction(q))
      92             :                         return 0;
      93             :                 newv = getArg(q, 1);
      94             :         }
      95             :         return 1;
      96             : }
      97             : 
      98             : #define isIntersect(p) (getModuleId(p) == algebraRef && getFunctionId(p) == intersectRef)
      99             : 
     100             : str
     101      445930 : OPTpushselectImplementation(Client cntxt, MalBlkPtr mb, MalStkPtr stk,
     102             :                                                         InstrPtr pci)
     103             : {
     104      445930 :         int i, j, limit, slimit, actions = 0, *vars, *nvars = NULL,
     105      445930 :                 *slices = NULL, push_down_delta = 0, nr_topn = 0, nr_likes = 0,
     106      445930 :                 no_mito = 0;
     107      445930 :         bool *rslices = NULL, *oclean = NULL;
     108      445930 :         InstrPtr p, *old = NULL;
     109      445930 :         subselect_t subselects;
     110      445930 :         str msg = MAL_SUCCEED;
     111             : 
     112      445930 :         subselects = (subselect_t) { 0 };
     113      445930 :         if (mb->errors)
     114           0 :                 throw(MAL, "optimizer.pushselect", "%s", mb->errors);
     115             : 
     116      445930 :         no_mito = !isOptimizerEnabled(mb, mitosisRef);
     117      445928 :         (void) stk;
     118      445928 :         vars = (int *) GDKzalloc(sizeof(int) * mb->vtop);
     119      445931 :         if (vars == NULL)
     120           0 :                 throw(MAL, "optimizer.pushselect", SQLSTATE(HY013) MAL_MALLOC_FAIL);
     121             : 
     122      445931 :         limit = mb->stop;
     123      445931 :         slimit = mb->ssize;
     124      445931 :         old = mb->stmt;
     125             : 
     126             :         /* check for bailout conditions */
     127    16829563 :         for (i = 1; mb->errors == NULL && i < limit; i++) {
     128    16395804 :                 int lastbat;
     129    16395804 :                 p = old[i];
     130             : 
     131    32641625 :                 for (j = 0; j < p->retc; j++) {
     132    16245821 :                         int res = getArg(p, j);
     133    16245821 :                         vars[res] = i;
     134             :                 }
     135             : 
     136    16395804 :                 if (getModuleId(p) == algebraRef
     137     1401271 :                         && ((!no_mito && getFunctionId(p) == intersectRef)
     138     1399518 :                                 || getFunctionId(p) == differenceRef)) {
     139       11847 :                         GDKfree(vars);
     140       11847 :                         goto wrapup;
     141             :                 }
     142             : 
     143    16383957 :                 if (isSlice(p))
     144       16730 :                         nr_topn++;
     145             : 
     146    16383843 :                 if (isLikeOp(p))
     147         318 :                         nr_likes++;
     148             : 
     149    16383807 :                 if (no_mito && isIntersect(p))
     150           8 :                         push_down_delta++;
     151             : 
     152    16383807 :                 if ((getModuleId(p) == sqlRef && getFunctionId(p) == deltaRef)
     153    16266433 :                         || (no_mito && getModuleId(p) == matRef
     154          14 :                                 && getFunctionId(p) == packRef && p->argc == (p->retc + 2)))
     155      117374 :                         push_down_delta++;
     156             : 
     157    16383807 :                 if ( /* DISABLES CODE */ (0) && getModuleId(p) == sqlRef && getFunctionId(p) == tidRef) {       /* rewrite equal table ids */
     158             :                         int sname = getArg(p, 2), tname = getArg(p, 3), s;
     159             : 
     160             :                         for (s = 0; s < subselects.nr; s++) {
     161             :                                 InstrPtr q = old[vars[subselects.tid[s]]];
     162             :                                 int Qsname = getArg(q, 2), Qtname = getArg(q, 3);
     163             : 
     164             :                                 if (no_updates(old, vars, getArg(q, 1), getArg(p, 1))
     165             :                                         && ((sname == Qsname && tname == Qtname)
     166             :                                                 || ( /* DISABLES CODE */ (0)
     167             :                                                         && strcmp(getVarConstant(mb, sname).val.sval,
     168             :                                                                           getVarConstant(mb, Qsname).val.sval) == 0
     169             :                                                         &&
     170             :                                                         strcmp(getVarConstant(mb, tname).val.sval,
     171             :                                                                    getVarConstant(mb,
     172             :                                                                                                   Qtname).val.sval) == 0))) {
     173             :                                         clrFunction(p);
     174             :                                         p->retc = 1;
     175             :                                         p->argc = 2;
     176             :                                         getArg(p, 1) = getArg(q, 0);
     177             :                                         break;
     178             :                                 }
     179             :                         }
     180             :                 }
     181    16383807 :                 lastbat = lastbat_arg(mb, p);
     182    16383807 :                 if (isSelect(p) && p->retc == 1 &&
     183      541443 :                         /* no cand list */ getArgType(mb, p,
     184             :                                                                                   lastbat) != newBatType(TYPE_oid)) {
     185      440146 :                         int i1 = getArg(p, 1), tid = 0;
     186      440146 :                         InstrPtr q = old[vars[i1]];
     187             : 
     188             :                         /* find the table ids */
     189      440146 :                         while (!tid) {
     190      457425 :                                 if (getModuleId(q) == algebraRef
     191       20976 :                                         && getFunctionId(q) == projectionRef) {
     192       18878 :                                         int i1 = getArg(q, 1);
     193       18878 :                                         InstrPtr s = old[vars[i1]];
     194             : 
     195       18878 :                                         if (getModuleId(s) == sqlRef && getFunctionId(s) == tidRef)
     196       18878 :                                                 tid = getArg(q, 1);
     197       18878 :                                         if (s->argc == 2 && s->retc == 1) {
     198           0 :                                                 int i1 = getArg(s, 1);
     199           0 :                                                 InstrPtr s = old[vars[i1]];
     200           0 :                                                 if (getModuleId(s) == sqlRef
     201           0 :                                                         && getFunctionId(s) == tidRef)
     202      440155 :                                                         tid = getArg(q, 1);
     203             :                                         }
     204             :                                         break;
     205      438547 :                                 } else if (isMapOp(q) && q->retc == 1 && q->argc >= 2
     206       17351 :                                                    && isaBatType(getArgType(mb, q, 1))) {
     207       17192 :                                         int i1 = getArg(q, 1);
     208       17192 :                                         q = old[vars[i1]];
     209      421364 :                                 } else if (isMapOp(q) && q->retc == 1 && q->argc >= 3
     210         158 :                                                    && isaBatType(getArgType(mb, q, 2))) {
     211          87 :                                         int i2 = getArg(q, 2);
     212          87 :                                         q = old[vars[i2]];
     213             :                                 } else {
     214             :                                         break;
     215             :                                 }
     216             :                         }
     217      440155 :                         if (tid && subselect_add(&subselects, tid, getArg(p, 0)) < 0) {
     218         311 :                                 GDKfree(vars);
     219         311 :                                 goto wrapup;
     220             :                         }
     221             :                 }
     222             :                 /* left hand side */
     223    16383632 :                 if ((ATOMIC_GET(&GDKdebug) & PUSHCANDMASK) && isMatJoinOp(p)
     224           0 :                         && p->retc == 2) {
     225           0 :                         int i1 = getArg(p, 2), tid = 0;
     226           0 :                         InstrPtr q = old[vars[i1]];
     227             : 
     228             :                         /* find the table ids */
     229           0 :                         while (!tid) {
     230           0 :                                 if (getModuleId(q) == algebraRef
     231           0 :                                         && getFunctionId(q) == projectionRef) {
     232           0 :                                         int i1 = getArg(q, 1);
     233           0 :                                         InstrPtr s = old[vars[i1]];
     234             : 
     235           0 :                                         if (getModuleId(s) == sqlRef && getFunctionId(s) == tidRef)
     236             :                                                 tid = getArg(q, 1);
     237             :                                         break;
     238           0 :                                 } else if (isMapOp(q) && q->argc >= 2
     239           0 :                                                    && isaBatType(getArgType(mb, q, 1))) {
     240           0 :                                         int i1 = getArg(q, 1);
     241           0 :                                         q = old[vars[i1]];
     242           0 :                                 } else if (isMapOp(q) && q->argc >= 3
     243           0 :                                                    && isaBatType(getArgType(mb, q, 2))) {
     244           0 :                                         int i2 = getArg(q, 2);
     245           0 :                                         q = old[vars[i2]];
     246             :                                 } else {
     247             :                                         break;
     248             :                                 }
     249             :                         }
     250           0 :                         if (tid && subselect_add(&subselects, tid, getArg(p, 0)) < 0) {
     251           0 :                                 GDKfree(vars);
     252           0 :                                 goto wrapup;
     253             :                         }
     254             :                 }
     255             :                 /* right hand side */
     256    16383632 :                 if ((ATOMIC_GET(&GDKdebug) & PUSHCANDMASK) && isMatJoinOp(p)
     257           0 :                         && p->retc == 2) {
     258           0 :                         int i1 = getArg(p, 3), tid = 0;
     259           0 :                         InstrPtr q = old[vars[i1]];
     260             : 
     261             :                         /* find the table ids */
     262           0 :                         while (!tid) {
     263           0 :                                 if (getModuleId(q) == algebraRef
     264           0 :                                         && getFunctionId(q) == projectionRef) {
     265           0 :                                         int i1 = getArg(q, 1);
     266           0 :                                         InstrPtr s = old[vars[i1]];
     267             : 
     268           0 :                                         if (getModuleId(s) == sqlRef && getFunctionId(s) == tidRef)
     269             :                                                 tid = getArg(q, 1);
     270             :                                         break;
     271           0 :                                 } else if (isMapOp(q) && q->argc >= 2
     272           0 :                                                    && isaBatType(getArgType(mb, q, 1))) {
     273           0 :                                         int i1 = getArg(q, 1);
     274           0 :                                         q = old[vars[i1]];
     275           0 :                                 } else if (isMapOp(q) && q->argc >= 3
     276           0 :                                                    && isaBatType(getArgType(mb, q, 2))) {
     277           0 :                                         int i2 = getArg(q, 2);
     278           0 :                                         q = old[vars[i2]];
     279             :                                 } else {
     280             :                                         break;
     281             :                                 }
     282             :                         }
     283           0 :                         if (tid && subselect_add(&subselects, tid, getArg(p, 1)) < 0) {
     284           0 :                                 GDKfree(vars);
     285           0 :                                 goto wrapup;
     286             :                         }
     287             :                 }
     288             :         }
     289             : 
     290      433759 :         if (nr_likes || subselects.nr) {
     291        5699 :                 if (newMalBlkStmt(mb, mb->ssize) < 0) {
     292           0 :                         GDKfree(vars);
     293           0 :                         goto wrapup;
     294             :                 }
     295             : 
     296        5700 :                 pushInstruction(mb, old[0]);
     297             : 
     298      304240 :                 for (i = 1; mb->errors == NULL && i < limit; i++) {
     299      292842 :                         p = old[i];
     300             : 
     301             :                         /* rewrite batalgebra.like + [theta]select -> likeselect */
     302      292842 :                         if (getModuleId(p) == algebraRef && p->retc == 1
     303       15109 :                                 && (getFunctionId(p) == selectRef
     304       15080 :                                         || getFunctionId(p) == thetaselectRef)) {
     305        7380 :                                 int var = getArg(p, 1);
     306        7380 :                                 InstrPtr q = mb->stmt[vars[var]];    /* BEWARE: the optimizer may not add or remove statements ! */
     307             : 
     308        7380 :                                 if (isLikeOp(q) && !isaBatType(getArgType(mb, q, 2)) && isVarConstant(mb, getArg(q, 2)) &&      /* pattern is a value */
     309           9 :                                         isVarConstant(mb, getArg(q, 3)) &&      /* escape is a value */
     310           9 :                                         isVarConstant(mb, getArg(q, 4)) &&      /* isensitive flag is a value */
     311           9 :                                         getArg(q, 0) == getArg(p,
     312             :                                                                                    1)
     313             :                                         /* the output variable from batalgebra.like is the input one for [theta]select */
     314             :                                         ) {
     315           9 :                                         int has_cand = (getArgType(mb, p, 2) == newBatType(TYPE_oid)),
     316           9 :                                                 offset = 0, anti = (getFunctionId(q)[0] == 'n');
     317           9 :                                         bit ignore_case = *(bit *) getVarValue(mb, getArg(q, 4)),
     318           9 :                                                 selectok = TRUE;
     319             : 
     320             :                                         /* TODO at the moment we cannot convert if the select statement has NULL semantics
     321             :                                            we can convert it into VAL is NULL or PATTERN is NULL or ESCAPE is NULL
     322             :                                          */
     323           9 :                                         if (getFunctionId(p) == selectRef
     324           0 :                                                 && isVarConstant(mb, getArg(p, 2 + has_cand))
     325           0 :                                                 && isVarConstant(mb, getArg(p, 3 + has_cand))
     326           0 :                                                 && isVarConstant(mb, getArg(p, 4 + has_cand))
     327           0 :                                                 && isVarConstant(mb, getArg(p, 5 + has_cand))
     328           0 :                                                 && isVarConstant(mb, getArg(p, 6 + has_cand))
     329           0 :                                                 && (p->argc < (has_cand ? 9 : 8)
     330           0 :                                                         || isVarConstant(mb, getArg(p, 7 + has_cand)))) {
     331           0 :                                                 bit low = *(bit *) getVarValue(mb,
     332             :                                                                                                            getArg(p, 2 + has_cand)),
     333           0 :                                                         high = *(bit *) getVarValue(mb,
     334             :                                                                                                                 getArg(p, 3 + has_cand));
     335           0 :                                                 bit li = *(bit *) getVarValue(mb,
     336             :                                                                                                           getArg(p, 4 + has_cand)),
     337           0 :                                                         hi = *(bit *) getVarValue(mb,
     338             :                                                                                                           getArg(p, 5 + has_cand));
     339           0 :                                                 bit santi = *(bit *) getVarValue(mb,
     340             :                                                                                                                  getArg(p, 6 + has_cand));
     341           0 :                                                 bit sunknown = (p->argc == (has_cand ? 9 : 8)) ? 0 : *(bit *) getVarValue(mb, getArg(p, 7 + has_cand));
     342             : 
     343             :                                                 /* semantic or not symmetric cases, it cannot be converted */
     344           0 :                                                 if (is_bit_nil(low) || is_bit_nil(li)
     345           0 :                                                         || is_bit_nil(santi) || low != high || li != hi
     346           0 :                                                         || sunknown)
     347           0 :                                                         selectok = FALSE;
     348             : 
     349             :                                                 /* there are no negative candidate lists so on = false situations swap anti flag */
     350           0 :                                                 if (low == 0)
     351           0 :                                                         anti = !anti;
     352           0 :                                                 if (li == 0)
     353           0 :                                                         anti = !anti;
     354           0 :                                                 if (santi)
     355           0 :                                                         anti = !anti;
     356           9 :                                         } else if (getFunctionId(p) == thetaselectRef
     357           9 :                                                            && isVarConstant(mb, getArg(p, 3))
     358           9 :                                                            && isVarConstant(mb, getArg(p, 4))) {
     359           9 :                                                 bit truth_value = *(bit *) getVarValue(mb, getArg(p, 3));
     360           9 :                                                 str comparison = (str) getVarValue(mb, getArg(p, 4));
     361             : 
     362             :                                                 /* there are no negative candidate lists so on = false situations swap anti flag */
     363           9 :                                                 if (truth_value == 0)
     364           2 :                                                         anti = !anti;
     365           7 :                                                 else if (is_bit_nil(truth_value))
     366           2 :                                                         selectok = FALSE;
     367           9 :                                                 if (strcmp(comparison, "<>") == 0)
     368           0 :                                                         anti = !anti;
     369           9 :                                                 else if (strcmp(comparison, "==") != 0)
     370             :                                                         selectok = FALSE;
     371             :                                         } else {
     372             :                                                 selectok = FALSE;
     373             :                                         }
     374             : 
     375           7 :                                         if (selectok) {
     376           7 :                                                 InstrPtr r = newInstruction(mb, algebraRef, likeselectRef);
     377           7 :                                                 if (r == NULL) {
     378           0 :                                                         msg = createException(MAL, "optimizer.pushselect",
     379             :                                                                                                   SQLSTATE(HY013)
     380             :                                                                                                   MAL_MALLOC_FAIL);
     381           0 :                                                         break;
     382             :                                                 }
     383           7 :                                                 getArg(r, 0) = getArg(p, 0);
     384           7 :                                                 r = pushArgument(mb, r, getArg(q, 1));
     385           7 :                                                 if (has_cand) {
     386           6 :                                                         r = pushArgument(mb, r, getArg(p, 2));
     387           6 :                                                         offset = 1;
     388           1 :                                                 } else if (isaBatType(getArgType(mb, q, 1))) {  /* likeselect calls have a candidate parameter */
     389           1 :                                                         r = pushNilBat(mb, r);
     390           1 :                                                         offset = 1;
     391             :                                                 }
     392          28 :                                                 for (int a = 2; a < q->argc; a++)
     393          21 :                                                         r = pushArgument(mb, r, getArg(q, a));
     394           7 :                                                 if (r->argc < (4 + offset))
     395           0 :                                                         r = pushStr(mb, r,
     396           0 :                                                                                 (str) getVarValue(mb, getArg(q, 3)));
     397           7 :                                                 if (r->argc < (5 + offset))
     398           0 :                                                         r = pushBit(mb, r, ignore_case);
     399           7 :                                                 if (r->argc < (6 + offset))
     400           7 :                                                         r = pushBit(mb, r, anti);
     401           7 :                                                 freeInstruction(p);
     402           7 :                                                 p = r;
     403           7 :                                                 actions++;
     404             :                                         }
     405             :                                 }
     406             :                         }
     407             : 
     408             :                         /* inject table ids into subselect
     409             :                          * s = subselect(c, C1..) => subselect(c, t, C1..)
     410             :                          */
     411      292842 :                         pushInstruction(mb, p);
     412             :                 }
     413        5698 :                 for (; i < limit; i++)
     414           0 :                         if (old[i])
     415           0 :                                 pushInstruction(mb, old[i]);
     416     1167174 :                 for (; i < slimit; i++)
     417     1161474 :                         if (old[i])
     418           0 :                                 freeInstruction(old[i]);
     419        5700 :                 GDKfree(old);
     420        5701 :                 if (msg != MAL_SUCCEED || !push_down_delta) {
     421        5681 :                         GDKfree(vars);
     422        5681 :                         goto wrapup;
     423             :                 }
     424             :         }
     425             : 
     426             :         /* now push selects through delta's */
     427      428080 :         limit = mb->stop;
     428      428080 :         slimit = mb->ssize;
     429      428080 :         old = mb->stmt;
     430             : 
     431      428080 :         nvars = (int *) GDKzalloc(sizeof(int) * mb->vtop);
     432      428089 :         slices = (int *) GDKzalloc(sizeof(int) * mb->vtop);
     433      428092 :         rslices = (bool *) GDKzalloc(sizeof(bool) * mb->vtop);
     434      428090 :         oclean = (bool *) GDKzalloc(sizeof(bool) * mb->vtop);
     435      856184 :         if (!nvars || !slices || !rslices || !oclean ||
     436      428092 :                 newMalBlkStmt(mb, mb->stop + (5 * push_down_delta) + (2 * nr_topn)) < 0) {
     437           0 :                 mb->stmt = old;
     438           0 :                 GDKfree(vars);
     439           0 :                 GDKfree(nvars);
     440           0 :                 GDKfree(slices);
     441           0 :                 GDKfree(rslices);
     442           0 :                 GDKfree(oclean);
     443           0 :                 goto wrapup;
     444             :         }
     445      428092 :         pushInstruction(mb, old[0]);
     446             : 
     447    16442600 :         for (i = 1; mb->errors == NULL && i < limit; i++) {
     448    15586427 :                 int lastbat;
     449    15586427 :                 p = old[i];
     450             : 
     451    30951960 :                 for (j = 0; j < p->retc; j++) {
     452    15365533 :                         int res = getArg(p, j);
     453    15365533 :                         vars[res] = i;
     454             :                 }
     455             : 
     456             :                 /* push subslice under projectdelta */
     457    15586427 :                 if (isSlice(p) && p->retc == 1) {
     458       16571 :                         int var = getArg(p, 1);
     459       16571 :                         InstrPtr q = old[vars[var]];
     460       16571 :                         if (q && getModuleId(q) == sqlRef
     461          11 :                                 && getFunctionId(q) == projectdeltaRef) {
     462           0 :                                 InstrPtr r = copyInstruction(p);
     463           0 :                                 InstrPtr s = copyInstruction(q);
     464           0 :                                 if (r == NULL || s == NULL) {
     465           0 :                                         freeInstruction(r);
     466           0 :                                         freeInstruction(s);
     467           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     468             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     469           0 :                                         break;
     470             :                                 }
     471             : 
     472           0 :                                 rslices[getArg(q, 0)] = true;   /* mark projectdelta as rewritten */
     473           0 :                                 rslices[getArg(p, 0)] = true;   /* mark slice as rewritten */
     474             : 
     475             :                                 /* slice the candidates */
     476           0 :                                 setFunctionId(r, sliceRef);
     477           0 :                                 nvars[getArg(p, 0)] = getArg(r, 0) = newTmpVariable(mb, getArgType(mb, r, 0));
     478           0 :                                 slices[getArg(q, 1)] = getArg(p, 0);
     479             : 
     480           0 :                                 setVarCList(mb, getArg(r, 0));
     481           0 :                                 getArg(r, 1) = getArg(s, 1);
     482           0 :                                 pushInstruction(mb, r);
     483             : 
     484           0 :                                 nvars[getArg(q, 0)] = getArg(s, 0) = newTmpVariable(mb, getArgType(mb, s, 0));
     485           0 :                                 getArg(s, 1) = getArg(r, 0);    /* use result of slice */
     486           0 :                                 pushInstruction(mb, s);
     487           0 :                                 oclean[i] = true;
     488           0 :                                 actions++;
     489           0 :                                 continue;
     490             :                         }
     491             :                 }
     492             :                 /* Leftfetchjoins involving rewritten sliced candidates ids need to be flattened
     493             :                  * l = projection(t, c); => l = c;
     494             :                  * and
     495             :                  * l = projection(s, ntids); => l = s;
     496             :                  */
     497    15570579 :                 else if (getModuleId(p) == algebraRef
     498     1171879 :                                  && getFunctionId(p) == projectionRef) {
     499     1021851 :                         int var = getArg(p, 1);
     500     1021851 :                         InstrPtr r = old[vars[var]], q;
     501             : 
     502     1021851 :                         if (r && isSlice(r) && rslices[var] && getArg(r, 0) == getArg(p, 1)) {
     503           0 :                                 int col = getArg(p, 2);
     504             : 
     505           0 :                                 if (!rslices[col]) {    /* was the deltaproject rewritten (sliced) */
     506           0 :                                         InstrPtr s = old[vars[col]], u = NULL;
     507             : 
     508           0 :                                         if (s && getModuleId(s) == algebraRef
     509           0 :                                                 && getFunctionId(s) == projectRef) {
     510           0 :                                                 col = getArg(s, 1);
     511           0 :                                                 u = s;
     512           0 :                                                 s = old[vars[col]];
     513             :                                         }
     514           0 :                                         if (s && getModuleId(s) == sqlRef
     515           0 :                                                 && getFunctionId(s) == projectdeltaRef) {
     516           0 :                                                 InstrPtr t = copyInstruction(s);
     517           0 :                                                 if (t == NULL) {
     518           0 :                                                         msg = createException(MAL, "optimizer.pushselect",
     519             :                                                                                                   SQLSTATE(HY013)
     520             :                                                                                                   MAL_MALLOC_FAIL);
     521           0 :                                                         break;
     522             :                                                 }
     523             : 
     524           0 :                                                 getArg(t, 1) = nvars[getArg(r, 0)];     /* use result of slice */
     525           0 :                                                 rslices[col] = true;
     526           0 :                                                 nvars[getArg(s, 0)] = getArg(t, 0) = newTmpVariable(mb, getArgType(mb, t, 0));
     527           0 :                                                 pushInstruction(mb, t);
     528           0 :                                                 if (u) {        /* add again */
     529           0 :                                                         if ((t = copyInstruction(u)) == NULL) {
     530           0 :                                                                 msg = createException(MAL,
     531             :                                                                                                           "optimizer.pushselect",
     532             :                                                                                                           SQLSTATE(HY013)
     533             :                                                                                                           MAL_MALLOC_FAIL);
     534           0 :                                                                 break;
     535             :                                                         }
     536           0 :                                                         getArg(t, 1) = nvars[getArg(t, 1)];
     537           0 :                                                         pushInstruction(mb, t);
     538             :                                                 }
     539             :                                         }
     540             :                                 }
     541           0 :                                 q = newAssignment(mb);
     542           0 :                                 if (q == NULL) {
     543           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     544             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     545           0 :                                         break;
     546             :                                 }
     547           0 :                                 getArg(q, 0) = getArg(p, 0);
     548           0 :                                 q = pushArgument(mb, q, getArg(p, 2));
     549           0 :                                 if (nvars[getArg(p, 2)] > 0)
     550           0 :                                         getArg(q, 1) = nvars[getArg(p, 2)];
     551           0 :                                 oclean[i] = true;
     552           0 :                                 actions++;
     553           0 :                                 pushInstruction(mb, q);
     554           0 :                                 continue;
     555             :                         }
     556    14548728 :                 } else if (p->argc >= 2 && slices[getArg(p, 1)] != 0) {
     557             :                         /* use new slice candidate list */
     558           0 :                         assert(slices[getArg(p, 1)] == nvars[getArg(p, 1)]);
     559           0 :                         getArg(p, 1) = slices[getArg(p, 1)];
     560             :                 }
     561             :                 /* remap */
     562    32918387 :                 for (j = p->retc; j < p->argc; j++) {
     563    17331188 :                         int var = getArg(p, j);
     564    17331188 :                         if (nvars[var] > 0) {
     565           0 :                                 getArg(p, j) = nvars[var];
     566             :                         }
     567             :                 }
     568             : 
     569             :                 /* c = delta(b, uid, uvl)
     570             :                  * s = select(c, C1..)
     571             :                  *
     572             :                  * nc = select(b, C1..)
     573             :                  * nu = select(uvl, C1..)
     574             :                  * s = subdelta(nc, uid, nu);
     575             :                  *
     576             :                  * doesn't handle Xselect(x, .. z, C1.. cases) ie multicolumn selects
     577             :                  *
     578             :                  * also handle (if no_mito)
     579             :                  * c = pack(b, ins)
     580             :                  * s = select(c, C1..)
     581             :                  */
     582    15587199 :                 lastbat = lastbat_arg(mb, p);
     583    15587199 :                 if (isSelect(p) && p->retc == 1 && lastbat == 2) {
     584       77297 :                         int var = getArg(p, 1);
     585       77297 :                         InstrPtr q = old[vars[var]];
     586             : 
     587       77297 :                         if (q && q->token == ASSIGNsymbol) {
     588       31912 :                                 var = getArg(q, 1);
     589       31912 :                                 q = old[vars[var]];
     590             :                         }
     591       77297 :                         if (no_mito && q && getModuleId(q) == matRef
     592           0 :                                 && getFunctionId(q) == packRef && q->argc == (q->retc + 2)) {
     593           0 :                                 InstrPtr r = copyInstruction(p);
     594           0 :                                 InstrPtr t = copyInstruction(p);
     595             : 
     596           0 :                                 if (r == NULL || t == NULL) {
     597           0 :                                         freeInstruction(r);
     598           0 :                                         freeInstruction(t);
     599           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     600             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     601           0 :                                         break;
     602             :                                 }
     603           0 :                                 getArg(r, 0) = newTmpVariable(mb, newBatType(TYPE_oid));
     604           0 :                                 setVarCList(mb, getArg(r, 0));
     605           0 :                                 getArg(r, 1) = getArg(q, 1);    /* column */
     606           0 :                                 r->typeresolved = false;
     607           0 :                                 pushInstruction(mb, r);
     608           0 :                                 getArg(t, 0) = newTmpVariable(mb, newBatType(TYPE_oid));
     609           0 :                                 setVarCList(mb, getArg(t, 0));
     610           0 :                                 getArg(t, 1) = getArg(q, 2);    /* inserts */
     611           0 :                                 pushInstruction(mb, t);
     612             : 
     613           0 :                                 InstrPtr u = copyInstruction(q);        /* pack result */
     614           0 :                                 if (u == NULL) {
     615           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     616             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     617           0 :                                         break;
     618             :                                 }
     619           0 :                                 getArg(u, 0) = getArg(p, 0);
     620           0 :                                 getArg(u, 1) = getArg(r, 0);
     621           0 :                                 getArg(u, 2) = getArg(t, 0);
     622           0 :                                 u->typeresolved = false;
     623           0 :                                 pushInstruction(mb, u);
     624           0 :                                 oclean[i] = true;
     625           0 :                                 continue;
     626       77297 :                         } else if (q && getModuleId(q) == sqlRef
     627       64354 :                                            && getFunctionId(q) == deltaRef) {
     628       25968 :                                 InstrPtr r = copyInstruction(p);
     629       25968 :                                 InstrPtr s = copyInstruction(p);
     630       25968 :                                 InstrPtr u = copyInstruction(q);
     631             : 
     632       25968 :                                 if (r == NULL || s == NULL || u == NULL) {
     633           0 :                                         freeInstruction(r);
     634           0 :                                         freeInstruction(s);
     635           0 :                                         freeInstruction(u);
     636           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     637             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     638           0 :                                         break;
     639             :                                 }
     640       25968 :                                 getArg(r, 0) = newTmpVariable(mb, newBatType(TYPE_oid));
     641       25968 :                                 setVarCList(mb, getArg(r, 0));
     642       25968 :                                 getArg(r, 1) = getArg(q, 1);    /* column */
     643       25968 :                                 r->typeresolved = false;
     644       25968 :                                 pushInstruction(mb, r);
     645       25968 :                                 getArg(s, 0) = newTmpVariable(mb, newBatType(TYPE_oid));
     646       25968 :                                 setVarCList(mb, getArg(s, 0));
     647       25968 :                                 getArg(s, 1) = getArg(q, 3);    /* updates */
     648       25968 :                                 s = ReplaceWithNil(mb, s, 2);   /* no candidate list */
     649       25968 :                                 setArgType(mb, s, 2, newBatType(TYPE_oid));
     650             :                                 /* make sure to resolve again */
     651       25968 :                                 s->token = ASSIGNsymbol;
     652       25968 :                                 s->typeresolved = false;
     653       25968 :                                 s->fcn = NULL;
     654       25968 :                                 s->blk = NULL;
     655       25968 :                                 pushInstruction(mb, s);
     656             : 
     657       25968 :                                 setFunctionId(u, subdeltaRef);
     658       25968 :                                 getArg(u, 0) = getArg(p, 0);
     659       25968 :                                 getArg(u, 1) = getArg(r, 0);
     660       25968 :                                 getArg(u, 2) = getArg(p, 2);    /* pre-cands */
     661       25968 :                                 getArg(u, 3) = getArg(q, 2);    /* update ids */
     662       25968 :                                 u = pushArgument(mb, u, getArg(s, 0));  /* selected updated values ids */
     663       25968 :                                 u->token = ASSIGNsymbol;
     664       25968 :                                 u->typeresolved = false;
     665       25968 :                                 u->fcn = NULL;
     666       25968 :                                 u->blk = NULL;
     667       25968 :                                 pushInstruction(mb, u);
     668       25968 :                                 oclean[i] = true;
     669       25968 :                                 continue;
     670             :                         }
     671    15509090 :                 } else if (getModuleId(p) == algebraRef
     672     1111131 :                                    && getFunctionId(p) == projectionRef) {
     673     1021852 :                         int id = getArg(p, 1);
     674     1021852 :                         InstrPtr s = old[vars[id]];
     675     1021852 :                         int var = getArg(p, 2);
     676     1021852 :                         InstrPtr q = old[vars[var]];
     677             : 
     678     1021852 :                         if (no_mito && getModuleId(q) == matRef
     679           2 :                                 && getFunctionId(q) == packRef && q->argc == 3
     680           0 :                                 && getModuleId(s) == matRef && getFunctionId(s) == packRef
     681           0 :                                 && s->argc == 3) {
     682           0 :                                 InstrPtr r = copyInstruction(p);
     683           0 :                                 InstrPtr t = copyInstruction(p);
     684             : 
     685           0 :                                 if (r == NULL || t == NULL) {
     686           0 :                                         freeInstruction(r);
     687           0 :                                         freeInstruction(t);
     688           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     689             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     690           0 :                                         break;
     691             :                                 }
     692           0 :                                 getArg(r, 0) = newTmpVariable(mb, getArgType(mb, p, 0));
     693           0 :                                 setVarCList(mb, getArg(r, 0));
     694           0 :                                 getArg(r, 1) = getArg(s, 1);
     695           0 :                                 getArg(r, 2) = getArg(q, 1);    /* column */
     696           0 :                                 r->typeresolved = false;
     697           0 :                                 pushInstruction(mb, r);
     698           0 :                                 getArg(t, 0) = newTmpVariable(mb, getArgType(mb, p, 0));
     699           0 :                                 setVarCList(mb, getArg(t, 0));
     700           0 :                                 getArg(t, 1) = getArg(s, 2);
     701           0 :                                 getArg(t, 2) = getArg(q, 2);    /* inserts */
     702           0 :                                 pushInstruction(mb, t);
     703             : 
     704           0 :                                 InstrPtr u = copyInstruction(q);        /* pack result */
     705           0 :                                 if (u == NULL) {
     706           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     707             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     708           0 :                                         break;
     709             :                                 }
     710           0 :                                 getArg(u, 0) = getArg(p, 0);
     711           0 :                                 getArg(u, 1) = getArg(r, 0);
     712           0 :                                 getArg(u, 2) = getArg(t, 0);
     713           0 :                                 u->typeresolved = false;
     714           0 :                                 pushInstruction(mb, u);
     715           0 :                                 oclean[i] = true;
     716           0 :                                 continue;
     717     1021852 :                         } else if (getModuleId(q) == sqlRef && getFunctionId(q) == deltaRef
     718       66075 :                                            && q->argc == 4) {
     719       66075 :                                 q = copyInstruction(q);
     720       66075 :                                 if (q == NULL) {
     721           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     722             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     723           0 :                                         break;
     724             :                                 }
     725       66075 :                                 setFunctionId(q, projectdeltaRef);
     726       66075 :                                 getArg(q, 0) = getArg(p, 0);
     727       66075 :                                 q = PushArgument(mb, q, getArg(p, 1), 1);
     728       66075 :                                 p = q;
     729       66075 :                                 oclean[i] = true;
     730       66075 :                                 actions++;
     731             :                         }
     732    14487238 :                 } else if (isIntersect(p) && p->retc == 1 && lastbat == 4) {
     733             :                         /* l = delta(b, uid, uvl)
     734             :                          * s = intersect(l, r, li, ..)
     735             :                          *
     736             :                          * nc = intersect(b, r, li..)
     737             :                          * nu = intersect(uvl, r, ..)
     738             :                          * s = subdelta(nc, li, uid, nu);
     739             :                          */
     740           8 :                         int var = getArg(p, 1);
     741           8 :                         InstrPtr q = old[vars[var]];
     742             : 
     743           8 :                         if (q && q->token == ASSIGNsymbol) {
     744           0 :                                 var = getArg(q, 1);
     745           0 :                                 q = old[vars[var]];
     746             :                         }
     747           8 :                         if (q && getModuleId(q) == sqlRef && getFunctionId(q) == deltaRef) {
     748           0 :                                 InstrPtr r = copyInstruction(p);
     749           0 :                                 InstrPtr s = copyInstruction(p);
     750           0 :                                 InstrPtr u = copyInstruction(q);
     751             : 
     752           0 :                                 if (r == NULL || s == NULL || u == NULL) {
     753           0 :                                         freeInstruction(r);
     754           0 :                                         freeInstruction(s);
     755           0 :                                         freeInstruction(u);
     756           0 :                                         msg = createException(MAL, "optimizer.pushselect",
     757             :                                                                                   SQLSTATE(HY013) MAL_MALLOC_FAIL);
     758           0 :                                         break;
     759             :                                 }
     760           0 :                                 getArg(r, 0) = newTmpVariable(mb, newBatType(TYPE_oid));
     761           0 :                                 setVarCList(mb, getArg(r, 0));
     762           0 :                                 getArg(r, 1) = getArg(q, 1);    /* column */
     763           0 :                                 r->typeresolved = false;
     764           0 :                                 pushInstruction(mb, r);
     765           0 :                                 getArg(s, 0) = newTmpVariable(mb, newBatType(TYPE_oid));
     766           0 :                                 setVarCList(mb, getArg(s, 0));
     767           0 :                                 getArg(s, 1) = getArg(q, 3);    /* updates */
     768           0 :                                 s = ReplaceWithNil(mb, s, 3);   /* no candidate list */
     769           0 :                                 setArgType(mb, s, 3, newBatType(TYPE_oid));
     770             :                                 /* make sure to resolve again */
     771           0 :                                 s->token = ASSIGNsymbol;
     772           0 :                                 s->typeresolved = false;
     773           0 :                                 s->fcn = NULL;
     774           0 :                                 s->blk = NULL;
     775           0 :                                 pushInstruction(mb, s);
     776             : 
     777           0 :                                 setFunctionId(u, subdeltaRef);
     778           0 :                                 getArg(u, 0) = getArg(p, 0);
     779           0 :                                 getArg(u, 1) = getArg(r, 0);
     780           0 :                                 getArg(u, 2) = getArg(p, 3);    /* pre-cands */
     781           0 :                                 getArg(u, 3) = getArg(q, 2);    /* update ids */
     782             :                                 //getArg(u, 4) = getArg(s,0);
     783           0 :                                 p = pushArgument(mb, u, getArg(s, 0));  /* push at end */
     784             :                                 /* make sure to resolve again */
     785           0 :                                 u->token = ASSIGNsymbol;
     786           0 :                                 u->typeresolved = false;
     787           0 :                                 u->fcn = NULL;
     788           0 :                                 u->blk = NULL;
     789           0 :                                 pushInstruction(mb, u);
     790           0 :                                 oclean[i] = true;
     791           0 :                                 continue;
     792             :                         }
     793             :                 }
     794    15560419 :                 assert(p == old[i] || oclean[i]);
     795    15560419 :                 pushInstruction(mb, p);
     796             :         }
     797    16016049 :         for (j = 1; j < i; j++)
     798    15587965 :                 if (old[j] && oclean[j])
     799       92043 :                         freeInstruction(old[j]);
     800    96436552 :         for (; i < slimit; i++)
     801    96008461 :                 if (old[i])
     802           0 :                         pushInstruction(mb, old[i]);
     803      428091 :         GDKfree(vars);
     804      428090 :         GDKfree(nvars);
     805      428092 :         GDKfree(slices);
     806      428092 :         GDKfree(rslices);
     807      428092 :         GDKfree(oclean);
     808      428092 :         GDKfree(old);
     809             : 
     810             :         /* Defense line against incorrect plans */
     811      428092 :         if (msg == MAL_SUCCEED && actions > 0) {
     812       18587 :                 msg = chkTypes(cntxt->usermodule, mb, FALSE);
     813       18587 :                 if (msg == MAL_SUCCEED)
     814       18587 :                         msg = chkFlow(mb);
     815       18587 :                 if (msg == MAL_SUCCEED)
     816       18587 :                         msg = chkDeclarations(mb);
     817             :         }
     818      409505 :   wrapup:
     819             :         /* keep actions taken as a fake argument */
     820      445931 :         (void) pushInt(mb, pci, actions);
     821      445931 :         return msg;
     822             : }

Generated by: LCOV version 1.14