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 "rel_optimizer_private.h"
15 : #include "rel_planner.h"
16 : #include "rel_exp.h"
17 : #include "rel_select.h"
18 : #include "rel_rewriter.h"
19 :
20 : /* Split_select optimizer splits case statements in select expressions. This is a step needed for cse */
21 : static void select_split_exps(mvc *sql, list *exps, sql_rel *rel);
22 :
23 : static sql_exp *
24 220202 : select_split_exp(mvc *sql, sql_exp *e, sql_rel *rel)
25 : {
26 220202 : switch(e->type) {
27 : case e_column:
28 : return e;
29 6905 : case e_convert:
30 6905 : e->l = select_split_exp(sql, e->l, rel);
31 6905 : return e;
32 24433 : case e_aggr:
33 : case e_func:
34 24433 : if (!is_analytic(e) && !exp_has_sideeffect(e)) {
35 24428 : sql_subfunc *f = e->f;
36 24428 : if (e->type == e_func && !f->func->s && is_caselike_func(f) /*is_ifthenelse_func(f)*/)
37 280 : return add_exp_too_project(sql, e, rel);
38 : }
39 : return e;
40 75348 : case e_cmp:
41 75348 : if (e->flag == cmp_or || e->flag == cmp_filter) {
42 18850 : select_split_exps(sql, e->l, rel);
43 18850 : select_split_exps(sql, e->r, rel);
44 56498 : } else if (e->flag == cmp_in || e->flag == cmp_notin) {
45 3254 : e->l = select_split_exp(sql, e->l, rel);
46 3254 : select_split_exps(sql, e->r, rel);
47 : } else {
48 53244 : e->l = select_split_exp(sql, e->l, rel);
49 53244 : e->r = select_split_exp(sql, e->r, rel);
50 53244 : if (e->f)
51 3590 : e->f = select_split_exp(sql, e->f, rel);
52 : }
53 : return e;
54 : case e_atom:
55 : case e_psm:
56 : return e;
57 : }
58 : return e;
59 : }
60 :
61 : static void
62 65793 : select_split_exps(mvc *sql, list *exps, sql_rel *rel)
63 : {
64 65793 : node *n;
65 :
66 65793 : if (!exps)
67 : return;
68 165758 : for(n=exps->h; n; n = n->next){
69 99965 : sql_exp *e = n->data;
70 :
71 99965 : e = select_split_exp(sql, e, rel);
72 99965 : n->data = e;
73 : }
74 : }
75 :
76 : static sql_rel *
77 1661700 : rel_split_select_(visitor *v, sql_rel *rel)
78 : {
79 1661700 : if (!rel || !is_select(rel->op) || list_empty(rel->exps) || !rel->l || mvc_highwater(v->sql))
80 1440247 : return rel;
81 :
82 221453 : bool funcs = false;
83 :
84 : /* are there functions */
85 473467 : for (node *n = rel->exps->h; n && !funcs; n = n->next)
86 252014 : funcs = exp_has_func(n->data);
87 :
88 : /* introduce extra project */
89 221453 : if (funcs) {
90 24839 : sql_rel *nrel = rel_project(v->sql->sa, rel->l,
91 24839 : rel_projections(v->sql, rel->l, NULL, 1, 1));
92 24839 : if (!nrel || !nrel->exps)
93 : return NULL;
94 24839 : rel->l = nrel;
95 : /* recursively split all functions and add those to the projection list */
96 24839 : select_split_exps(v->sql, rel->exps, nrel);
97 24839 : return rel;
98 : }
99 : return rel;
100 : }
101 :
102 : static sql_rel *
103 55815 : rel_split_select(visitor *v, global_props *gp, sql_rel *rel)
104 : {
105 55815 : (void) gp;
106 55815 : return rel_visitor_bottomup(v, rel, &rel_split_select_);
107 : }
108 :
109 : run_optimizer
110 623152 : bind_split_select(visitor *v, global_props *gp)
111 : {
112 623152 : int flag = v->sql->sql_optimizer;
113 547421 : return gp->opt_cycle == 0 && gp->opt_level == 1 && (flag & split_select)
114 1170565 : && gp->cnt[op_select] ? rel_split_select : NULL;
115 : }
116 :
117 :
118 : /*
119 : * Remove a redundant join
120 : *
121 : * join (L, Distinct Project(join(L,P) [ p.key == l.lkey]) [p.key]) [ p.key == l.lkey]
122 : * =>
123 : * join(L, P) [p.key==l.lkey]
124 : */
125 : static sql_rel *
126 1468863 : rel_remove_redundant_join_(visitor *v, sql_rel *rel)
127 : {
128 1468863 : if ((is_join(rel->op) || is_semi(rel->op)) && !list_empty(rel->exps)) {
129 218483 : sql_rel *l = rel->l, *r = rel->r, *b, *p = NULL, *j;
130 :
131 218483 : if (is_basetable(l->op) && is_simple_project(r->op) && need_distinct(r)) {
132 9 : b = l;
133 9 : p = r;
134 9 : j = p->l;
135 218474 : } else if (is_basetable(r->op) && is_simple_project(l->op) && need_distinct(l)) {
136 2333 : b = r;
137 2333 : p = l;
138 2333 : j = p->l;
139 : }
140 218483 : if (!p || !j || j->op != rel->op)
141 : return rel;
142 : /* j must have b->l (ie table) */
143 12 : sql_rel *jl = j->l, *jr = j->r;
144 12 : if ((is_basetable(jl->op) && jl->l == b->l) ||
145 8 : (is_basetable(jr->op) && jr->l == b->l)) {
146 11 : int left = 0;
147 11 : if (is_basetable(jl->op) && jl->l == b->l)
148 11 : left = 1;
149 11 : if (!list_empty(p->exps)) {
150 19 : for (node *n=p->exps->h; n; n = n->next) { /* all exps of 'p' must be bound to the opposite side */
151 11 : sql_exp *e = n->data;
152 :
153 18 : if (!rel_rebind_exp(v->sql, left ? jr : jl, e))
154 : return rel;
155 : }
156 : }
157 8 : if (exp_match_list(j->exps, rel->exps)) {
158 0 : p->l = (left)?rel_dup(jr):rel_dup(jl);
159 0 : rel_destroy(j);
160 0 : set_nodistinct(p);
161 0 : v->changes++;
162 0 : return rel;
163 : }
164 : }
165 : }
166 : return rel;
167 : }
168 :
169 : static sql_rel *
170 48246 : rel_remove_redundant_join(visitor *v, global_props *gp, sql_rel *rel)
171 : {
172 48246 : (void) gp;
173 48246 : return rel_visitor_bottomup(v, rel, &rel_remove_redundant_join_); /* this optimizer has to run before rel_first_level_optimizations */
174 : }
175 :
176 : run_optimizer
177 623220 : bind_remove_redundant_join(visitor *v, global_props *gp)
178 : {
179 623220 : int flag = v->sql->sql_optimizer;
180 547512 : return gp->opt_cycle == 0 && gp->opt_level == 1 && (gp->cnt[op_left] || gp->cnt[op_right]
181 529175 : || gp->cnt[op_full] || gp->cnt[op_join] || gp->cnt[op_semi] || gp->cnt[op_anti]) &&
182 671243 : (flag & remove_redundant_join) ? rel_remove_redundant_join : NULL;
183 : }
184 :
185 :
186 : static list *
187 1125309 : exp_merge_range(visitor *v, sql_rel *rel, list *exps)
188 : {
189 1125309 : node *n, *m;
190 2418319 : for (n=exps->h; n; n = n->next) {
191 1294903 : sql_exp *e = n->data;
192 1294903 : sql_exp *le = e->l;
193 1294903 : sql_exp *re = e->r;
194 :
195 : /* handle the and's in the or lists */
196 1294903 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e)) {
197 45309 : e->l = exp_merge_range(v, rel, e->l);
198 45309 : e->r = exp_merge_range(v, rel, e->r);
199 : /* only look for gt, gte, lte, lt */
200 1249594 : } else if (n->next &&
201 165391 : e->type == e_cmp && e->flag < cmp_equal && !e->f &&
202 6818 : re->card == CARD_ATOM && !is_anti(e)) {
203 1268 : for (m=n->next; m; m = m->next) {
204 777 : sql_exp *f = m->data;
205 777 : sql_exp *lf = f->l;
206 777 : sql_exp *rf = f->r;
207 777 : int c_le = is_numeric_upcast(le), c_lf = is_numeric_upcast(lf);
208 :
209 777 : if (f->type == e_cmp && f->flag < cmp_equal && !f->f &&
210 645 : rf->card == CARD_ATOM && !is_anti(f) &&
211 322 : exp_match_exp(c_le?le->l:le, c_lf?lf->l:lf)) {
212 172 : sql_exp *ne;
213 172 : int swap = 0, lt = 0, gt = 0;
214 172 : sql_subtype super;
215 : /* for now only c1 <[=] x <[=] c2 */
216 :
217 172 : swap = lt = (e->flag == cmp_lt || e->flag == cmp_lte);
218 172 : gt = !lt;
219 :
220 172 : if (gt &&
221 150 : (f->flag == cmp_gt ||
222 : f->flag == cmp_gte))
223 32 : continue;
224 22 : if (lt &&
225 22 : (f->flag == cmp_lt ||
226 : f->flag == cmp_lte))
227 0 : continue;
228 :
229 140 : cmp_supertype(&super, exp_subtype(le), exp_subtype(lf));
230 140 : if (!(rf = exp_check_type(v->sql, &super, rel, rf, type_equal)) ||
231 140 : !(le = exp_check_type(v->sql, &super, rel, le, type_equal)) ||
232 140 : !(re = exp_check_type(v->sql, &super, rel, re, type_equal))) {
233 0 : v->sql->session->status = 0;
234 0 : v->sql->errstr[0] = 0;
235 0 : continue;
236 : }
237 140 : if (!swap)
238 118 : ne = exp_compare2(v->sql->sa, le, re, rf, compare2range(e->flag, f->flag), 0);
239 : else
240 22 : ne = exp_compare2(v->sql->sa, le, rf, re, compare2range(f->flag, e->flag), 0);
241 :
242 140 : list_remove_data(exps, NULL, e);
243 140 : list_remove_data(exps, NULL, f);
244 140 : list_append(exps, ne);
245 140 : v->changes++;
246 140 : return exp_merge_range(v, rel, exps);
247 : }
248 : }
249 1248963 : } else if (n->next &&
250 164760 : e->type == e_cmp && e->flag < cmp_equal && !e->f &&
251 6187 : re->card > CARD_ATOM && !is_anti(e)) {
252 11218 : for (m=n->next; m; m = m->next) {
253 6784 : sql_exp *f = m->data;
254 6784 : sql_exp *lf = f->l;
255 6784 : sql_exp *rf = f->r;
256 :
257 6784 : if (f->type == e_cmp && f->flag < cmp_equal && !f->f &&
258 4979 : rf->card > CARD_ATOM && !is_anti(f)) {
259 4978 : sql_exp *ne, *t;
260 4978 : int swap = 0, lt = 0, gt = 0;
261 4978 : comp_type ef = (comp_type) e->flag, ff = (comp_type) f->flag;
262 4978 : int c_re = is_numeric_upcast(re), c_rf = is_numeric_upcast(rf);
263 4978 : int c_le = is_numeric_upcast(le), c_lf = is_numeric_upcast(lf), c;
264 4978 : sql_subtype super;
265 :
266 : /* both swapped ? */
267 4978 : if (exp_match_exp(c_re?re->l:re, c_rf?rf->l:rf)) {
268 19 : t = re;
269 19 : re = le;
270 19 : le = t;
271 19 : c = c_re; c_re = c_le; c_le = c;
272 19 : ef = swap_compare(ef);
273 19 : t = rf;
274 19 : rf = lf;
275 19 : lf = t;
276 19 : c = c_rf; c_rf = c_lf; c_lf = c;
277 19 : ff = swap_compare(ff);
278 : }
279 :
280 : /* is left swapped ? */
281 4978 : if (exp_match_exp(c_re?re->l:re, c_lf?lf->l:lf)) {
282 104 : t = re;
283 104 : re = le;
284 104 : le = t;
285 104 : c = c_re; c_re = c_le; c_le = c;
286 104 : ef = swap_compare(ef);
287 : }
288 :
289 : /* is right swapped ? */
290 4978 : if (exp_match_exp(c_le?le->l:le, c_rf?rf->l:rf)) {
291 177 : t = rf;
292 177 : rf = lf;
293 177 : lf = t;
294 177 : c = c_rf; c_rf = c_lf; c_lf = c;
295 177 : ff = swap_compare(ff);
296 : }
297 :
298 4978 : if (!exp_match_exp(c_le?le->l:le, c_lf?lf->l:lf))
299 3225 : continue;
300 :
301 : /* for now only c1 <[=] x <[=] c2 */
302 1830 : swap = lt = (ef == cmp_lt || ef == cmp_lte);
303 1830 : gt = !lt;
304 :
305 1830 : if (gt && (ff == cmp_gt || ff == cmp_gte))
306 73 : continue;
307 1757 : if (lt && (ff == cmp_lt || ff == cmp_lte))
308 4 : continue;
309 :
310 1753 : cmp_supertype(&super, exp_subtype(le), exp_subtype(lf));
311 1753 : if (!(rf = exp_check_type(v->sql, &super, rel, rf, type_equal)) ||
312 1753 : !(le = exp_check_type(v->sql, &super, rel, le, type_equal)) ||
313 1753 : !(re = exp_check_type(v->sql, &super, rel, re, type_equal))) {
314 0 : v->sql->session->status = 0;
315 0 : v->sql->errstr[0] = 0;
316 0 : continue;
317 : }
318 1753 : if (!swap)
319 1653 : ne = exp_compare2(v->sql->sa, le, re, rf, compare2range(ef, ff), 0);
320 : else
321 100 : ne = exp_compare2(v->sql->sa, le, rf, re, compare2range(ff, ef), 0);
322 :
323 1753 : list_remove_data(exps, NULL, e);
324 1753 : list_remove_data(exps, NULL, f);
325 1753 : list_append(exps, ne);
326 1753 : v->changes++;
327 1753 : return exp_merge_range(v, rel, exps);
328 : }
329 : }
330 : }
331 : }
332 : return exps;
333 : }
334 :
335 : static int
336 41819 : exps_cse( mvc *sql, list *oexps, list *l, list *r )
337 : {
338 41819 : list *nexps;
339 41819 : node *n, *m;
340 41819 : char *lu, *ru;
341 41819 : int lc = 0, rc = 0, match = 0, res = 0;
342 :
343 41819 : if (list_length(l) == 0 || list_length(r) == 0)
344 0 : return 0;
345 :
346 : /* first recursive exps_cse */
347 41819 : nexps = new_exp_list(sql->sa);
348 88493 : for (n = l->h; n; n = n->next) {
349 46674 : sql_exp *e = n->data;
350 :
351 46674 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e)) {
352 16178 : res = exps_cse(sql, nexps, e->l, e->r);
353 : } else {
354 30496 : append(nexps, e);
355 : }
356 : }
357 41819 : l = nexps;
358 :
359 41819 : nexps = new_exp_list(sql->sa);
360 91224 : for (n = r->h; n; n = n->next) {
361 49405 : sql_exp *e = n->data;
362 :
363 49405 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e)) {
364 3320 : res = exps_cse(sql, nexps, e->l, e->r);
365 : } else {
366 46085 : append(nexps, e);
367 : }
368 : }
369 41819 : r = nexps;
370 :
371 : /* simplify true or .. and .. or true */
372 41819 : if (list_length(l) == list_length(r) && list_length(l) == 1) {
373 36716 : sql_exp *le = l->h->data, *re = r->h->data;
374 :
375 36716 : if (exp_is_true(le)) {
376 14 : append(oexps, le);
377 14 : return 1;
378 : }
379 36702 : if (exp_is_true(re)) {
380 5 : append(oexps, re);
381 5 : return 1;
382 : }
383 : }
384 :
385 41800 : lu = SA_ZNEW_ARRAY(sql->ta, char, list_length(l));
386 41800 : ru = SA_ZNEW_ARRAY(sql->ta, char, list_length(r));
387 89181 : for (n = l->h, lc = 0; n; n = n->next, lc++) {
388 47381 : sql_exp *le = n->data;
389 :
390 106569 : for ( m = r->h, rc = 0; m; m = m->next, rc++) {
391 59188 : sql_exp *re = m->data;
392 :
393 59188 : if (!ru[rc] && exp_match_exp(le,re)) {
394 796 : lu[lc] = 1;
395 796 : ru[rc] = 1;
396 796 : match = 1;
397 : }
398 : }
399 : }
400 41800 : if (match) {
401 415 : list *nl = new_exp_list(sql->sa);
402 415 : list *nr = new_exp_list(sql->sa);
403 :
404 1625 : for (n = l->h, lc = 0; n; n = n->next, lc++)
405 1210 : if (!lu[lc])
406 418 : append(nl, n->data);
407 1639 : for (n = r->h, rc = 0; n; n = n->next, rc++)
408 1224 : if (!ru[rc])
409 428 : append(nr, n->data);
410 :
411 415 : if (list_length(nl) && list_length(nr))
412 387 : append(oexps, exp_or(sql->sa, nl, nr, 0));
413 :
414 1625 : for (n = l->h, lc = 0; n; n = n->next, lc++) {
415 1210 : if (lu[lc])
416 792 : append(oexps, n->data);
417 : }
418 : res = 1;
419 : } else {
420 41385 : append(oexps, exp_or(sql->sa, list_dup(l, (fdup)NULL),
421 : list_dup(r, (fdup)NULL), 0));
422 : }
423 : return res;
424 : }
425 :
426 : static int
427 93562 : are_equality_exps( list *exps, sql_exp **L)
428 : {
429 93562 : sql_exp *l = *L;
430 :
431 93562 : if (list_length(exps) == 1) {
432 89455 : sql_exp *e = exps->h->data, *le = e->l, *re = e->r;
433 :
434 89455 : if (e->type == e_cmp && e->flag == cmp_equal && le->card != CARD_ATOM && re->card == CARD_ATOM && !is_semantics(e)) {
435 22376 : if (!l) {
436 10841 : *L = l = le;
437 10841 : if (!is_column(le->type))
438 : return 0;
439 : }
440 22376 : return (exp_match(l, le));
441 : }
442 67079 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e) && !is_semantics(e))
443 72707 : return (are_equality_exps(e->l, L) && are_equality_exps(e->r, L));
444 : }
445 : return 0;
446 : }
447 :
448 : static void
449 4283 : get_exps( list *n, list *l )
450 : {
451 5996 : sql_exp *e = l->h->data, *re = e->r;
452 :
453 5996 : if (e->type == e_cmp && e->flag == cmp_equal && re->card == CARD_ATOM)
454 4283 : list_append(n, re);
455 5996 : if (e->type == e_cmp && e->flag == cmp_or) {
456 1713 : get_exps(n, e->l);
457 1713 : get_exps(n, e->r);
458 : }
459 4283 : }
460 :
461 : static sql_exp *
462 1285 : equality_exps_2_in( mvc *sql, sql_exp *ce, list *l, list *r)
463 : {
464 1285 : list *nl = new_exp_list(sql->sa);
465 :
466 1285 : get_exps(nl, l);
467 1285 : get_exps(nl, r);
468 :
469 1285 : return exp_in( sql->sa, ce, nl, cmp_in);
470 : }
471 :
472 : static list *
473 606616 : merge_ors(mvc *sql, list *exps, int *changes)
474 : {
475 606616 : list *nexps = NULL;
476 606616 : int needed = 0;
477 :
478 1295984 : for (node *n = exps->h; n && !needed; n = n->next) {
479 689368 : sql_exp *e = n->data;
480 :
481 689368 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e))
482 689368 : needed = 1;
483 : }
484 :
485 606616 : if (needed) {
486 42224 : nexps = new_exp_list(sql->sa);
487 92839 : for (node *n = exps->h; n; n = n->next) {
488 50615 : sql_exp *e = n->data, *l = NULL;
489 :
490 50615 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e) && are_equality_exps(e->l, &l) && are_equality_exps(e->r, &l) && l) {
491 1285 : (*changes)++;
492 1285 : append(nexps, equality_exps_2_in(sql, l, e->l, e->r));
493 : } else {
494 49330 : append(nexps, e);
495 : }
496 : }
497 : } else {
498 : nexps = exps;
499 : }
500 :
501 1303320 : for (node *n = nexps->h; n ; n = n->next) {
502 696704 : sql_exp *e = n->data;
503 :
504 696704 : if (e->type == e_cmp && e->flag == cmp_or) {
505 42051 : e->l = merge_ors(sql, e->l, changes);
506 42051 : e->r = merge_ors(sql, e->r, changes);
507 : }
508 : }
509 :
510 606616 : return nexps;
511 : }
512 :
513 : #define TRIVIAL_NOT_EQUAL_CMP(e) \
514 : ((e)->type == e_cmp && (e)->flag == cmp_notequal && !is_anti((e)) && !is_semantics((e)) && ((sql_exp*)(e)->l)->card != CARD_ATOM && ((sql_exp*)(e)->r)->card == CARD_ATOM)
515 :
516 : static list *
517 606616 : merge_notequal(mvc *sql, list *exps, int *changes)
518 : {
519 606616 : list *inequality_groups = NULL, *nexps = NULL;
520 606616 : int needed = 0;
521 :
522 1303320 : for (node *n = exps->h; n; n = n->next) {
523 696704 : sql_exp *e = n->data;
524 :
525 696704 : if (TRIVIAL_NOT_EQUAL_CMP(e)) {
526 82328 : bool appended = false;
527 :
528 82328 : if (inequality_groups) {
529 9095 : for (node *m = inequality_groups->h; m && !appended; m = m->next) {
530 5212 : list *next = m->data;
531 5212 : sql_exp *first = (sql_exp*) next->h->data;
532 :
533 5212 : if (exp_match(first->l, e->l)) {
534 1166 : list_append(next, e);
535 1166 : appended = true;
536 : }
537 : }
538 : }
539 3883 : if (!appended) {
540 81162 : if (!inequality_groups)
541 78445 : inequality_groups = new_exp_list(sql->sa);
542 81162 : list_append(inequality_groups, list_append(new_exp_list(sql->sa), e));
543 : }
544 : }
545 : }
546 :
547 606616 : if (inequality_groups) { /* if one list of inequalities has more than one entry, then the re-write is needed */
548 159607 : for (node *n = inequality_groups->h; n; n = n->next) {
549 81162 : list *next = n->data;
550 :
551 81162 : if (list_length(next) > 1)
552 1081 : needed = 1;
553 : }
554 : }
555 :
556 78445 : if (needed) {
557 999 : nexps = new_exp_list(sql->sa);
558 2614 : for (node *n = inequality_groups->h; n; n = n->next) {
559 1615 : list *next = n->data;
560 1615 : sql_exp *first = (sql_exp*) next->h->data;
561 :
562 1615 : if (list_length(next) > 1) {
563 1081 : list *notin = new_exp_list(sql->sa);
564 :
565 3328 : for (node *m = next->h; m; m = m->next) {
566 2247 : sql_exp *e = m->data;
567 2247 : list_append(notin, e->r);
568 : }
569 1081 : list_append(nexps, exp_in(sql->sa, first->l, notin, cmp_notin));
570 : } else {
571 534 : list_append(nexps, first);
572 : }
573 : }
574 :
575 4138 : for (node *n = exps->h; n; n = n->next) {
576 3139 : sql_exp *e = n->data;
577 :
578 3139 : if (!TRIVIAL_NOT_EQUAL_CMP(e))
579 358 : list_append(nexps, e);
580 : }
581 999 : (*changes)++;
582 : } else {
583 : nexps = exps;
584 : }
585 :
586 1302154 : for (node *n = nexps->h; n ; n = n->next) {
587 695538 : sql_exp *e = n->data;
588 :
589 695538 : if (e->type == e_cmp && e->flag == cmp_or) {
590 42051 : e->l = merge_notequal(sql, e->l, changes);
591 42051 : e->r = merge_notequal(sql, e->r, changes);
592 : }
593 : }
594 :
595 606616 : return nexps;
596 : }
597 :
598 : int
599 175257 : is_numeric_upcast(sql_exp *e)
600 : {
601 175257 : if (is_convert(e->type)) {
602 4675 : sql_subtype *f = exp_fromtype(e);
603 4675 : sql_subtype *t = exp_totype(e);
604 :
605 4675 : if (f->type->eclass == t->type->eclass && EC_COMPUTE(f->type->eclass)) {
606 1640 : if (f->type->localtype < t->type->localtype)
607 1620 : return 1;
608 : }
609 : }
610 : return 0;
611 : }
612 :
613 : /* optimize (a = b) or (a is null and b is null) -> a = b with null semantics */
614 : static sql_exp *
615 45166 : try_rewrite_equal_or_is_null(mvc *sql, sql_rel *rel, sql_exp *or, list *l1, list *l2)
616 : {
617 45166 : if (list_length(l1) == 1) {
618 41598 : bool valid = true, first_is_null_found = false, second_is_null_found = false;
619 41598 : sql_exp *cmp = l1->h->data;
620 41598 : sql_exp *first = cmp->l, *second = cmp->r;
621 :
622 41598 : if (is_compare(cmp->type) && !is_anti(cmp) && !cmp->f && cmp->flag == cmp_equal) {
623 7290 : int fupcast = is_numeric_upcast(first), supcast = is_numeric_upcast(second);
624 14845 : for(node *n = l2->h ; n && valid; n = n->next) {
625 7555 : sql_exp *e = n->data, *l = e->l, *r = e->r;
626 :
627 7555 : if (is_compare(e->type) && e->flag == cmp_equal && !e->f &&
628 4655 : !is_anti(e) && is_semantics(e)) {
629 1861 : int lupcast = is_numeric_upcast(l);
630 1861 : int rupcast = is_numeric_upcast(r);
631 1861 : sql_exp *rr = rupcast ? r->l : r;
632 :
633 1861 : if (rr->type == e_atom && rr->l && atom_null(rr->l)) {
634 1861 : if (exp_match_exp(fupcast?first->l:first, lupcast?l->l:l))
635 : first_is_null_found = true;
636 353 : else if (exp_match_exp(supcast?second->l:second, lupcast?l->l:l))
637 : second_is_null_found = true;
638 : else
639 5782 : valid = false;
640 : } else {
641 : valid = false;
642 : }
643 : } else {
644 : valid = false;
645 : }
646 : }
647 7290 : if (valid && first_is_null_found && second_is_null_found) {
648 262 : sql_subtype super;
649 :
650 262 : cmp_supertype(&super, exp_subtype(first), exp_subtype(second)); /* first and second must have the same type */
651 262 : if (!(first = exp_check_type(sql, &super, rel, first, type_equal)) ||
652 262 : !(second = exp_check_type(sql, &super, rel, second, type_equal))) {
653 0 : sql->session->status = 0;
654 0 : sql->errstr[0] = 0;
655 0 : return or;
656 : }
657 262 : sql_exp *res = exp_compare(sql->sa, first, second, cmp->flag);
658 262 : set_semantics(res);
659 262 : if (exp_name(or))
660 0 : exp_prop_alias(sql->sa, res, or);
661 262 : return res;
662 : }
663 : }
664 : }
665 : return or;
666 : }
667 :
668 : static list *
669 1032798 : merge_cmp_or_null(mvc *sql, sql_rel *rel, list *exps, int *changes)
670 : {
671 2220726 : for (node *n = exps->h; n ; n = n->next) {
672 1187928 : sql_exp *e = n->data;
673 :
674 1187928 : if (is_compare(e->type) && e->flag == cmp_or && !is_anti(e)) {
675 22583 : sql_exp *ne = try_rewrite_equal_or_is_null(sql, rel, e, e->l, e->r);
676 22583 : if (ne != e) {
677 260 : (*changes)++;
678 260 : n->data = ne;
679 : }
680 22583 : ne = try_rewrite_equal_or_is_null(sql, rel, e, e->r, e->l);
681 22583 : if (ne != e) {
682 2 : (*changes)++;
683 2 : n->data = ne;
684 : }
685 : }
686 : }
687 1032798 : return exps;
688 : }
689 :
690 : static list *
691 1032798 : cleanup_equal_exps(mvc *sql, sql_rel *rel, list *exps, int *changes)
692 : {
693 1032798 : if (list_length(exps) <= 1)
694 : return exps;
695 141499 : if (is_join(rel->op)) {
696 213576 : for(node *n = exps->h; n; n = n->next) {
697 143619 : sql_exp *e = n->data;
698 287069 : if (e->type == e_cmp && !e->f && e->flag <= cmp_notequal &&
699 185229 : !rel_find_exp(rel->l, e->l) && !rel_find_exp(rel->r, e->r) &&
700 41603 : !find_prop(e->p, PROP_HASHCOL) && !find_prop(e->p, PROP_JOINIDX)) {
701 20720 : exp_swap(e);
702 : }
703 : }
704 : }
705 141499 : bool needed = false;
706 439186 : for(node *n = exps->h; !needed && n; n = n->next) {
707 589518 : for (node *m = n->next; !needed && m; m = m->next) {
708 291831 : if (exp_match_exp_semantics(n->data, m->data, true))
709 89 : needed = true;
710 : }
711 : }
712 141499 : if (needed) {
713 89 : list *nexps = sa_list(sql->sa);
714 :
715 293 : for(node *n = exps->h; n; n = n->next) {
716 204 : bool done = false;
717 629 : for (node *m = exps->h; m && !done; m = m->next) {
718 425 : if (n != m && exp_match_exp_semantics(n->data, m->data, false)) {
719 179 : sql_exp *e1 = n->data, *e2 = m->data;
720 179 : if ((is_any(e1) || is_semantics(e1)) || (!is_any(e2) && !is_semantics(e2))) {
721 179 : append(nexps, e1);
722 179 : if ((!is_any(e2) && !is_semantics(e2)) && is_left(rel->op) && list_length(rel->attr) == 1) {
723 : /* nil is false */
724 0 : sql_exp *m = rel->attr->h->data;
725 0 : if (exp_is_atom(m))
726 0 : set_no_nil(m);
727 : }
728 : }
729 : done = true;
730 : }
731 : }
732 204 : if (!done)
733 25 : append(nexps, n->data);
734 : }
735 : return nexps;
736 : }
737 : (void)changes;
738 : return exps;
739 : }
740 :
741 : static inline sql_rel *
742 1032798 : rel_select_cse(visitor *v, sql_rel *rel)
743 : {
744 1032798 : if ((is_select(rel->op) || is_join(rel->op) || is_semi(rel->op)) && rel->exps) /* cleanup equal expressions */
745 1032798 : rel->exps = cleanup_equal_exps(v->sql, rel, rel->exps, &v->changes); /* (a = b) and (a += b) */
746 :
747 1032798 : if (is_select(rel->op) && rel->exps)
748 522514 : rel->exps = merge_ors(v->sql, rel->exps, &v->changes); /* x = 1 or x = 2 => x in (1, 2)*/
749 :
750 1032798 : if (is_select(rel->op) && rel->exps)
751 522514 : rel->exps = merge_notequal(v->sql, rel->exps, &v->changes); /* x <> 1 and x <> 2 => x not in (1, 2)*/
752 :
753 1032798 : if ((is_select(rel->op) || is_join(rel->op) || is_semi(rel->op)) && rel->exps)
754 1032798 : rel->exps = merge_cmp_or_null(v->sql, rel, rel->exps, &v->changes); /* (a = b) or (a is null and b is null) -> a = b with null semantics */
755 :
756 1032798 : if ((is_select(rel->op) || is_join(rel->op) || is_semi(rel->op)) && rel->exps) {
757 1032798 : node *n;
758 1032798 : list *nexps;
759 1032798 : int needed = 0;
760 :
761 2214104 : for (n=rel->exps->h; n && !needed; n = n->next) {
762 1181306 : sql_exp *e = n->data;
763 :
764 1181306 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e))
765 1181306 : needed = 1;
766 : }
767 1032798 : if (!needed)
768 : return rel;
769 21512 : nexps = new_exp_list(v->sql->sa);
770 51042 : for (n=rel->exps->h; n; n = n->next) {
771 29530 : sql_exp *e = n->data;
772 :
773 29530 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e)) {
774 : /* split the common expressions */
775 22321 : v->changes += exps_cse(v->sql, nexps, e->l, e->r);
776 : } else {
777 7209 : append(nexps, e);
778 : }
779 : }
780 21512 : rel->exps = nexps;
781 : }
782 : return rel;
783 : }
784 :
785 : static list *
786 14249 : exps_merge_select_rse( mvc *sql, list *l, list *r, bool *merged)
787 : {
788 14249 : node *n, *m, *o;
789 14249 : list *nexps = NULL, *lexps, *rexps;
790 14249 : bool lmerged = true, rmerged = true;
791 :
792 14249 : lexps = new_exp_list(sql->sa);
793 30232 : for (n = l->h; n; n = n->next) {
794 15983 : sql_exp *e = n->data;
795 :
796 15983 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e) && !is_semantics(e)) {
797 5488 : lmerged = false;
798 5488 : list *nexps = exps_merge_select_rse(sql, e->l, e->r, &lmerged);
799 6004 : for (o = nexps->h; o; o = o->next)
800 516 : append(lexps, o->data);
801 : } else {
802 10495 : append(lexps, e);
803 : }
804 : }
805 14249 : if (lmerged)
806 8820 : lmerged = (list_length(lexps) == 1);
807 14249 : rexps = new_exp_list(sql->sa);
808 30846 : for (n = r->h; n; n = n->next) {
809 16597 : sql_exp *e = n->data;
810 :
811 16597 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e) && !is_semantics(e)) {
812 1104 : rmerged = false;
813 1104 : list *nexps = exps_merge_select_rse(sql, e->l, e->r, &rmerged);
814 1127 : for (o = nexps->h; o; o = o->next)
815 23 : append(rexps, o->data);
816 : } else {
817 15493 : append(rexps, e);
818 : }
819 : }
820 14249 : if (rmerged)
821 13163 : rmerged = (list_length(r) == 1);
822 :
823 14249 : nexps = new_exp_list(sql->sa);
824 :
825 : /* merge merged lists first ? */
826 25260 : for (n = lexps->h; n; n = n->next) {
827 11011 : sql_exp *le = n->data, *re, *fnd = NULL;
828 :
829 11011 : if (le->type != e_cmp || le->flag == cmp_or || is_anti(le) || is_semantics(le) || is_symmetric(le))
830 622 : continue;
831 21207 : for (m = rexps->h; !fnd && m; m = m->next) {
832 10818 : re = m->data;
833 10818 : if (exps_match_col_exps(le, re))
834 1737 : fnd = re;
835 : }
836 10389 : if (fnd && (is_anti(fnd) || is_semantics(fnd)))
837 52 : continue;
838 : /* cases
839 : * 1) 2 values (cmp_equal)
840 : * 2) 1 value (cmp_equal), and cmp_in
841 : * (also cmp_in, cmp_equal)
842 : * 3) 2 cmp_in
843 : * 4) ranges
844 : */
845 1685 : if (fnd) {
846 1685 : re = fnd;
847 1685 : fnd = NULL;
848 1685 : if (is_anti(le) || is_anti(re) || is_symmetric(re))
849 0 : continue;
850 2239 : if (le->flag == cmp_equal && re->flag == cmp_equal) {
851 554 : list *exps = new_exp_list(sql->sa);
852 :
853 554 : append(exps, le->r);
854 554 : append(exps, re->r);
855 554 : fnd = exp_in(sql->sa, le->l, exps, cmp_in);
856 1302 : } else if (le->flag == cmp_equal && re->flag == cmp_in){
857 171 : list *exps = new_exp_list(sql->sa);
858 :
859 171 : append(exps, le->r);
860 171 : list_merge(exps, re->r, NULL);
861 171 : fnd = exp_in(sql->sa, le->l, exps, cmp_in);
862 1291 : } else if (le->flag == cmp_in && re->flag == cmp_equal){
863 331 : list *exps = new_exp_list(sql->sa);
864 :
865 331 : append(exps, re->r);
866 331 : list_merge(exps, le->r, NULL);
867 331 : fnd = exp_in(sql->sa, le->l, exps, cmp_in);
868 749 : } else if (le->flag == cmp_in && re->flag == cmp_in){
869 120 : list *exps = new_exp_list(sql->sa);
870 :
871 120 : list_merge(exps, le->r, NULL);
872 120 : list_merge(exps, re->r, NULL);
873 120 : fnd = exp_in(sql->sa, le->l, exps, cmp_in);
874 509 : } else if (le->f && re->f && /* merge ranges */
875 20 : le->flag == re->flag && le->flag <= cmp_lt) {
876 20 : sql_exp *mine = NULL, *maxe = NULL;
877 :
878 20 : if (!(mine = rel_binop_(sql, NULL, exp_copy(sql, le->r), exp_copy(sql, re->r), "sys", "sql_min", card_value, true))) {
879 0 : sql->session->status = 0;
880 0 : sql->errstr[0] = '\0';
881 0 : continue;
882 : }
883 20 : if (!(maxe = rel_binop_(sql, NULL, exp_copy(sql, le->f), exp_copy(sql, re->f), "sys", "sql_max", card_value, true))) {
884 0 : sql->session->status = 0;
885 0 : sql->errstr[0] = '\0';
886 0 : continue;
887 : }
888 20 : fnd = exp_compare2(sql->sa, exp_copy(sql, le->l), mine, maxe, le->flag, 0);
889 20 : lmerged = false;
890 : }
891 1196 : if (fnd) {
892 1196 : append(nexps, fnd);
893 2229 : *merged = (fnd && lmerged && rmerged);
894 : }
895 : }
896 : }
897 14249 : return nexps;
898 : }
899 :
900 : /* merge related sub expressions
901 : *
902 : * ie (x = a and y > 1 and y < 5) or
903 : * (x = c and y > 1 and y < 10) or
904 : * (x = e and y > 1 and y < 20)
905 : * ->
906 : * ((x = a and y > 1 and y < 5) or
907 : * (x = c and y > 1 and y < 10) or
908 : * (x = e and y > 1 and y < 20)) and
909 : * x in (a,c,e) and
910 : * y > 1 and y < 20
911 : *
912 : * for single expression or's we can do better
913 : * x in (a, b, c) or x in (d, e, f)
914 : * ->
915 : * x in (a, b, c, d, e, f)
916 : * */
917 : static inline sql_rel *
918 396752 : rel_merge_select_rse(visitor *v, sql_rel *rel)
919 : {
920 : /* only execute once per select */
921 396752 : if ((is_select(rel->op) || is_join(rel->op) || is_semi(rel->op)) && rel->exps && !is_rel_merge_select_rse_used(rel->used)) {
922 141140 : node *n, *o;
923 141140 : list *nexps = new_exp_list(v->sql->sa);
924 :
925 301952 : for (n=rel->exps->h; n; n = n->next) {
926 160812 : sql_exp *e = n->data;
927 :
928 168469 : if (e->type == e_cmp && e->flag == cmp_or && !is_anti(e) && !is_semantics(e)) {
929 : /* possibly merge related expressions */
930 7657 : bool merged = false;
931 :
932 7657 : list *ps = exps_merge_select_rse(v->sql, e->l, e->r, &merged);
933 8314 : for (o = ps->h; o; o = o->next)
934 657 : append(nexps, o->data);
935 7657 : if (merged)
936 86 : v->changes++;
937 : else
938 7571 : append(nexps, e);
939 : } else {
940 153155 : append(nexps, e);
941 : }
942 : }
943 141140 : rel->exps = nexps;
944 141140 : rel->used |= rel_merge_select_rse_used;
945 : }
946 396752 : return rel;
947 : }
948 :
949 : /* pack optimizers into a single function call to avoid iterations in the AST */
950 : static sql_rel *
951 3629656 : rel_optimize_select_and_joins_bottomup_(visitor *v, sql_rel *rel)
952 : {
953 3629656 : if (!rel || (!is_join(rel->op) && !is_semi(rel->op) && !is_select(rel->op)) || list_empty(rel->exps))
954 2596858 : return rel;
955 1032798 : uint8_t cycle = *(uint8_t*) v->data;
956 :
957 1032798 : rel->exps = exp_merge_range(v, rel, rel->exps);
958 1032798 : rel = rel_select_cse(v, rel);
959 1032798 : if (cycle == 1)
960 396752 : rel = rel_merge_select_rse(v, rel);
961 1032798 : rel = rewrite_simplify(v, cycle, v->value_based_opt, rel);
962 1032798 : return rel;
963 : }
964 :
965 : static sql_rel *
966 126483 : rel_optimize_select_and_joins_bottomup(visitor *v, global_props *gp, sql_rel *rel)
967 : {
968 126483 : v->data = &gp->opt_cycle;
969 126483 : rel = rel_visitor_bottomup(v, rel, &rel_optimize_select_and_joins_bottomup_);
970 126483 : v->data = gp;
971 126483 : return rel;
972 : }
973 :
974 : run_optimizer
975 623154 : bind_optimize_select_and_joins_bottomup(visitor *v, global_props *gp)
976 : {
977 623154 : int flag = v->sql->sql_optimizer;
978 622951 : return gp->opt_level == 1 && (gp->cnt[op_join] || gp->cnt[op_left] || gp->cnt[op_right] ||
979 538939 : gp->cnt[op_full] || gp->cnt[op_semi] || gp->cnt[op_anti] ||
980 1246105 : gp->cnt[op_select]) && (flag & optimize_select_and_joins_bottomup) ? rel_optimize_select_and_joins_bottomup : NULL;
981 : }
982 :
983 :
984 : static inline sql_rel *
985 3429853 : rel_push_join_exps_down(visitor *v, sql_rel *rel)
986 : {
987 : /* push select exps part of join expressions down */
988 : /* TODO CHECK WHY not semi enabled */
989 3429853 : if ((is_innerjoin(rel->op) || is_left(rel->op) || is_right(rel->op) /*|| is_semi(rel->op)*/) && !list_empty(rel->exps)) {
990 483249 : int left = is_innerjoin(rel->op) || is_right(rel->op) || rel->op == op_semi;
991 483249 : int right = is_innerjoin(rel->op) || is_left(rel->op) || is_semi(rel->op);
992 483249 : sql_rel *jl = rel->l, *ojl = jl, *jr = rel->r, *ojr = jr;
993 :
994 483249 : set_processed(jl);
995 483249 : set_processed(jr);
996 1040318 : for (node *n = rel->exps->h; n;) {
997 557069 : node *next = n->next;
998 557069 : sql_exp *e = n->data;
999 :
1000 557069 : if (left && rel_rebind_exp(v->sql, jl, e) && !is_any(e)) { /* select expressions on left */
1001 1335 : if (!is_select(jl->op) || rel_is_ref(jl))
1002 1325 : rel->l = jl = rel_select(v->sql->sa, jl, NULL);
1003 1335 : rel_select_add_exp(v->sql->sa, jl, e);
1004 1335 : list_remove_node(rel->exps, NULL, n);
1005 1335 : v->changes++;
1006 555734 : } else if (right && rel_rebind_exp(v->sql, jr, e) && !is_any(e)) { /* select expressions on right */
1007 42 : if (!is_select(jr->op) || rel_is_ref(jr))
1008 40 : rel->r = jr = rel_select(v->sql->sa, jr, NULL);
1009 42 : rel_select_add_exp(v->sql->sa, jr, e);
1010 42 : list_remove_node(rel->exps, NULL, n);
1011 42 : v->changes++;
1012 : }
1013 : n = next;
1014 : }
1015 483249 : if (ojl != jl)
1016 1325 : set_processed(jl);
1017 483249 : if (ojr != jr)
1018 40 : set_processed(jr);
1019 : }
1020 3429853 : return rel;
1021 : }
1022 :
1023 : static inline bool
1024 3429853 : is_non_trivial_select_applied_to_outer_join(sql_rel *rel)
1025 : {
1026 3429853 : return is_select(rel->op) && rel->exps && is_outerjoin(((sql_rel*) rel->l)->op);
1027 : }
1028 :
1029 : extern list *list_append_before(list *l, node *n, void *data);
1030 :
1031 : static void replace_column_references_with_nulls_2(mvc *sql, list* crefs, sql_exp* e);
1032 :
1033 : static void
1034 5650 : replace_column_references_with_nulls_1(mvc *sql, list* crefs, list* exps) {
1035 5650 : if (list_empty(exps))
1036 : return;
1037 14364 : for(node* n = exps->h; n; n=n->next) {
1038 8714 : sql_exp* e = n->data;
1039 8714 : replace_column_references_with_nulls_2(sql, crefs, e);
1040 : }
1041 : }
1042 :
1043 : static void
1044 28038 : replace_column_references_with_nulls_2(mvc *sql, list* crefs, sql_exp* e) {
1045 35735 : if (e == NULL) {
1046 : return;
1047 : }
1048 :
1049 29606 : switch (e->type) {
1050 8853 : case e_column:
1051 : {
1052 8853 : sql_exp *c = NULL;
1053 8853 : if (e->nid)
1054 8853 : c = exps_bind_nid(crefs, e->nid);
1055 8853 : if (c) {
1056 2449 : e->type = e_atom;
1057 2449 : e->l = atom_general(sql->sa, &e->tpe, NULL, 0);
1058 2449 : e->r = e->f = NULL;
1059 : }
1060 : }
1061 : break;
1062 9264 : case e_cmp:
1063 9264 : switch (e->flag) {
1064 6555 : case cmp_gt:
1065 : case cmp_gte:
1066 : case cmp_lte:
1067 : case cmp_lt:
1068 : case cmp_equal:
1069 : case cmp_notequal:
1070 : {
1071 6555 : sql_exp* l = e->l;
1072 6555 : sql_exp* r = e->r;
1073 6555 : sql_exp* f = e->f;
1074 :
1075 6555 : replace_column_references_with_nulls_2(sql, crefs, l);
1076 6555 : replace_column_references_with_nulls_2(sql, crefs, r);
1077 6555 : replace_column_references_with_nulls_2(sql, crefs, f);
1078 6555 : break;
1079 : }
1080 1945 : case cmp_filter:
1081 : case cmp_or:
1082 : {
1083 1945 : list* l = e->l;
1084 1945 : list* r = e->r;
1085 1945 : replace_column_references_with_nulls_1(sql, crefs, l);
1086 1945 : replace_column_references_with_nulls_1(sql, crefs, r);
1087 1945 : break;
1088 : }
1089 764 : case cmp_in:
1090 : case cmp_notin:
1091 : {
1092 764 : sql_exp* l = e->l;
1093 764 : list* r = e->r;
1094 764 : replace_column_references_with_nulls_2(sql, crefs, l);
1095 764 : replace_column_references_with_nulls_1(sql, crefs, r);
1096 764 : break;
1097 : }
1098 : default:
1099 : break;
1100 : }
1101 : break;
1102 996 : case e_func:
1103 : {
1104 996 : list* l = e->l;
1105 996 : replace_column_references_with_nulls_1(sql, crefs, l);
1106 996 : break;
1107 : }
1108 1142 : case e_convert:
1109 : {
1110 1142 : sql_exp* l = e->l;
1111 1142 : replace_column_references_with_nulls_2(sql, crefs, l);
1112 1142 : break;
1113 : }
1114 : default:
1115 : break;
1116 : }
1117 : }
1118 :
1119 : static sql_rel *
1120 4966 : out2inner(visitor *v, sql_rel* sel, sql_rel* join, sql_rel* inner_join_side, operator_type new_type) {
1121 :
1122 : /* handle inner_join relations with a simple select */
1123 4966 : if (is_select(inner_join_side->op) && inner_join_side->l)
1124 4966 : inner_join_side = inner_join_side->l;
1125 4966 : if (!is_base(inner_join_side->op) && !is_simple_project(inner_join_side->op)) {
1126 : // Nothing to do here.
1127 : return sel;
1128 : }
1129 :
1130 4836 : list* inner_join_column_references = inner_join_side->exps;
1131 4836 : list* select_predicates = exps_copy(v->sql, sel->exps);
1132 :
1133 10203 : for(node* n = select_predicates->h; n; n=n->next) {
1134 5450 : sql_exp* e = n->data;
1135 5450 : replace_column_references_with_nulls_2(v->sql, inner_join_column_references, e);
1136 :
1137 5450 : if (exp_is_false(e)) {
1138 83 : join->op = new_type;
1139 83 : v->changes++;
1140 83 : break;
1141 : }
1142 : }
1143 :
1144 : return sel;
1145 : }
1146 :
1147 : static inline sql_rel *
1148 3429853 : rel_out2inner(visitor *v, sql_rel *rel) {
1149 :
1150 3429853 : if (!is_non_trivial_select_applied_to_outer_join(rel)) {
1151 : // Nothing to do here.
1152 : return rel;
1153 : }
1154 :
1155 4943 : sql_rel* join = (sql_rel*) rel->l;
1156 :
1157 4943 : if (rel_is_ref(join)) {
1158 : /* Do not alter a multi-referenced join relation.
1159 : * This is problematic (e.g. in the case of the plan of a merge statement)
1160 : * basically because there are no guarantees on the other container relations.
1161 : * In particular there is no guarantee that the other referencing relations are
1162 : * select relations with null-rejacting predicates on the inner join side.
1163 : */
1164 : return rel;
1165 : }
1166 :
1167 4943 : sql_rel* inner_join_side;
1168 4943 : if (is_left(join->op)) {
1169 4894 : inner_join_side = join->r;
1170 4894 : return out2inner(v, rel, join, inner_join_side, op_join);
1171 : }
1172 49 : else if (is_right(join->op)) {
1173 26 : inner_join_side = join->l;
1174 26 : return out2inner(v, rel, join, inner_join_side, op_join);
1175 : }
1176 : else /*full outer join*/ {
1177 : // First check if left side can degenerate from full outer join to just right outer join.
1178 23 : inner_join_side = join->r;
1179 23 : rel = out2inner(v, rel, join, inner_join_side, op_right);
1180 : /* Now test if the right side can degenerate to
1181 : * a normal inner join or a left outer join
1182 : * depending on the result of previous call to out2inner.
1183 : */
1184 :
1185 23 : inner_join_side = join->l;
1186 38 : return out2inner(v, rel, join, inner_join_side, is_right(join->op)? op_join: op_left);
1187 : }
1188 : }
1189 :
1190 : static bool
1191 4219 : exps_uses_any(list *exps, list *l)
1192 : {
1193 4219 : bool uses_any = false;
1194 :
1195 4219 : if (list_empty(exps) || list_empty(l))
1196 14 : return false;
1197 8416 : for (node *n = l->h; n && !uses_any; n = n->next) {
1198 4211 : sql_exp *e = n->data;
1199 4211 : uses_any |= list_exps_uses_exp(exps, exp_relname(e), exp_name(e)) != NULL;
1200 : }
1201 :
1202 : return uses_any;
1203 : }
1204 :
1205 : /* TODO At the moment I have to disable the new join2semi because the join order optimizer doesn't take semi-joins into account,
1206 : so plans get deteriorated if more joins are optimized into semi-joins. Later I will review the join order with semi-joins and hopefully,
1207 : I will be able to re-enable the new join2semi. */
1208 : #if 0
1209 : #define NO_EXP_FOUND 0
1210 : #define FOUND_WITH_DUPLICATES 1
1211 : #define MAY_HAVE_DUPLICATE_NULLS 2
1212 : #define ALL_VALUES_DISTINCT 3
1213 :
1214 : static int
1215 : find_projection_for_join2semi(sql_rel *rel, sql_exp *jc)
1216 : {
1217 : sql_rel *res = NULL;
1218 : sql_exp *e = NULL;
1219 : bool underjoin = false;
1220 :
1221 : if ((e = rel_find_exp_and_corresponding_rel(rel, jc, &res, &underjoin))) {
1222 : if (underjoin || e->type != e_column)
1223 : return FOUND_WITH_DUPLICATES;
1224 : /* if just one groupby column is projected or the relation needs distinct values and one column is projected or is a primary key, it will be distinct */
1225 : if (is_unique(e) ||
1226 : (is_groupby(res->op) && list_length(res->r) == 1 && exps_find_exp(res->r, e)) ||
1227 : ((is_project(res->op) || is_base(res->op)) && ((need_distinct(res) && list_length(res->exps) == 1) || res->card < CARD_AGGR)))
1228 : return has_nil(e) ? MAY_HAVE_DUPLICATE_NULLS : ALL_VALUES_DISTINCT;
1229 : return FOUND_WITH_DUPLICATES;
1230 : }
1231 : return NO_EXP_FOUND;
1232 : }
1233 :
1234 : static int
1235 : subrel_uses_exp_outside_subrel(visitor *v, sql_rel *rel, list *l, sql_rel *j)
1236 : {
1237 : if (rel == j)
1238 : return 0;
1239 : if (mvc_highwater(v->sql))
1240 : return 1;
1241 : switch(rel->op){
1242 : case op_join:
1243 : case op_left:
1244 : case op_right:
1245 : case op_full:
1246 : return exps_uses_any(rel->exps, l) ||
1247 : subrel_uses_exp_outside_subrel(v, rel->l, l, j) || subrel_uses_exp_outside_subrel(v, rel->r, l, j);
1248 : case op_semi:
1249 : case op_anti:
1250 : case op_select:
1251 : return exps_uses_any(rel->exps, l) ||
1252 : subrel_uses_exp_outside_subrel(v, rel->l, l, j);
1253 : case op_project:
1254 : case op_groupby:
1255 : return exps_uses_any(rel->exps, l) || exps_uses_any(rel->r, l);
1256 : case op_basetable:
1257 : case op_table:
1258 : case op_union:
1259 : case op_except:
1260 : case op_inter:
1261 : return exps_uses_any(rel->exps, l);
1262 : case op_topn:
1263 : case op_sample:
1264 : return subrel_uses_exp_outside_subrel(v, rel->l, l, j);
1265 : default:
1266 : return 1;
1267 : }
1268 : }
1269 :
1270 : static int
1271 : projrel_uses_exp_outside_subrel(visitor *v, sql_rel *rel, list *l, sql_rel *j)
1272 : {
1273 : /* test if projecting relation uses any of the join expressions */
1274 : assert((is_simple_project(rel->op) || is_groupby(rel->op)) && rel->l);
1275 : return exps_uses_any(rel->exps, l) || exps_uses_any(rel->r, l) || subrel_uses_exp_outside_subrel(v, rel->l, l, j);
1276 : }
1277 :
1278 : static sql_rel *
1279 : rewrite_joins2semi(visitor *v, sql_rel *proj, sql_rel *rel)
1280 : {
1281 : /* generalize possibility : we need the visitor 'step' here */
1282 : if (rel_is_ref(rel) || mvc_highwater(v->sql)) /* if the join has multiple references, it's dangerous to convert it into a semijoin */
1283 : return rel;
1284 : if (is_innerjoin(rel->op) && !list_empty(rel->exps)) {
1285 : sql_rel *l = rel->l, *r = rel->r;
1286 : bool left_unique = true, right_unique = true;
1287 :
1288 : /* these relations don't project anything, so skip them */
1289 : while (is_topn(l->op) || is_sample(l->op) || is_select(l->op) || is_semi(l->op))
1290 : l = l->l;
1291 : /* joins will expand values, so don't search on those */
1292 : if (!is_base(l->op) && !is_project(l->op))
1293 : left_unique = false;
1294 : while (is_topn(r->op) || is_sample(r->op) || is_select(r->op) || is_semi(r->op))
1295 : r = r->l;
1296 : if (!is_base(r->op) && !is_project(r->op))
1297 : right_unique = false;
1298 : /* if all columns used in equi-joins from one of the sides are unique, the join can be rewritten into a semijoin */
1299 : for (node *n=rel->exps->h; n && (left_unique || right_unique); n = n->next) {
1300 : sql_exp *e = n->data, *el = e->l, *er = e->r;
1301 :
1302 : if (!is_compare(e->type) || e->flag != cmp_equal || exp_has_func(el) || exp_has_func(er)) {
1303 : left_unique = right_unique = false;
1304 : } else {
1305 : int found = 0;
1306 :
1307 : if (left_unique && (found = find_projection_for_join2semi(l, el)) > NO_EXP_FOUND)
1308 : left_unique &= (found == ALL_VALUES_DISTINCT || (found == MAY_HAVE_DUPLICATE_NULLS && (!is_semantics(e) || !has_nil(er))));
1309 : if (left_unique && (found = find_projection_for_join2semi(l, er)) > NO_EXP_FOUND)
1310 : left_unique &= (found == ALL_VALUES_DISTINCT || (found == MAY_HAVE_DUPLICATE_NULLS && (!is_semantics(e) || !has_nil(el))));
1311 : if (right_unique && (found = find_projection_for_join2semi(r, el)) > NO_EXP_FOUND)
1312 : right_unique &= (found == ALL_VALUES_DISTINCT || (found == MAY_HAVE_DUPLICATE_NULLS && (!is_semantics(e) || !has_nil(er))));
1313 : if (right_unique && (found = find_projection_for_join2semi(r, er)) > NO_EXP_FOUND)
1314 : right_unique &= (found == ALL_VALUES_DISTINCT || (found == MAY_HAVE_DUPLICATE_NULLS && (!is_semantics(e) || !has_nil(el))));
1315 : }
1316 : }
1317 :
1318 : /* now we need to check relation's expressions are not used */
1319 : if (left_unique && !projrel_uses_exp_outside_subrel(v, proj, l->exps, rel)) {
1320 : sql_rel *tmp = rel->r;
1321 : rel->r = rel->l;
1322 : rel->l = tmp;
1323 : rel->op = op_semi;
1324 : v->changes++;
1325 : } else if (right_unique && !projrel_uses_exp_outside_subrel(v, proj, r->exps, rel)) {
1326 : rel->op = op_semi;
1327 : v->changes++;
1328 : }
1329 : }
1330 : if (is_join(rel->op)) {
1331 : rel->l = rewrite_joins2semi(v, proj, rel->l);
1332 : rel->r = rewrite_joins2semi(v, proj, rel->r);
1333 : } else if (is_topn(rel->op) || is_sample(rel->op) || is_select(rel->op) || is_semi(rel->op)) {
1334 : rel->l = rewrite_joins2semi(v, proj, rel->l);
1335 : }
1336 : return rel;
1337 : }
1338 :
1339 : static inline sql_rel *
1340 : rel_join2semijoin(visitor *v, sql_rel *rel)
1341 : {
1342 : if ((is_simple_project(rel->op) || is_groupby(rel->op)) && rel->l)
1343 : rel->l = rewrite_joins2semi(v, rel, rel->l);
1344 : return rel;
1345 : }
1346 : #endif
1347 :
1348 : #define NO_PROJECTION_FOUND 0
1349 : #define MAY_HAVE_DUPLICATE_NULLS 1
1350 : #define ALL_VALUES_DISTINCT 2
1351 :
1352 : static int
1353 704805 : find_projection_for_join2semi(sql_rel *rel)
1354 : {
1355 704805 : if (is_simple_project(rel->op) || is_groupby(rel->op) || is_inter(rel->op) || is_except(rel->op) || is_base(rel->op) || (is_union(rel->op) && need_distinct(rel))) {
1356 373516 : if (rel->card < CARD_AGGR) /* const or groupby without group by exps */
1357 : return ALL_VALUES_DISTINCT;
1358 372369 : if (list_length(rel->exps) == 1) {
1359 6627 : sql_exp *e = rel->exps->h->data;
1360 : /* a single group by column in the projection list from a group by relation is guaranteed to be unique, but not an aggregate */
1361 6627 : if (e->type == e_column) {
1362 6597 : sql_rel *res = NULL;
1363 6597 : sql_exp *found = NULL;
1364 6597 : bool underjoin = false;
1365 :
1366 : /* if just one groupby column is projected or the relation needs distinct values and one column is projected or is a primary key, it will be distinct */
1367 6597 : if ((is_groupby(rel->op) && list_length(rel->r) == 1 && exps_find_exp(rel->r, e)) || (need_distinct(rel) && list_length(rel->exps) == 1))
1368 9528 : return ALL_VALUES_DISTINCT;
1369 2958 : if (is_unique(e))
1370 2249 : return has_nil(e) ? MAY_HAVE_DUPLICATE_NULLS : ALL_VALUES_DISTINCT;
1371 :
1372 709 : if ((is_simple_project(rel->op) || is_groupby(rel->op) || is_inter(rel->op) || is_except(rel->op)) &&
1373 129 : (found = rel_find_exp_and_corresponding_rel(rel->l, e, false, &res, &underjoin)) && !underjoin) { /* grouping column on inner relation */
1374 113 : if (need_distinct(res) && list_length(res->exps) == 1)
1375 : return ALL_VALUES_DISTINCT;
1376 113 : if (is_unique(found))
1377 0 : return has_nil(e) ? MAY_HAVE_DUPLICATE_NULLS : ALL_VALUES_DISTINCT;
1378 113 : if (found->type == e_column && found->card <= CARD_AGGR) {
1379 1 : if (!is_groupby(res->op) && list_length(res->exps) != 1)
1380 : return NO_PROJECTION_FOUND;
1381 0 : for (node *n = res->exps->h ; n ; n = n->next) { /* must be the single column in the group by expression list */
1382 0 : sql_exp *e = n->data;
1383 0 : if (e != found && e->type == e_column)
1384 : return NO_PROJECTION_FOUND;
1385 : }
1386 : return ALL_VALUES_DISTINCT;
1387 : }
1388 : }
1389 : }
1390 : }
1391 : }
1392 : return NO_PROJECTION_FOUND;
1393 : }
1394 :
1395 : static sql_rel *
1396 2245280 : find_candidate_join2semi(visitor *v, sql_rel *rel, bool *swap)
1397 : {
1398 : /* generalize possibility : we need the visitor 'step' here */
1399 2245800 : if (rel_is_ref(rel)) /* if the join has multiple references, it's dangerous to convert it into a semijoin */
1400 : return NULL;
1401 2128448 : if (rel->op == op_join && !list_empty(rel->exps) && list_empty(rel->attr)) {
1402 354569 : sql_rel *l = rel->l, *r = rel->r;
1403 354569 : int foundr = NO_PROJECTION_FOUND, foundl = NO_PROJECTION_FOUND, found = NO_PROJECTION_FOUND;
1404 354569 : bool ok = false;
1405 :
1406 354569 : foundr = find_projection_for_join2semi(r);
1407 354569 : if (foundr < ALL_VALUES_DISTINCT)
1408 350236 : foundl = find_projection_for_join2semi(l);
1409 354569 : if (foundr && foundr > foundl) {
1410 4339 : *swap = false;
1411 4339 : found = foundr;
1412 350230 : } else if (foundl) {
1413 2689 : *swap = true;
1414 2689 : found = foundl;
1415 : }
1416 :
1417 7028 : if (found > NO_PROJECTION_FOUND) {
1418 : /* if all join expressions can be pushed down or have function calls, then it cannot be rewritten into a semijoin */
1419 14060 : for (node *n=rel->exps->h; n && !ok; n = n->next) {
1420 7032 : sql_exp *e = n->data;
1421 :
1422 10796 : ok |= e->type == e_cmp && e->flag == cmp_equal && !exp_has_func(e) && !rel_rebind_exp(v->sql, l, e) && !rel_rebind_exp(v->sql, r, e) &&
1423 7 : (found == ALL_VALUES_DISTINCT || !is_semantics(e) || !has_nil((sql_exp *)e->l) || !has_nil((sql_exp *)e->r));
1424 : }
1425 : }
1426 :
1427 354569 : if (ok)
1428 : return rel;
1429 : }
1430 2125173 : if (is_join(rel->op) || is_semi(rel->op)) {
1431 534169 : sql_rel *c;
1432 :
1433 534169 : if ((c=find_candidate_join2semi(v, rel->l, swap)) != NULL ||
1434 533779 : (c=find_candidate_join2semi(v, rel->r, swap)) != NULL)
1435 412 : if (list_empty(c->attr))
1436 : return c;
1437 : }
1438 2124761 : if (is_topn(rel->op) || is_sample(rel->op))
1439 520 : return find_candidate_join2semi(v, rel->l, swap);
1440 : return NULL;
1441 : }
1442 :
1443 : static int
1444 2507 : subrel_uses_exp_outside_subrel(sql_rel *rel, list *l, sql_rel *c)
1445 : {
1446 2508 : if (rel == c)
1447 : return 0;
1448 : /* for subrel only expect joins (later possibly selects) */
1449 825 : if (is_join(rel->op) || is_semi(rel->op)) {
1450 418 : if (exps_uses_any(rel->exps, l))
1451 : return 1;
1452 818 : if (subrel_uses_exp_outside_subrel(rel->l, l, c) ||
1453 408 : subrel_uses_exp_outside_subrel(rel->r, l, c))
1454 5 : return 1;
1455 : }
1456 812 : if (is_topn(rel->op) || is_sample(rel->op))
1457 1 : return subrel_uses_exp_outside_subrel(rel->l, l, c);
1458 : return 0;
1459 : }
1460 :
1461 : static int
1462 3275 : rel_uses_exp_outside_subrel(sql_rel *rel, list *l, sql_rel *c)
1463 : {
1464 : /* for now we only expect sub relations of type project, selects (rel) or join/semi */
1465 3275 : if (is_simple_project(rel->op) || is_groupby(rel->op) || is_select(rel->op)) {
1466 3275 : if (!list_empty(rel->exps) && exps_uses_any(rel->exps, l))
1467 : return 1;
1468 1689 : if ((is_simple_project(rel->op) || is_groupby(rel->op)) && !list_empty(rel->r) && exps_uses_any(rel->r, l))
1469 : return 1;
1470 1689 : if (rel->l)
1471 1689 : return subrel_uses_exp_outside_subrel(rel->l, l, c);
1472 : }
1473 0 : if (is_topn(rel->op) || is_sample(rel->op))
1474 0 : return subrel_uses_exp_outside_subrel(rel->l, l, c);
1475 : return 1;
1476 : }
1477 :
1478 : static inline sql_rel *
1479 3429853 : rel_join2semijoin(visitor *v, sql_rel *rel)
1480 : {
1481 3429853 : if ((is_simple_project(rel->op) || is_groupby(rel->op)) && rel->l) {
1482 1177332 : bool swap = false;
1483 1177332 : sql_rel *l = rel->l;
1484 1177332 : sql_rel *c = find_candidate_join2semi(v, l, &swap);
1485 :
1486 1177332 : if (c) {
1487 : /* 'p' is a project */
1488 3275 : sql_rel *p = swap ? c->l : c->r;
1489 :
1490 : /* now we need to check if ce is only used at the level of c */
1491 3275 : if (!rel_uses_exp_outside_subrel(rel, p->exps, c)) {
1492 1681 : c->op = op_semi;
1493 1681 : if (swap) {
1494 518 : sql_rel *tmp = c->r;
1495 518 : c->r = c->l;
1496 518 : c->l = tmp;
1497 : }
1498 1681 : v->changes++;
1499 : }
1500 : }
1501 : }
1502 3429853 : return rel;
1503 : }
1504 :
1505 : static inline sql_rel *
1506 3429853 : rel_push_join_down_outer(visitor *v, sql_rel *rel)
1507 : {
1508 3429853 : if (is_join(rel->op) && !is_outerjoin(rel->op) && !is_single(rel) && !list_empty(rel->exps) && !rel_is_ref(rel)) {
1509 365209 : sql_rel *l = rel->l, *r = rel->r;
1510 :
1511 365209 : if (is_left(r->op) && (is_select(l->op) || (is_join(l->op) && !is_outerjoin(l->op))) && !rel_is_ref(l) &&
1512 888 : !rel_is_ref(r)) {
1513 888 : sql_rel *rl = r->l;
1514 888 : sql_rel *rr = r->r;
1515 888 : if (rel_is_ref(rl) || rel_is_ref(rr))
1516 : return rel;
1517 : /* join exps should only include l and r.l */
1518 888 : list *njexps = sa_list(v->sql->sa);
1519 2025 : for(node *n = rel->exps->h; n; n = n->next) {
1520 1137 : sql_exp *je = n->data;
1521 :
1522 1137 : assert(je->type == e_cmp);
1523 1137 : if (je->f)
1524 : return rel;
1525 1137 : if ((rel_find_exp(l, je->l) && rel_find_exp(rl, je->r)) || (rel_find_exp(l, je->r) && rel_find_exp(rl, je->l))) {
1526 1137 : list_append(njexps, je);
1527 : } else {
1528 0 : return rel;
1529 : }
1530 : }
1531 888 : sql_rel *nl = rel_crossproduct(v->sql->sa, rel_dup(l), rl, rel->op);
1532 888 : r->l = nl;
1533 888 : nl->exps = njexps;
1534 888 : nl->attr = rel->attr;
1535 888 : rel->attr = NULL;
1536 888 : set_processed(nl);
1537 888 : rel_dup(r);
1538 888 : rel_destroy(rel);
1539 888 : rel = r;
1540 888 : v->changes++;
1541 : }
1542 : }
1543 : return rel;
1544 : }
1545 :
1546 : static sql_rel *
1547 3429853 : rel_optimize_joins_(visitor *v, sql_rel *rel)
1548 : {
1549 3429853 : rel = rel_push_join_exps_down(v, rel);
1550 3429853 : rel = rel_out2inner(v, rel);
1551 3429853 : rel = rel_join2semijoin(v, rel);
1552 3429853 : rel = rel_push_join_down_outer(v, rel);
1553 3429853 : return rel;
1554 : }
1555 :
1556 : static sql_rel *
1557 89528 : rel_optimize_joins(visitor *v, global_props *gp, sql_rel *rel)
1558 : {
1559 89528 : (void) gp;
1560 89528 : return rel_visitor_topdown(v, rel, &rel_optimize_joins_);
1561 : }
1562 :
1563 : run_optimizer
1564 623186 : bind_optimize_joins(visitor *v, global_props *gp)
1565 : {
1566 623186 : int flag = v->sql->sql_optimizer;
1567 622991 : return gp->opt_level == 1 && (gp->cnt[op_join] || gp->cnt[op_left] || gp->cnt[op_right]
1568 1246177 : || gp->cnt[op_full] || gp->cnt[op_semi] || gp->cnt[op_anti]) && (flag & optimize_joins) ? rel_optimize_joins : NULL;
1569 : }
1570 :
1571 :
1572 : static sql_rel *rel_join_order_(visitor *v, sql_rel *rel);
1573 :
1574 : static void
1575 701252 : get_relations(visitor *v, sql_rel *rel, list *rels)
1576 : {
1577 975255 : if (list_empty(rel->attr) && !rel_is_ref(rel) && rel->op == op_join && rel->exps == NULL) {
1578 274003 : sql_rel *l = rel->l;
1579 274003 : sql_rel *r = rel->r;
1580 :
1581 274003 : get_relations(v, l, rels);
1582 274003 : get_relations(v, r, rels);
1583 274003 : rel->l = NULL;
1584 274003 : rel->r = NULL;
1585 274003 : rel_destroy(rel);
1586 : } else {
1587 427249 : rel = rel_join_order_(v, rel);
1588 427249 : append(rels, rel);
1589 : }
1590 701252 : }
1591 :
1592 : static void
1593 189006 : get_inner_relations(mvc *sql, sql_rel *rel, list *rels)
1594 : {
1595 299996 : if (!rel_is_ref(rel) && is_join(rel->op)) {
1596 110990 : sql_rel *l = rel->l;
1597 110990 : sql_rel *r = rel->r;
1598 :
1599 110990 : get_inner_relations(sql, l, rels);
1600 110990 : get_inner_relations(sql, r, rels);
1601 : } else {
1602 189006 : append(rels, rel);
1603 : }
1604 189006 : }
1605 :
1606 : static int
1607 1054364 : exp_count(int *cnt, sql_exp *e)
1608 : {
1609 1054364 : int flag;
1610 1054364 : if (!e)
1611 : return 0;
1612 1054364 : if (find_prop(e->p, PROP_JOINIDX))
1613 3541 : *cnt += 100;
1614 1054364 : if (find_prop(e->p, PROP_HASHCOL))
1615 34193 : *cnt += 100;
1616 1054364 : if (find_prop(e->p, PROP_HASHIDX))
1617 0 : *cnt += 100;
1618 1054364 : switch(e->type) {
1619 389386 : case e_cmp:
1620 389386 : if (!is_complex_exp(e->flag)) {
1621 330960 : exp_count(cnt, e->l);
1622 330960 : exp_count(cnt, e->r);
1623 330960 : if (e->f)
1624 3052 : exp_count(cnt, e->f);
1625 : }
1626 389386 : flag = e->flag;
1627 389386 : switch (flag) {
1628 307130 : case cmp_equal:
1629 307130 : *cnt += 90;
1630 307130 : return 90;
1631 17330 : case cmp_notequal:
1632 17330 : *cnt += 7;
1633 17330 : return 7;
1634 6500 : case cmp_gt:
1635 : case cmp_gte:
1636 : case cmp_lt:
1637 : case cmp_lte:
1638 6500 : *cnt += 6;
1639 6500 : if (e->l) {
1640 6500 : sql_exp *l = e->l;
1641 6500 : sql_subtype *t = exp_subtype(l);
1642 6500 : if (EC_TEMP(t->type->eclass)) /* give preference to temporal ranges */
1643 172 : *cnt += 90;
1644 : }
1645 6500 : if (e->f){ /* range */
1646 3052 : *cnt += 6;
1647 3052 : return 12;
1648 : }
1649 : return 6;
1650 693 : case cmp_filter:
1651 693 : if (exps_card(e->r) > CARD_AGGR) {
1652 : /* filters for joins are special */
1653 5 : *cnt += 1000;
1654 5 : return 1000;
1655 : }
1656 688 : *cnt += 2;
1657 688 : return 2;
1658 52932 : case cmp_in:
1659 : case cmp_notin: {
1660 52932 : list *l = e->r;
1661 52932 : int c = 9 - 10*list_length(l);
1662 52932 : *cnt += c;
1663 52932 : return c;
1664 : }
1665 4801 : case cmp_or: /* prefer or over functions */
1666 4801 : *cnt += 3;
1667 4801 : return 3;
1668 : default:
1669 : return 0;
1670 : }
1671 525030 : case e_column:
1672 525030 : *cnt += 20;
1673 525030 : return 20;
1674 121516 : case e_atom:
1675 121516 : *cnt += 10;
1676 121516 : return 10;
1677 2856 : case e_func:
1678 : /* functions are more expensive, depending on the number of columns involved. */
1679 2856 : if (e->card == CARD_ATOM)
1680 : return 0;
1681 2565 : *cnt -= 5*list_length(e->l);
1682 2565 : return 5*list_length(e->l);
1683 15576 : case e_convert:
1684 : /* functions are more expensive, depending on the number of columns involved. */
1685 15576 : if (e->card == CARD_ATOM)
1686 : return 0;
1687 : /* fall through */
1688 : default:
1689 14916 : *cnt -= 5;
1690 14916 : return -5;
1691 : }
1692 : }
1693 :
1694 : int
1695 264669 : exp_keyvalue(sql_exp *e)
1696 : {
1697 264669 : int cnt = 0;
1698 58433 : exp_count(&cnt, e);
1699 264669 : return cnt;
1700 : }
1701 :
1702 : static sql_exp *
1703 740372 : joinexp_col(sql_exp *e, sql_rel *r)
1704 : {
1705 740372 : if (e->type == e_cmp) {
1706 740372 : if (rel_has_exp(r, e->l, false) >= 0)
1707 395954 : return e->l;
1708 344418 : return e->r;
1709 : }
1710 0 : assert(0);
1711 : return NULL;
1712 : }
1713 :
1714 : static sql_column *
1715 489858 : table_colexp(sql_exp *e, sql_rel *r)
1716 : {
1717 489858 : sql_table *t = r->l;
1718 :
1719 489858 : if (e->type == e_column) {
1720 475940 : const char *name = exp_name(e);
1721 475940 : node *cn;
1722 :
1723 475940 : if (r->exps) { /* use alias */
1724 889895 : for (cn = r->exps->h; cn; cn = cn->next) {
1725 885621 : sql_exp *ce = cn->data;
1726 885621 : if (strcmp(exp_name(ce), name) == 0) {
1727 471666 : name = ce->r;
1728 471666 : break;
1729 : }
1730 : }
1731 : }
1732 1043226 : for (cn = ol_first_node(t->columns); cn; cn = cn->next) {
1733 1038952 : sql_column *c = cn->data;
1734 1038952 : if (strcmp(c->base.name, name) == 0)
1735 471666 : return c;
1736 : }
1737 : }
1738 : return NULL;
1739 : }
1740 :
1741 : static list *
1742 356032 : matching_joins(allocator *sa, list *rels, list *exps, sql_exp *je)
1743 : {
1744 356032 : sql_rel *l, *r;
1745 :
1746 356032 : assert (je->type == e_cmp);
1747 :
1748 356032 : l = find_rel(rels, je->l);
1749 356032 : r = find_rel(rels, je->r);
1750 356032 : if (l && r) {
1751 355888 : list *res;
1752 355888 : list *n_rels = sa_list(sa);
1753 :
1754 355888 : append(n_rels, l);
1755 355888 : append(n_rels, r);
1756 355888 : res = list_select(exps, n_rels, (fcmp) &exp_joins_rels, (fdup)NULL);
1757 355888 : return res;
1758 : }
1759 144 : return sa_list(sa);
1760 : }
1761 :
1762 : static int
1763 6439 : sql_column_kc_cmp(sql_column *c, sql_kc *kc)
1764 : {
1765 : /* return on equality */
1766 6439 : return (c->colnr - kc->c->colnr);
1767 : }
1768 :
1769 : static sql_idx *
1770 460637 : find_fk_index(mvc *sql, sql_table *l, list *lcols, sql_table *r, list *rcols)
1771 : {
1772 460637 : sql_trans *tr = sql->session->tr;
1773 :
1774 460637 : if (l->idxs) {
1775 460637 : node *in;
1776 547732 : for (in = ol_first_node(l->idxs); in; in = in->next){
1777 87935 : sql_idx *li = in->data;
1778 87935 : if (li->type == join_idx) {
1779 8717 : sql_key *rk = (sql_key*)os_find_id(tr->cat->objects, tr, ((sql_fkey*)li->key)->rkey);
1780 8717 : fcmp cmp = (fcmp)&sql_column_kc_cmp;
1781 :
1782 9666 : if (rk->t == r &&
1783 1789 : list_match(lcols, li->columns, cmp) == 0 &&
1784 840 : list_match(rcols, rk->columns, cmp) == 0) {
1785 840 : return li;
1786 : }
1787 : }
1788 : }
1789 : }
1790 : return NULL;
1791 : }
1792 :
1793 : static sql_rel *
1794 712064 : find_basetable( sql_rel *r)
1795 : {
1796 1065865 : if (!r)
1797 : return NULL;
1798 1058204 : switch(r->op) {
1799 572827 : case op_basetable:
1800 572827 : if (!r->l)
1801 0 : return NULL;
1802 : return r;
1803 353801 : case op_semi:
1804 : case op_anti:
1805 : case op_project:
1806 : case op_select:
1807 : case op_topn:
1808 : case op_sample:
1809 353801 : return find_basetable(r->l);
1810 : default:
1811 : return NULL;
1812 : }
1813 : }
1814 :
1815 : static int
1816 98938 : exps_count(list *exps)
1817 : {
1818 98938 : node *n;
1819 98938 : int cnt = 0;
1820 :
1821 98938 : if (!exps)
1822 : return 0;
1823 223661 : for (n = exps->h; n; n=n->next)
1824 124723 : exp_count(&cnt, n->data);
1825 98938 : return cnt;
1826 : }
1827 :
1828 : static list *
1829 253956 : order_join_expressions(mvc *sql, list *dje, list *rels)
1830 : {
1831 253956 : node *n;
1832 253956 : int cnt = list_length(dje);
1833 :
1834 253956 : if (cnt <= 1)
1835 : return dje;
1836 :
1837 65473 : list *res = sa_list(sql->sa);
1838 65473 : int i, *keys = SA_NEW_ARRAY(sql->ta, int, cnt);
1839 65473 : void **data = SA_NEW_ARRAY(sql->ta, void*, cnt);
1840 :
1841 271709 : for (n = dje->h, i = 0; n; n = n->next, i++) {
1842 206236 : sql_exp *e = n->data;
1843 :
1844 206236 : keys[i] = exp_keyvalue(e);
1845 : /* add some weight for the selections */
1846 206236 : if (e->type == e_cmp && !is_complex_exp(e->flag)) {
1847 206210 : sql_rel *l = find_rel(rels, e->l);
1848 206210 : sql_rel *r = find_rel(rels, e->r);
1849 :
1850 206210 : if (l && is_select(l->op) && l->exps)
1851 52981 : keys[i] += list_length(l->exps)*10 + exps_count(l->exps);
1852 206210 : if (r && is_select(r->op) && r->exps)
1853 45957 : keys[i] += list_length(r->exps)*10 + exps_count(r->exps);
1854 : }
1855 206236 : data[i] = n->data;
1856 : }
1857 : /* sort descending */
1858 65473 : GDKqsort(keys, data, NULL, cnt, sizeof(int), sizeof(void *), TYPE_int, true, true);
1859 337182 : for(i=0; i<cnt; i++) {
1860 206236 : list_append(res, data[i]);
1861 : }
1862 : return res;
1863 : }
1864 :
1865 : static int
1866 237512 : find_join_rels(list **L, list **R, list *exps, list *rels)
1867 : {
1868 237512 : node *n;
1869 :
1870 237512 : *L = sa_list(exps->sa);
1871 237512 : *R = sa_list(exps->sa);
1872 237512 : if (!exps || list_length(exps) <= 1)
1873 : return -1;
1874 289772 : for(n = exps->h; n; n = n->next) {
1875 218682 : sql_exp *e = n->data;
1876 218682 : sql_rel *l = NULL, *r = NULL;
1877 :
1878 218682 : if (!is_complex_exp(e->flag)){
1879 218675 : l = find_rel(rels, e->l);
1880 218675 : r = find_rel(rels, e->r);
1881 : }
1882 218675 : if (l<r) {
1883 141498 : list_append(*L, l);
1884 141498 : list_append(*R, r);
1885 : } else {
1886 77184 : list_append(*L, r);
1887 77184 : list_append(*R, l);
1888 : }
1889 : }
1890 : return 0;
1891 : }
1892 :
1893 : static list *
1894 71090 : distinct_join_exps(list *aje, list *lrels, list *rrels)
1895 : {
1896 71090 : node *n, *m, *o, *p;
1897 71090 : int len = list_length(aje), i, j;
1898 71090 : char *used = SA_ZNEW_ARRAY(aje->sa, char, len);
1899 71090 : list *res = sa_list(aje->sa);
1900 :
1901 71090 : assert(len == list_length(lrels));
1902 289772 : for(n = lrels->h, m = rrels->h, j = 0; n && m;
1903 218682 : n = n->next, m = m->next, j++) {
1904 218682 : if (n->data && m->data)
1905 948712 : for(o = n->next, p = m->next, i = j+1; o && p;
1906 730082 : o = o->next, p = p->next, i++) {
1907 730082 : if (o->data == n->data && p->data == m->data)
1908 30290 : used[i] = 1;
1909 : }
1910 : }
1911 289772 : for (i = 0, n = aje->h; i < len; n = n->next, i++) {
1912 218682 : if (!used[i])
1913 193484 : list_append(res, n->data);
1914 : }
1915 71090 : return res;
1916 : }
1917 :
1918 : static list *
1919 237512 : find_fk( mvc *sql, list *rels, list *exps)
1920 : {
1921 237512 : node *djn;
1922 237512 : list *sdje, *aje, *dje;
1923 237512 : list *lrels, *rrels;
1924 :
1925 : /* first find the distinct join expressions */
1926 237512 : aje = list_select(exps, rels, (fcmp) &exp_is_join, (fdup)NULL);
1927 : /* add left/right relation */
1928 237512 : if (find_join_rels(&lrels, &rrels, aje, rels) < 0)
1929 : dje = aje;
1930 : else
1931 71090 : dje = distinct_join_exps(aje, lrels, rrels);
1932 597044 : for(djn=dje->h; djn; djn = djn->next) {
1933 : /* equal join expressions */
1934 359625 : sql_idx *idx = NULL;
1935 359625 : sql_exp *je = djn->data, *le = je->l, *re = je->r;
1936 :
1937 359625 : if (is_complex_exp(je->flag))
1938 : break;
1939 359532 : if (!find_prop(je->p, PROP_JOINIDX)) {
1940 356032 : int swapped = 0;
1941 356032 : list *aaje = matching_joins(sql->sa, rels, aje, je);
1942 356032 : list *eje = list_select(aaje, (void*)1, (fcmp) &exp_is_eqjoin, (fdup)NULL);
1943 356032 : sql_rel *lr = find_rel(rels, le), *olr = lr;
1944 356032 : sql_rel *rr = find_rel(rels, re), *orr = rr;
1945 356032 : sql_rel *bt = NULL;
1946 356032 : char *iname;
1947 :
1948 356032 : sql_table *l, *r;
1949 356032 : list *lexps = list_map(eje, lr, (fmap) &joinexp_col);
1950 356032 : list *rexps = list_map(eje, rr, (fmap) &joinexp_col);
1951 356032 : list *lcols, *rcols;
1952 :
1953 356032 : lr = find_basetable(lr);
1954 356032 : rr = find_basetable(rr);
1955 356032 : if (!lr || !rr)
1956 232825 : continue;
1957 248588 : l = lr->l;
1958 248588 : r = rr->l;
1959 248588 : lcols = list_map(lexps, lr, (fmap) &table_colexp);
1960 248588 : rcols = list_map(rexps, rr, (fmap) &table_colexp);
1961 248588 : lcols->destroy = NULL;
1962 248588 : rcols->destroy = NULL;
1963 248588 : if (list_length(lcols) != list_length(rcols))
1964 17936 : continue;
1965 :
1966 230652 : idx = find_fk_index(sql, l, lcols, r, rcols);
1967 230652 : if (!idx) {
1968 229985 : idx = find_fk_index(sql, r, rcols, l, lcols);
1969 229985 : swapped = 1;
1970 : }
1971 :
1972 230652 : if (idx && (iname = sa_strconcat( sql->sa, "%", idx->base.name)) != NULL &&
1973 667 : ((!swapped && name_find_column(olr, NULL, iname, -2, &bt) == NULL) ||
1974 173 : ( swapped && name_find_column(orr, NULL, iname, -2, &bt) == NULL)))
1975 : idx = NULL;
1976 :
1977 : if (idx) {
1978 751 : prop *p;
1979 751 : node *n;
1980 751 : sql_exp *t = NULL, *i = NULL;
1981 :
1982 751 : if (list_length(lcols) > 1 || !mvc_debug_on(sql, 512)) {
1983 :
1984 : /* Add join between idx and TID */
1985 751 : if (swapped) {
1986 155 : sql_exp *s = je->l, *l = je->r;
1987 :
1988 155 : t = rel_find_column(sql, olr, s->l, TID);
1989 155 : i = rel_find_column(sql, orr, l->l, iname);
1990 155 : if (!t || !i)
1991 1 : continue;
1992 154 : t->p = NULL;
1993 154 : i->p = NULL;
1994 154 : je = exp_compare(sql->sa, i, t, cmp_equal);
1995 : } else {
1996 596 : sql_exp *s = je->r, *l = je->l;
1997 :
1998 596 : t = rel_find_column(sql, orr, s->l, TID);
1999 596 : i = rel_find_column(sql, olr, l->l, iname);
2000 596 : if (!t || !i)
2001 0 : continue;
2002 596 : t->p = NULL;
2003 596 : i->p = NULL;
2004 596 : je = exp_compare(sql->sa, i, t, cmp_equal);
2005 : }
2006 :
2007 : /* Remove all join expressions */
2008 1502 : for (n = eje->h; n; n = n->next)
2009 752 : list_remove_data(exps, NULL, n->data);
2010 750 : append(exps, je);
2011 750 : djn->data = je;
2012 0 : } else if (swapped) { /* else keep je for single column expressions */
2013 0 : je = exp_compare(sql->sa, je->r, je->l, cmp_equal);
2014 : /* Remove all join expressions */
2015 0 : for (n = eje->h; n; n = n->next)
2016 0 : list_remove_data(exps, NULL, n->data);
2017 0 : append(exps, je);
2018 0 : djn->data = je;
2019 : }
2020 750 : je->p = p = prop_create(sql->sa, PROP_JOINIDX, je->p);
2021 750 : p->value.pval = idx;
2022 : }
2023 : }
2024 : }
2025 :
2026 : /* sort expressions on weighted number of reducing operators */
2027 237512 : sdje = order_join_expressions(sql, dje, rels);
2028 237512 : return sdje;
2029 : }
2030 :
2031 : static int
2032 549778 : rels_find_one_rel( sql_rel **rels, int nr, sql_exp *e)
2033 : {
2034 549778 : int fnd = 0;
2035 :
2036 4412406 : for(int i = 1; i<=nr; i++) {
2037 3862694 : if (rel_has_exp(rels[i], e, false) == 0) {
2038 549537 : if (fnd)
2039 : return 0;
2040 : fnd = i;
2041 : }
2042 : }
2043 : return fnd;
2044 : }
2045 :
2046 : /* TODO move popcount and popcount64 into gdk_*.h, used in gdk_cand, strimps and here */
2047 : static inline int
2048 : popcount64(uint64_t x)
2049 : {
2050 : #ifdef __has_builtin
2051 : #if __has_builtin(__builtin_popcountll)
2052 : return (uint32_t) __builtin_popcountll(x);
2053 : #define BUILTIN_USED
2054 : #endif
2055 : #endif
2056 : #ifndef BUILTIN_USED
2057 : #if defined(_MSC_VER)
2058 : #if SIZEOF_OID == 4
2059 : /* no __popcnt64 on 32 bit Windows */
2060 : return (int) (__popcnt((uint32_t) x) + __popcnt((uint32_t) (x >> 32)));
2061 : #else
2062 : return (uint32_t) __popcnt64(x);
2063 : #endif
2064 : #else
2065 : x = (x & UINT64_C(0x5555555555555555)) + ((x >> 1) & UINT64_C(0x5555555555555555));
2066 : x = (x & UINT64_C(0x3333333333333333)) + ((x >> 2) & UINT64_C(0x3333333333333333));
2067 : x = (x & UINT64_C(0x0F0F0F0F0F0F0F0F)) + ((x >> 4) & UINT64_C(0x0F0F0F0F0F0F0F0F));
2068 : return (x * UINT64_C(0x0101010101010101)) >> 56;
2069 : #endif
2070 : #endif
2071 : #undef BUILTIN_USED
2072 : }
2073 :
2074 : static sql_rel *
2075 153246 : order_joins(visitor *v, list *rels, list *exps)
2076 : {
2077 153246 : sql_rel *top = NULL, *l = NULL, *r = NULL;
2078 153246 : sql_exp *cje;
2079 153246 : node *djn;
2080 153246 : list *sdje, *n_rels = NULL;
2081 153246 : int fnd = 0;
2082 153246 : unsigned int rsingle;
2083 153246 : int direct = 1;
2084 :
2085 : /* find foreign keys and reorder the expressions on reducing quality */
2086 153246 : sdje = find_fk(v->sql, rels, exps);
2087 :
2088 428030 : for(djn = sdje->h; djn; djn = djn->next ) {
2089 274784 : sql_exp *e = djn->data;
2090 274784 : list_remove_data(exps, NULL, e);
2091 : }
2092 153246 : if (list_length(rels) > 2 && mvc_debug_on(v->sql, 256)) {
2093 0 : top = rel_planner(v->sql, rels, sdje, exps);
2094 0 : return top;
2095 : }
2096 :
2097 153246 : int nr_exps = list_length(sdje), nr_rels = list_length(rels), ci = 1;
2098 153246 : if (nr_rels > 64) {
2099 1 : direct = 0;
2100 1 : n_rels = sa_list(v->sql->ta);
2101 : }
2102 153246 : sql_rel **rels_a = SA_NEW_ARRAY(v->sql->ta, sql_rel*, nr_rels+1); /* don't use slot 0 */
2103 153246 : rels_a[0] = NULL;
2104 580495 : for (node *n = rels->h; n; n = n->next, ci++) {
2105 427249 : rels_a[ci] = n->data;
2106 : }
2107 153246 : ulng *h = SA_NEW_ARRAY(v->sql->ta, ulng, nr_exps), rel_mask = 0; /* bit field (for > 64 its an imprint) */
2108 153246 : uint16_t *r1 = SA_NEW_ARRAY(v->sql->ta, uint16_t, nr_exps);
2109 153246 : uint16_t *r2 = SA_NEW_ARRAY(v->sql->ta, uint16_t, nr_exps);
2110 : /* change r3 into rest list's */
2111 153246 : int *r3 = SA_NEW_ARRAY(v->sql->ta, int, nr_exps);
2112 :
2113 153246 : ci = 0;
2114 428030 : for (node *n = sdje->h; n; n = n->next, ci++) {
2115 274784 : sql_exp *cje = n->data;
2116 :
2117 274784 : h[ci] = r1[ci] = r2[ci] = 0;
2118 274784 : r3[ci] = 0;
2119 : /* h[ci] = exp_find_rels(cje, rels) */
2120 274784 : if (cje->type != e_cmp || !is_complex_exp(cje->flag) || !find_prop(cje->p, PROP_HASHCOL) ||
2121 0 : (cje->type == e_cmp && cje->f == NULL)) {
2122 274784 : cje->tmp = ci;
2123 274784 : r1[ci] = rels_find_one_rel(rels_a, nr_rels, cje->l);
2124 274784 : r2[ci] = rels_find_one_rel(rels_a, nr_rels, cje->r);
2125 274784 : if (r1[ci])
2126 274650 : h[ci] |= ((ulng)1)<<((r1[ci]-1)%64);
2127 274784 : if (r2[ci])
2128 274631 : h[ci] |= ((ulng)1)<<((r2[ci]-1)%64);
2129 274784 : if (cje->f) {
2130 210 : r3[ci] = rels_find_one_rel(rels_a, nr_rels, cje->f);
2131 210 : if (r3[ci] == r2[ci])
2132 190 : r3[ci] = 0;
2133 210 : if (r3[ci])
2134 10 : h[ci] |= ((ulng)1)<<((r3[ci]-1)%64);
2135 : }
2136 : }
2137 : }
2138 : /* open problem, some expressions use more than 2 relations */
2139 : /* For example a.x = b.y * c.z; */
2140 153246 : if (list_length(rels) >= 2 && sdje->h) {
2141 306458 : for (node *n = sdje->h; n && !l && !r; n = n->next, ci++) {
2142 153234 : cje = n->data;
2143 :
2144 : /* find the involved relations */
2145 :
2146 : /* complex expressions may touch multiple base tables
2147 : * Should be pushed up to extra selection.
2148 : * */
2149 153234 : if (0 && popcount64(h[cje->tmp]) > 2)
2150 : assert(0);
2151 153234 : if (cje->type != e_cmp || !is_complex_exp(cje->flag) || !find_prop(cje->p, PROP_HASHCOL) ||
2152 0 : (cje->type == e_cmp && cje->f == NULL)) {
2153 153234 : l = rels_a[r1[cje->tmp]];
2154 153234 : r = rels_a[r2[cje->tmp]];
2155 153234 : if (l && r)
2156 153099 : rel_mask |= h[cje->tmp];
2157 : }
2158 : }
2159 153224 : cje->tmp = 0;
2160 :
2161 153224 : if (l && r && l != r)
2162 153099 : list_remove_data(sdje, NULL, cje);
2163 : }
2164 :
2165 153246 : if (l && r && l != r) {
2166 153099 : list_remove_data(rels, NULL, l);
2167 153099 : list_remove_data(rels, NULL, r);
2168 153099 : if (!direct) {
2169 1 : list_append(n_rels, l);
2170 1 : list_append(n_rels, r);
2171 : }
2172 :
2173 : /* Create a relation between l and r. Since the calling
2174 : functions rewrote the join tree, into a list of expressions
2175 : and a list of (simple) relations, there are no outer joins
2176 : involved, we can simply do a crossproduct here.
2177 : */
2178 153099 : rsingle = is_single(r);
2179 153099 : reset_single(r);
2180 153099 : top = rel_crossproduct(v->sql->sa, l, r, op_join);
2181 153099 : if (rsingle)
2182 5088 : set_single(r);
2183 153099 : rel_join_add_exp(v->sql->sa, top, cje);
2184 :
2185 : /* all other join expressions on these 2 relations */
2186 157418 : for (node *en = exps->h; en; ) {
2187 4319 : node *next = en->next;
2188 4319 : sql_exp *e = en->data;
2189 4319 : if (rel_rebind_exp(v->sql, top, e)) {
2190 4141 : rel_join_add_exp(v->sql->sa, top, e);
2191 4141 : list_remove_data(exps, NULL, e);
2192 : }
2193 : en = next;
2194 : }
2195 : fnd = 1;
2196 : }
2197 : /* build join tree using the ordered list */
2198 271477 : while(list_length(sdje) && fnd) {
2199 118231 : fnd = 0;
2200 : /* find the first expression which could be added */
2201 873967 : for(djn = sdje->h; djn && !fnd && rels->h; djn = (!fnd)?djn->next:NULL) {
2202 377868 : node *en;
2203 377868 : l = r = NULL;
2204 :
2205 377868 : cje = djn->data;
2206 377868 : if ((h[cje->tmp] & rel_mask) > 0) {
2207 118147 : if (rel_mask & (((ulng)1)<<((r1[cje->tmp]-1)%64)))
2208 66425 : l = rels_a[r1[cje->tmp]];
2209 118147 : if (rel_mask & (((ulng)1)<<((r2[cje->tmp]-1)%64)))
2210 51723 : r = rels_a[r2[cje->tmp]];
2211 : }
2212 377868 : if (!direct) { /* check if at least one side in n_rels */
2213 1040 : if (l && !list_find(n_rels, l, NULL))
2214 1007 : l = NULL;
2215 1040 : if (r && !list_find(n_rels, r, NULL))
2216 1 : r = NULL;
2217 : }
2218 :
2219 377868 : if (l && r) {
2220 0 : assert(0);
2221 : /* create a selection on the current */
2222 : rel_join_add_exp(v->sql->sa, top, cje);
2223 : fnd = 1;
2224 377868 : } else if (l || r) {
2225 : /* TODO: handle case for joins which need > 2 relations, ie where the current 'top' of the
2226 : * join tree needs to add more then one relation */
2227 118147 : rel_mask |= h[cje->tmp];
2228 118147 : if (l) {
2229 66425 : r = rels_a[r2[cje->tmp]];
2230 : } else {
2231 51722 : l = r;
2232 51722 : r = rels_a[r1[cje->tmp]];
2233 : }
2234 118147 : if (!r) {
2235 0 : fnd = 1; /* not really, but this bails out */
2236 0 : list_remove_data(sdje, NULL, cje); /* handle later as select */
2237 0 : continue;
2238 : }
2239 :
2240 : /* remove the expression from the lists */
2241 118147 : list_remove_data(sdje, NULL, cje);
2242 :
2243 118147 : list_remove_data(rels, NULL, r);
2244 118147 : if (!direct)
2245 63 : append(n_rels, r);
2246 :
2247 : /* create a join using the current expression */
2248 118147 : rsingle = is_single(r);
2249 118147 : reset_single(r);
2250 118147 : top = rel_crossproduct(v->sql->sa, top, r, op_join);
2251 118147 : if (rsingle)
2252 0 : set_single(r);
2253 118147 : rel_join_add_exp(v->sql->sa, top, cje);
2254 :
2255 : /* all join expressions on these tables */
2256 118554 : for (en = exps->h; en; ) {
2257 407 : node *next = en->next;
2258 407 : sql_exp *e = en->data;
2259 407 : if (rel_rebind_exp(v->sql, top, e)) {
2260 176 : rel_join_add_exp(v->sql->sa, top, e);
2261 176 : list_remove_data(exps, NULL, e);
2262 : }
2263 : en = next;
2264 : }
2265 : /* Remove other joins on the current 'n_rels'
2266 : set in the distinct list too */
2267 692304 : for (en = sdje->h; en; ) {
2268 574157 : node *next = en->next;
2269 574157 : sql_exp *e = en->data;
2270 574157 : if ((direct && ((e->flag <= cmp_notequal && (h[e->tmp] & rel_mask) == h[e->tmp] && h[e->tmp]) || (e->flag > cmp_notequal && rel_rebind_exp(v->sql, top, e)))) ||
2271 1953 : (!direct && rel_rebind_exp(v->sql, top, e))) {
2272 3324 : rel_join_add_exp(v->sql->sa, top, e);
2273 3324 : list_remove_data(sdje, NULL, en->data);
2274 : }
2275 : en = next;
2276 : }
2277 : fnd = 1;
2278 : }
2279 : }
2280 : }
2281 153246 : if (list_length(rels)) { /* more relations */
2282 1188 : node *n;
2283 4092 : for(n=rels->h; n; n = n->next) {
2284 2904 : sql_rel *nr = n->data;
2285 :
2286 2904 : if (top) {
2287 2757 : rsingle = is_single(nr);
2288 2757 : reset_single(nr);
2289 2757 : top = rel_crossproduct(v->sql->sa, top, nr, op_join);
2290 2757 : if (rsingle)
2291 0 : set_single(nr);
2292 : } else
2293 : top = nr;
2294 : }
2295 : }
2296 153246 : if (list_length(sdje)) {
2297 209 : if (list_empty(exps))
2298 : exps = sdje;
2299 : else
2300 0 : exps = list_merge(exps, sdje, (fdup)NULL);
2301 : }
2302 153246 : if (list_length(exps)) { /* more expressions (add selects) */
2303 233 : top = rel_select(v->sql->sa, top, NULL);
2304 471 : for(node *n=exps->h; n; n = n->next) {
2305 238 : sql_exp *e = n->data;
2306 :
2307 238 : if (exp_is_join_exp(e) == 0) {
2308 226 : sql_rel *nr = NULL;
2309 226 : if (is_theta_exp(e->flag)) {
2310 138 : nr = rel_push_join(v->sql, top->l, e->l, e->r, e->f, e, 0);
2311 88 : } else if (e->flag == cmp_filter || e->flag == cmp_or) {
2312 88 : sql_exp *l = exps_find_one_multi_exp(e->l), *r = exps_find_one_multi_exp(e->r);
2313 88 : if (l && r)
2314 82 : nr = rel_push_join(v->sql, top->l, l, r, NULL, e, 0);
2315 : }
2316 220 : if (!nr)
2317 6 : rel_join_add_exp(v->sql->sa, top->l, e);
2318 : } else
2319 12 : rel_select_add_exp(v->sql->sa, top, e);
2320 : }
2321 233 : if (list_empty(top->exps)) { /* empty select */
2322 221 : sql_rel *l = top->l;
2323 221 : top->l = NULL;
2324 221 : rel_destroy(top);
2325 221 : top = l;
2326 : }
2327 : }
2328 : return top;
2329 : }
2330 :
2331 : static int
2332 427249 : rel_neg_in_size(sql_rel *r)
2333 : {
2334 427249 : if ((is_union(r->op) /*|| is_munion(r->op)*/) && r->nrcols == 0)
2335 0 : return -1 + rel_neg_in_size(r->l);
2336 427249 : if (is_project(r->op) && r->nrcols == 0)
2337 0 : return -1;
2338 : return 0;
2339 : }
2340 :
2341 427249 : static void _rel_destroy(void *dummy, sql_rel *rel)
2342 : {
2343 427249 : (void)dummy;
2344 427249 : rel_destroy(rel);
2345 427249 : }
2346 :
2347 : static list *
2348 153246 : push_in_join_down(mvc *sql, list *rels, list *exps)
2349 : {
2350 153246 : node *n;
2351 153246 : int restart = 1;
2352 153246 : list *nrels;
2353 :
2354 : /* we should sort these first, ie small in's before large one's */
2355 153246 : nrels = list_sort(rels, (fkeyvalue)&rel_neg_in_size, (fdup)&rel_dup);
2356 :
2357 : /* we need to cleanup, the new refs ! */
2358 153246 : rels->destroy = (fdestroy)_rel_destroy;
2359 153246 : list_destroy(rels);
2360 153246 : rels = nrels;
2361 :
2362 : /* one of the rels should be a op_union with nrcols == 0 */
2363 459738 : while (restart) {
2364 580495 : for (n = rels->h; n; n = n->next) {
2365 427249 : sql_rel *r = n->data;
2366 :
2367 427249 : restart = 0;
2368 427249 : if (is_project(r->op) && r->nrcols == 0) {
2369 : /* next step find expression on this relation */
2370 0 : node *m;
2371 0 : sql_rel *l = NULL;
2372 0 : sql_exp *je = NULL;
2373 :
2374 0 : for(m = exps->h; !je && m; m = m->next) {
2375 0 : sql_exp *e = m->data;
2376 :
2377 0 : if (e->type == e_cmp && e->flag == cmp_equal) {
2378 : /* in values are on
2379 : the right of the join */
2380 0 : if (rel_has_exp(r, e->r, false) >= 0)
2381 0 : je = e;
2382 : }
2383 : }
2384 : /* with this expression find other relation */
2385 0 : if (je && (l = find_rel(rels, je->l)) != NULL) {
2386 0 : unsigned int rsingle = is_single(r);
2387 0 : reset_single(r);
2388 0 : sql_rel *nr = rel_crossproduct(sql->sa, l, r, op_join);
2389 0 : if (rsingle)
2390 0 : set_single(r);
2391 0 : rel_join_add_exp(sql->sa, nr, je);
2392 0 : list_append(rels, nr);
2393 0 : list_remove_data(rels, NULL, l);
2394 0 : list_remove_data(rels, NULL, r);
2395 0 : list_remove_data(exps, NULL, je);
2396 0 : restart = 1;
2397 0 : break;
2398 : }
2399 :
2400 : }
2401 : }
2402 : }
2403 153246 : return rels;
2404 : }
2405 :
2406 : static list *
2407 729808 : push_up_join_exps( mvc *sql, sql_rel *rel)
2408 : {
2409 729808 : if (rel_is_ref(rel))
2410 : return NULL;
2411 :
2412 689396 : switch(rel->op) {
2413 286274 : case op_join: {
2414 286274 : sql_rel *rl = rel->l;
2415 286274 : sql_rel *rr = rel->r;
2416 286274 : list *l, *r;
2417 :
2418 286274 : if (rel_is_ref(rl) && rel_is_ref(rr)) {
2419 16 : l = rel->exps;
2420 16 : rel->exps = NULL;
2421 16 : return l;
2422 : }
2423 286258 : l = push_up_join_exps(sql, rl);
2424 286258 : r = push_up_join_exps(sql, rr);
2425 286258 : if (l && r) {
2426 34 : l = list_merge(l, r, (fdup)NULL);
2427 34 : r = NULL;
2428 286224 : } else if (!l) {
2429 174121 : l = r;
2430 174121 : r = NULL;
2431 : }
2432 286258 : if (rel->exps) {
2433 254823 : if (l && !r)
2434 101521 : r = l;
2435 254823 : l = list_merge(rel->exps, r, (fdup)NULL);
2436 : }
2437 286258 : rel->exps = NULL;
2438 286258 : return l;
2439 : }
2440 : default:
2441 : return NULL;
2442 : }
2443 : }
2444 :
2445 : static sql_rel *
2446 284433 : reorder_join(visitor *v, sql_rel *rel)
2447 : {
2448 284433 : list *exps, *rels;
2449 :
2450 284433 : if (is_innerjoin(rel->op) && !is_single(rel) && !rel_is_ref(rel) && list_empty(rel->attr)) {
2451 174550 : if (list_empty(rel->exps)) {
2452 22113 : sql_rel *l = rel->l, *r = rel->r;
2453 22113 : if (!is_innerjoin(l->op) && !is_innerjoin(r->op))
2454 : return rel;
2455 : }
2456 157292 : rel->exps = push_up_join_exps(v->sql, rel);
2457 : }
2458 :
2459 267175 : if (!is_innerjoin(rel->op) || is_single(rel) || rel_is_ref(rel) || list_empty(rel->exps) || !list_empty(rel->attr)) {
2460 113929 : if (!list_empty(rel->exps)) { /* cannot add join idxs to cross products */
2461 78016 : exps = rel->exps;
2462 78016 : rel->exps = NULL; /* should be all crosstables by now */
2463 78016 : rels = sa_list(v->sql->ta);
2464 : /* try to use an join index also for outer joins */
2465 78016 : get_inner_relations(v->sql, rel, rels);
2466 78016 : int cnt = list_length(exps);
2467 78016 : rel->exps = find_fk(v->sql, rels, exps);
2468 78016 : if (list_length(rel->exps) != cnt)
2469 16444 : rel->exps = order_join_expressions(v->sql, exps, rels);
2470 : }
2471 113929 : rel->l = rel_join_order_(v, rel->l);
2472 113929 : rel->r = rel_join_order_(v, rel->r);
2473 : } else {
2474 153246 : exps = rel->exps;
2475 153246 : rel->exps = NULL; /* should be all crosstables by now */
2476 153246 : rels = sa_list(v->sql->ta);
2477 153246 : get_relations(v, rel, rels);
2478 153246 : if (list_length(rels) > 1) {
2479 153246 : rels = push_in_join_down(v->sql, rels, exps);
2480 153246 : rel = order_joins(v, rels, exps);
2481 : } else {
2482 0 : rel->exps = exps;
2483 : }
2484 : }
2485 : return rel;
2486 : }
2487 :
2488 : static sql_rel *
2489 2399978 : rel_join_order_(visitor *v, sql_rel *rel)
2490 : {
2491 2399978 : if (!rel)
2492 : return rel;
2493 :
2494 2384178 : switch (rel->op) {
2495 : case op_basetable:
2496 : break;
2497 3851 : case op_table:
2498 3851 : if (IS_TABLE_PROD_FUNC(rel->flag) || rel->flag == TABLE_FROM_RELATION)
2499 3851 : rel->l = rel_join_order_(v, rel->l);
2500 : break;
2501 : case op_join:
2502 : case op_left:
2503 : case op_right:
2504 : case op_full:
2505 : break;
2506 :
2507 11415 : case op_semi:
2508 : case op_anti:
2509 :
2510 : case op_union:
2511 : case op_inter:
2512 : case op_except:
2513 : case op_merge:
2514 11415 : rel->l = rel_join_order_(v, rel->l);
2515 11415 : rel->r = rel_join_order_(v, rel->r);
2516 11415 : break;
2517 115473 : case op_munion:
2518 115473 : assert(rel->l);
2519 466254 : for (node *n = ((list*)rel->l)->h; n; n = n->next)
2520 350781 : n->data = rel_join_order_(v, n->data);
2521 : break;
2522 1273547 : case op_project:
2523 : case op_select:
2524 : case op_groupby:
2525 : case op_topn:
2526 : case op_sample:
2527 1273547 : rel->l = rel_join_order_(v, rel->l);
2528 1273547 : break;
2529 52 : case op_ddl:
2530 52 : if (rel->flag == ddl_output || rel->flag == ddl_create_seq || rel->flag == ddl_alter_seq || rel->flag == ddl_alter_table || rel->flag == ddl_create_table || rel->flag == ddl_create_view) {
2531 0 : rel->l = rel_join_order_(v, rel->l);
2532 : } else if (rel->flag == ddl_list || rel->flag == ddl_exception) {
2533 52 : rel->l = rel_join_order_(v, rel->l);
2534 52 : rel->r = rel_join_order_(v, rel->r);
2535 : }
2536 : break;
2537 11205 : case op_insert:
2538 : case op_update:
2539 : case op_delete:
2540 11205 : rel->r = rel_join_order_(v, rel->r);
2541 11205 : break;
2542 : case op_truncate:
2543 : break;
2544 : }
2545 2384178 : if (is_join(rel->op))
2546 284433 : rel = reorder_join(v, rel);
2547 : return rel;
2548 : }
2549 :
2550 : static sql_rel *
2551 82553 : rel_join_order(visitor *v, global_props *gp, sql_rel *rel)
2552 : {
2553 82553 : (void) gp;
2554 82553 : sql_rel *r = rel_join_order_(v, rel);
2555 82553 : sa_reset(v->sql->ta);
2556 82553 : return r;
2557 : }
2558 :
2559 : run_optimizer
2560 623246 : bind_join_order(visitor *v, global_props *gp)
2561 : {
2562 623246 : int flag = v->sql->sql_optimizer;
2563 623050 : return gp->opt_level == 1 && gp->opt_cycle < 10 && !gp->cnt[op_update] && (gp->cnt[op_join] || gp->cnt[op_left] ||
2564 1240247 : gp->cnt[op_right] || gp->cnt[op_full]) && (flag & join_order) ? rel_join_order : NULL;
2565 : }
2566 :
2567 : /* this join order is to be done once after statistics are gathered */
2568 : run_optimizer
2569 705008 : bind_join_order2(visitor *v, global_props *gp)
2570 : {
2571 : /*int flag = v->sql->sql_optimizer;
2572 : return gp->opt_level == 1 && !gp->has_special_modify && !gp->cnt[op_update] && (gp->cnt[op_join] || gp->cnt[op_left] ||
2573 : gp->cnt[op_right] || gp->cnt[op_full]) && (flag & join_order) ? rel_join_order : NULL;*/
2574 : /* TODO we have to propagate count statistics here */
2575 705008 : (void) v;
2576 705008 : (void) gp;
2577 705008 : return NULL;
2578 : }
2579 :
2580 :
2581 : static int
2582 13683 : is_identity_of(sql_exp *e, sql_rel *l)
2583 : {
2584 13683 : if (e->type != e_cmp)
2585 : return 0;
2586 13656 : if (!is_identity(e->l, l) || !is_identity(e->r, l) || (e->f && !is_identity(e->f, l)))
2587 13656 : return 0;
2588 : return 1;
2589 : }
2590 :
2591 : static inline sql_rel *
2592 17863 : rel_rewrite_semijoin(visitor *v, sql_rel *rel)
2593 : {
2594 17863 : assert(is_semi(rel->op));
2595 : {
2596 17863 : sql_rel *l = rel->l;
2597 17863 : sql_rel *r = rel->r;
2598 17863 : sql_rel *rl = (r->l)?r->l:NULL;
2599 17863 : int on_identity = 1;
2600 :
2601 17863 : if (!rel->exps || list_length(rel->exps) != 1 || !is_identity_of(rel->exps->h->data, l))
2602 : on_identity = 0;
2603 :
2604 : /* rewrite {semi,anti}join (A, join(A,B)) into {semi,anti}join (A,B)
2605 : * and {semi,anti}join (A, join(B,A)) into {semi,anti}join (A,B)
2606 : * Where the semi/anti join is done using the identity */
2607 0 : if (on_identity && l->ref.refcnt == 2 && ((is_join(r->op) && (l == r->l || l == r->r)) ||
2608 0 : (is_project(r->op) && rl && is_join(rl->op) && (l == rl->l || l == rl->r)))){
2609 0 : sql_rel *or = r;
2610 :
2611 0 : if (is_project(r->op))
2612 0 : r = rl;
2613 :
2614 0 : if (l == r->r)
2615 0 : rel->r = rel_dup(r->l);
2616 : else
2617 0 : rel->r = rel_dup(r->r);
2618 :
2619 0 : rel->exps = r->exps;
2620 0 : r->exps = NULL;
2621 0 : rel->attr = r->attr;
2622 0 : r->attr = NULL;
2623 0 : rel_destroy(or);
2624 0 : v->changes++;
2625 : }
2626 : }
2627 : {
2628 17863 : sql_rel *l = rel->l, *rl = NULL;
2629 17863 : sql_rel *r = rel->r, *or = r;
2630 :
2631 17863 : if (r)
2632 17863 : rl = r->l;
2633 17863 : if (r && is_project(r->op)) {
2634 14759 : r = rl;
2635 14759 : if (r)
2636 14542 : rl = r->l;
2637 : }
2638 :
2639 : /* More general case is (join reduction)
2640 : {semi,anti}join (A, join(A,B) [A.c1 == B.c1]) [ A.c1 == B.c1 ]
2641 : into {semi,anti}join (A,B) [ A.c1 == B.c1 ]
2642 :
2643 : for semijoin also A.c1 == B.k1 ] [ A.c1 == B.k2 ] could be rewritten
2644 : */
2645 17863 : if (l && r && rl &&
2646 16540 : is_basetable(l->op) && is_basetable(rl->op) &&
2647 3986 : is_join(r->op) && l->l == rl->l)
2648 : {
2649 26 : node *n, *m;
2650 26 : list *exps;
2651 :
2652 51 : if (!rel->exps || !r->exps ||
2653 25 : list_length(rel->exps) != list_length(r->exps))
2654 1 : return rel;
2655 25 : exps = new_exp_list(v->sql->sa);
2656 :
2657 : /* are the join conditions equal */
2658 25 : for (n = rel->exps->h, m = r->exps->h;
2659 25 : n && m; n = n->next, m = m->next)
2660 : {
2661 25 : sql_exp *le = NULL, *oe = n->data;
2662 25 : sql_exp *re = NULL, *ne = m->data;
2663 25 : sql_column *cl;
2664 25 : int equal = 0;
2665 :
2666 25 : if (oe->type != e_cmp || ne->type != e_cmp ||
2667 25 : oe->flag != cmp_equal ||
2668 25 : ne->flag != cmp_equal || is_anti(oe) || is_anti(ne))
2669 : return rel;
2670 :
2671 25 : if ((cl = exp_find_column(rel->l, oe->l, -2)) != NULL) {
2672 25 : le = oe->l;
2673 25 : re = oe->r;
2674 0 : } else if ((cl = exp_find_column(rel->l, oe->r, -2)) != NULL) {
2675 0 : le = oe->r;
2676 0 : re = oe->l;
2677 : } else
2678 : return rel;
2679 :
2680 25 : if (exp_find_column(rl, ne->l, -2) == cl) {
2681 25 : sql_exp *e = (or != r)?rel_find_exp(or, re):re;
2682 :
2683 25 : if (e)
2684 24 : equal = exp_match_exp(ne->r, e);
2685 25 : if (!e || !equal)
2686 : return rel;
2687 0 : re = ne->r;
2688 0 : } else if (exp_find_column(rl, ne->r, -2) == cl) {
2689 0 : sql_exp *e = (or != r)?rel_find_exp(or, re):re;
2690 :
2691 0 : if (e)
2692 0 : equal = exp_match_exp(ne->l, e);
2693 0 : if (!e || !equal)
2694 : return rel;
2695 0 : re = ne->l;
2696 : } else
2697 : return rel;
2698 :
2699 0 : ne = exp_compare(v->sql->sa, le, re, cmp_equal);
2700 0 : append(exps, ne);
2701 : }
2702 :
2703 0 : rel->r = rel_dup(r->r);
2704 0 : rel->exps = exps;
2705 0 : rel_destroy(or);
2706 0 : v->changes++;
2707 : }
2708 : }
2709 : return rel;
2710 : }
2711 :
2712 : /*
2713 : * Push semijoins down, pushes the semijoin through a join.
2714 : *
2715 : * semijoin( join(A, B) [ A.x == B.y ], C ) [ A.z == C.c ]
2716 : * ->
2717 : * join( semijoin(A, C) [ A.z == C.c ], B ) [ A.x == B.y ]
2718 : *
2719 : * also push simple expressions of a semijoin down if they only
2720 : * involve the left sided of the semijoin.
2721 : *
2722 : * in some cases the other way is useful, ie push join down
2723 : * semijoin. When the join reduces (ie when there are selects on it).
2724 : *
2725 : * At the moment, we only flag changes by this optimizer on the first level of optimization
2726 : */
2727 : static inline sql_rel *
2728 461816 : rel_push_semijoin_down_or_up(visitor *v, sql_rel *rel)
2729 : {
2730 461816 : uint8_t cycle = *(uint8_t*) v->data;
2731 :
2732 461816 : if (rel->op == op_join && rel->exps && rel->l) {
2733 425930 : sql_rel *l = rel->l, *r = rel->r;
2734 :
2735 425930 : if (is_semi(l->op) && !rel_is_ref(l) && is_select(r->op) && !rel_is_ref(r)) {
2736 22 : rel->l = l->l;
2737 22 : l->l = rel;
2738 22 : if (cycle <= 0)
2739 0 : v->changes++;
2740 22 : return l;
2741 : }
2742 : }
2743 : /* also case with 2 joins */
2744 : /* join ( join ( semijoin(), table), select (table)); */
2745 461794 : if (rel->op == op_join && rel->exps && rel->l) {
2746 425908 : sql_rel *l = rel->l, *r = rel->r;
2747 425908 : sql_rel *ll;
2748 :
2749 425908 : if (is_join(l->op) && !rel_is_ref(l) && is_select(r->op) && !rel_is_ref(r)) {
2750 8095 : ll = l->l;
2751 8095 : if (is_semi(ll->op) && !rel_is_ref(ll)) {
2752 51 : l->l = ll->l;
2753 51 : ll->l = rel;
2754 51 : if (cycle <= 0)
2755 12 : v->changes++;
2756 51 : return ll;
2757 : }
2758 : }
2759 : }
2760 : /* first push down the expressions involving only A */
2761 461743 : if (rel->op == op_semi && rel->exps && rel->l) {
2762 12119 : sql_rel *jl = rel->l, *ojl = jl;
2763 :
2764 12119 : set_processed(jl);
2765 27731 : for (node *n = rel->exps->h; n;) {
2766 15612 : node *next = n->next;
2767 15612 : sql_exp *e = n->data;
2768 :
2769 15612 : if (n != rel->exps->h && e->type == e_cmp && rel_rebind_exp(v->sql, jl, e)) {
2770 16 : if (!is_select(jl->op) || rel_is_ref(jl))
2771 14 : rel->l = jl = rel_select(v->sql->sa, jl, NULL);
2772 16 : rel_select_add_exp(v->sql->sa, jl, e);
2773 16 : list_remove_node(rel->exps, NULL, n);
2774 16 : if (cycle <= 0)
2775 16 : v->changes++;
2776 : }
2777 : n = next;
2778 : }
2779 12119 : if (ojl != jl)
2780 14 : set_processed(jl);
2781 : }
2782 461743 : if (rel->op == op_semi && rel->exps && rel->l) {
2783 12119 : operator_type op = rel->op, lop;
2784 12119 : node *n;
2785 12119 : sql_rel *l = rel->l, *ll = NULL, *lr = NULL;
2786 12119 : sql_rel *r = rel->r;
2787 12119 : list *exps = rel->exps, *nsexps, *njexps, *nsattr, *njattr;
2788 12119 : int left = 1, right = 1;
2789 :
2790 : /* handle project
2791 : if (l->op == op_project && !need_distinct(l))
2792 : l = l->l;
2793 : */
2794 :
2795 12119 : if (!is_join(l->op) || is_full(l->op) || rel_is_ref(l) || is_single(l))
2796 : return rel;
2797 :
2798 1170 : lop = l->op;
2799 1170 : ll = l->l;
2800 1170 : lr = l->r;
2801 :
2802 : /* check which side is used and other exps are atoms or from right of semijoin */
2803 2250 : for(n = exps->h; n; n = n->next) {
2804 1274 : sql_exp *sje = n->data;
2805 :
2806 1274 : if (sje->type != e_cmp || is_complex_exp(sje->flag))
2807 : return rel;
2808 : /* sje->l from ll and sje->r/f from semijoin r ||
2809 : * sje->l from semijoin r and sje->r/f from ll ||
2810 : * sje->l from lr and sje->r/f from semijoin r ||
2811 : * sje->l from semijoin r and sje->r/f from lr */
2812 2484 : if (left &&
2813 2484 : ((rel_rebind_exp(v->sql, ll, sje->l) && rel_rebind_exp(v->sql, rel->r, sje->r) && (!sje->f || rel_rebind_exp(v->sql, rel->r, sje->f))) ||
2814 1233 : (rel_rebind_exp(v->sql, rel->r, sje->l) && rel_rebind_exp(v->sql, ll, sje->r) && (!sje->f || rel_rebind_exp(v->sql, ll, sje->f)))))
2815 : right = 0;
2816 : else
2817 906 : left = 0;
2818 1710 : if (right &&
2819 1608 : ((rel_rebind_exp(v->sql, lr, sje->l) && rel_rebind_exp(v->sql, rel->r, sje->r) && (!sje->f || rel_rebind_exp(v->sql, rel->r, sje->f))) ||
2820 566 : (rel_rebind_exp(v->sql, rel->r, sje->l) && rel_rebind_exp(v->sql, lr, sje->r) && (!sje->f || rel_rebind_exp(v->sql, lr, sje->f)))))
2821 : left = 0;
2822 : else
2823 : right = 0;
2824 1242 : if (!right && !left)
2825 : return rel;
2826 : }
2827 976 : if (left && is_right(lop))
2828 : return rel;
2829 975 : if (right && is_left(lop))
2830 : return rel;
2831 973 : nsexps = exps_copy(v->sql, rel->exps);
2832 973 : nsattr = exps_copy(v->sql, rel->attr);
2833 973 : njexps = exps_copy(v->sql, l->exps);
2834 973 : njattr = exps_copy(v->sql, l->attr);
2835 973 : if (left)
2836 231 : l = rel_crossproduct(v->sql->sa, rel_dup(ll), rel_dup(r), op);
2837 : else
2838 742 : l = rel_crossproduct(v->sql->sa, rel_dup(lr), rel_dup(r), op);
2839 973 : l->exps = nsexps;
2840 973 : l->attr = nsattr;
2841 973 : set_processed(l);
2842 973 : if (left)
2843 231 : l = rel_crossproduct(v->sql->sa, l, rel_dup(lr), lop);
2844 : else
2845 742 : l = rel_crossproduct(v->sql->sa, rel_dup(ll), l, lop);
2846 973 : l->exps = njexps;
2847 973 : l->attr = njattr;
2848 973 : set_processed(l);
2849 973 : rel_destroy(rel);
2850 973 : rel = l;
2851 973 : if (cycle <= 0)
2852 570 : v->changes++;
2853 : }
2854 : return rel;
2855 : }
2856 :
2857 : /* antijoin(a, union(b,c)) -> antijoin(antijoin(a,b), c) */
2858 : static inline sql_rel *
2859 5690 : rel_rewrite_antijoin(visitor *v, sql_rel *rel)
2860 : {
2861 5690 : sql_rel *l = rel->l;
2862 5690 : sql_rel *r = rel->r;
2863 :
2864 5690 : assert(rel->op == op_anti);
2865 5690 : if (l && !rel_is_ref(l) && r && !rel_is_ref(r) && is_union(r->op) && !is_single(r)) {
2866 0 : sql_rel *rl = rel_dup(r->l), *nl;
2867 0 : sql_rel *rr = rel_dup(r->r);
2868 :
2869 0 : if (!is_project(rl->op))
2870 0 : rl = rel_project(v->sql->sa, rl,
2871 : rel_projections(v->sql, rl, NULL, 1, 1));
2872 0 : if (!is_project(rr->op))
2873 0 : rr = rel_project(v->sql->sa, rr,
2874 : rel_projections(v->sql, rr, NULL, 1, 1));
2875 0 : rel_rename_exps(v->sql, r->exps, rl->exps);
2876 0 : rel_rename_exps(v->sql, r->exps, rr->exps);
2877 :
2878 0 : nl = rel_crossproduct(v->sql->sa, rel->l, rl, op_anti);
2879 0 : nl->exps = exps_copy(v->sql, rel->exps);
2880 0 : nl->attr = exps_copy(v->sql, rel->attr);
2881 0 : set_processed(nl);
2882 0 : rel->l = nl;
2883 0 : rel->r = rr;
2884 0 : rel_destroy(r);
2885 0 : v->changes++;
2886 0 : return rel;
2887 : }
2888 : return rel;
2889 : }
2890 :
2891 : static sql_rel *
2892 3429868 : rel_optimize_semi_and_anti_(visitor *v, sql_rel *rel)
2893 : {
2894 : /* rewrite semijoin (A, join(A,B)) into semijoin (A,B) */
2895 3429868 : if (rel && is_semi(rel->op))
2896 17863 : rel = rel_rewrite_semijoin(v, rel);
2897 : /* push semijoin through join */
2898 3429868 : if (rel && (is_semi(rel->op) || is_innerjoin(rel->op)))
2899 461816 : rel = rel_push_semijoin_down_or_up(v, rel);
2900 : /* antijoin(a, union(b,c)) -> antijoin(antijoin(a,b), c) */
2901 3429868 : if (rel && rel->op == op_anti)
2902 5690 : rel = rel_rewrite_antijoin(v, rel);
2903 3429868 : return rel;
2904 : }
2905 :
2906 : static sql_rel *
2907 89528 : rel_optimize_semi_and_anti(visitor *v, global_props *gp, sql_rel *rel)
2908 : {
2909 89528 : v->data = &gp->opt_cycle;
2910 89528 : rel = rel_visitor_bottomup(v, rel, &rel_optimize_semi_and_anti_);
2911 89528 : v->data = gp;
2912 89528 : return rel;
2913 : }
2914 :
2915 : run_optimizer
2916 623343 : bind_optimize_semi_and_anti(visitor *v, global_props *gp)
2917 : {
2918 : /* Important -> Re-write semijoins after rel_join_order */
2919 623343 : int flag = v->sql->sql_optimizer;
2920 623134 : return gp->opt_level == 1 && (gp->cnt[op_join] || gp->cnt[op_left] || gp->cnt[op_right]
2921 1246477 : || gp->cnt[op_full] || gp->cnt[op_semi] || gp->cnt[op_anti]) && (flag & optimize_semi_and_anti) ? rel_optimize_semi_and_anti : NULL;
2922 : }
2923 :
2924 :
2925 : static sql_rel *
2926 1586870 : rel_semijoin_use_fk(visitor *v, sql_rel *rel)
2927 : {
2928 1586870 : if (is_semi(rel->op) && rel->exps) {
2929 6250 : list *exps = rel->exps;
2930 6250 : list *rels = sa_list(v->sql->sa);
2931 :
2932 6250 : rel->exps = NULL;
2933 6250 : append(rels, rel->l);
2934 6250 : append(rels, rel->r);
2935 6250 : (void) find_fk( v->sql, rels, exps);
2936 :
2937 6250 : rel->exps = exps;
2938 : }
2939 1586870 : return rel;
2940 : }
2941 :
2942 : /*
2943 : * Push {semi}joins down, pushes the joins through group by expressions.
2944 : * When the join is on the group by columns, we can push the joins left
2945 : * under the group by. This should only be done, iff the new semijoin would
2946 : * reduce the input table to the groupby. So there should be a reduction
2947 : * (selection) on the table A and this should be propagated to the groupby via
2948 : * for example a primary key.
2949 : *
2950 : * {semi}join( A, groupby( B ) [gbe][aggrs] ) [ gbe == A.x ]
2951 : * ->
2952 : * {semi}join( A, groupby( semijoin(B,A) [gbe == A.x] ) [gbe][aggrs] ) [ gbe == A.x ]
2953 : */
2954 : static inline sql_rel *
2955 1586870 : rel_push_join_down(visitor *v, sql_rel *rel)
2956 : {
2957 1586870 : if (!rel_is_ref(rel) && ((is_left(rel->op) || rel->op == op_join || is_semi(rel->op)) && rel->l && rel->exps)) {
2958 231962 : sql_rel *gb = rel->r, *ogb = gb, *l = NULL, *rell = rel->l;
2959 :
2960 231962 : if (is_simple_project(gb->op) && !rel_is_ref(gb))
2961 45558 : gb = gb->l;
2962 :
2963 231962 : if (rel_is_ref(rell) || !gb || rel_is_ref(gb))
2964 : return rel;
2965 :
2966 222168 : if (is_groupby(gb->op) && gb->r && list_length(gb->r)) {
2967 190 : list *exps = rel->exps, *jes = new_exp_list(v->sql->sa), *gbes = gb->r;
2968 190 : node *n, *m;
2969 : /* find out if all group by expressions are used in the join */
2970 195 : for(n = gbes->h; n; n = n->next) {
2971 190 : sql_exp *gbe = n->data;
2972 190 : int fnd = 0;
2973 190 : const char *rname = NULL, *name = NULL;
2974 :
2975 : /* project in between, ie find alias */
2976 : /* first find expression in expression list */
2977 190 : gbe = exps_uses_exp( gb->exps, gbe);
2978 190 : if (!gbe)
2979 0 : continue;
2980 190 : if (ogb != gb)
2981 167 : gbe = exps_uses_exp( ogb->exps, gbe);
2982 190 : if (gbe) {
2983 149 : rname = exp_find_rel_name(gbe);
2984 149 : name = exp_name(gbe);
2985 : }
2986 :
2987 149 : if (!name)
2988 41 : return rel;
2989 :
2990 370 : for (m = exps->h; m && !fnd; m = m->next) {
2991 221 : sql_exp *je = m->data;
2992 :
2993 221 : if (je->card >= CARD_ATOM && je->type == e_cmp &&
2994 221 : !is_complex_exp(je->flag)) {
2995 : /* expect right expression to match */
2996 221 : sql_exp *r = je->r;
2997 :
2998 221 : if (r == 0 || r->type != e_column)
2999 10 : continue;
3000 211 : if (r->l && rname && strcmp(r->l, rname) == 0 && strcmp(r->r, name)==0) {
3001 : fnd = 1;
3002 83 : } else if (!r->l && !rname && strcmp(r->r, name)==0) {
3003 : fnd = 1;
3004 : }
3005 : if (fnd) {
3006 128 : sql_exp *le = je->l;
3007 128 : sql_exp *re = exp_push_down_prj(v->sql, r, gb, gb->l);
3008 128 : if (!re || (list_length(jes) == 0 && !find_prop(le->p, PROP_HASHCOL))) {
3009 : fnd = 0;
3010 : } else {
3011 5 : int anti = is_anti(je), semantics = is_semantics(je);
3012 :
3013 5 : je = exp_compare(v->sql->sa, le, re, je->flag);
3014 5 : if (anti) set_anti(je);
3015 5 : if (semantics) set_semantics(je);
3016 5 : list_append(jes, je);
3017 : }
3018 : }
3019 : }
3020 : }
3021 149 : if (!fnd)
3022 : return rel;
3023 : }
3024 5 : l = rel_dup(rel->l);
3025 :
3026 : /* push join's left side (as semijoin) down group by */
3027 5 : l = gb->l = rel_crossproduct(v->sql->sa, gb->l, l, op_semi);
3028 5 : l->exps = jes;
3029 5 : set_processed(l);
3030 5 : v->changes++;
3031 5 : return rel;
3032 : }
3033 : }
3034 : return rel;
3035 : }
3036 :
3037 : static bool
3038 728 : check_projection_on_foreignside(sql_rel *r, list *pexps, int fk_left)
3039 : {
3040 : /* projection columns from the foreign side */
3041 728 : if (list_empty(pexps))
3042 : return true;
3043 3156 : for (node *n = pexps->h; n; n = n->next) {
3044 3091 : sql_exp *pe = n->data;
3045 :
3046 3091 : if (pe && is_atom(pe->type))
3047 23 : continue;
3048 3068 : if (pe && !is_alias(pe->type))
3049 : return false;
3050 : /* check for columns from the pk side, then keep the join with the pk */
3051 3047 : if ((fk_left && rel_find_exp(r->r, pe)) || (!fk_left && rel_find_exp(r->l, pe)))
3052 594 : return false;
3053 : }
3054 : return true;
3055 : }
3056 :
3057 : static sql_rel *
3058 237704 : rel_simplify_project_fk_join(mvc *sql, sql_rel *r, list *pexps, list *orderexps, int *changes)
3059 : {
3060 237704 : sql_rel *rl = r->l, *rr = r->r, *nr = NULL;
3061 237704 : sql_exp *je, *le, *nje, *re;
3062 237704 : int fk_left = 1;
3063 :
3064 : /* check for foreign key join */
3065 237704 : if (list_length(r->exps) != 1 || !list_empty(r->attr))
3066 7433 : return r;
3067 230271 : if (!(je = exps_find_prop(r->exps, PROP_JOINIDX)) || je->flag != cmp_equal)
3068 : return r;
3069 : /* je->l == foreign expression, je->r == primary expression */
3070 1204 : if (rel_find_exp(r->l, je->l)) {
3071 : fk_left = 1;
3072 34 : } else if (rel_find_exp(r->r, je->l)) {
3073 : fk_left = 0;
3074 : } else { /* not found */
3075 : return r;
3076 : }
3077 :
3078 : /* primary side must be a full table */
3079 1170 : if ((fk_left && (!is_left(r->op) && !is_full(r->op)) && !is_basetable(rr->op)) ||
3080 34 : (!fk_left && (!is_right(r->op) && !is_full(r->op)) && !is_basetable(rl->op)))
3081 554 : return r;
3082 :
3083 650 : if (!check_projection_on_foreignside(r, pexps, fk_left) || !check_projection_on_foreignside(r, orderexps, fk_left))
3084 613 : return r;
3085 :
3086 : /* rewrite, ie remove pkey side if possible */
3087 37 : le = (sql_exp*)je->l, re = (sql_exp*)je->l;
3088 :
3089 : /* both have NULL and there are semantics, the join cannot be removed */
3090 37 : if (is_semantics(je) && has_nil(le) && has_nil(re))
3091 : return r;
3092 :
3093 37 : (*changes)++;
3094 : /* if the foreign key column doesn't have NULL values, then return it */
3095 37 : if (!has_nil(le) || is_full(r->op) || (fk_left && is_left(r->op)) || (!fk_left && is_right(r->op))) {
3096 : /* if ->attr, introduce group by on index */
3097 31 : if (fk_left) {
3098 23 : nr = rel_dup(r->l);
3099 : } else {
3100 8 : nr = rel_dup(r->r);
3101 : }
3102 31 : if (!list_empty(r->attr)) {
3103 0 : nr = rel_groupby(sql, nr, NULL);
3104 0 : if (nr) {
3105 : // printf("# introduced groupby \n");
3106 0 : nr->r = append(sa_list(sql->sa), le);
3107 0 : nr->exps = r->attr;
3108 : }
3109 : }
3110 31 : return nr;
3111 : }
3112 :
3113 : /* remove NULL values, ie generate a select not null */
3114 6 : nje = exp_compare(sql->sa, exp_ref(sql, le), exp_atom(sql->sa, atom_general(sql->sa, exp_subtype(le), NULL, 0)), cmp_equal);
3115 6 : set_anti(nje);
3116 6 : set_has_no_nil(nje);
3117 6 : set_semantics(nje);
3118 6 : if (fk_left) {
3119 6 : nr = rel_dup(r->l);
3120 : } else {
3121 0 : nr = rel_dup(r->r);
3122 : }
3123 6 : nr = rel_select(sql->sa, nr, nje);
3124 6 : set_processed(nr);
3125 6 : return nr;
3126 : }
3127 :
3128 : static sql_rel *
3129 1853 : rel_simplify_count_fk_join(mvc *sql, sql_rel *r, list *gexps, list *gcols, int *changes)
3130 : {
3131 1853 : sql_rel *rl = r->l, *rr = r->r, *nr = NULL;
3132 1853 : sql_exp *je, *le, *nje, *re, *oce;
3133 1853 : int fk_left = 1;
3134 :
3135 : /* check for foreign key join */
3136 1853 : if (list_length(r->exps) != 1)
3137 : return r;
3138 1852 : if (!(je = exps_find_prop(r->exps, PROP_JOINIDX)) || je->flag != cmp_equal)
3139 : return r;
3140 : /* je->l == foreign expression, je->r == primary expression */
3141 63 : if (rel_find_exp(r->l, je->l)) {
3142 : fk_left = 1;
3143 7 : } else if (rel_find_exp(r->r, je->l)) {
3144 : fk_left = 0;
3145 : } else { /* not found */
3146 : return r;
3147 : }
3148 :
3149 63 : oce = gexps->h->data;
3150 63 : if (oce->l) /* we only handle COUNT(*) */
3151 : return r;
3152 :
3153 : /* primary side must be a full table */
3154 62 : if ((fk_left && (!is_left(r->op) && !is_full(r->op)) && !is_basetable(rr->op)) ||
3155 6 : (!fk_left && (!is_right(r->op) && !is_full(r->op)) && !is_basetable(rl->op)))
3156 : return r;
3157 :
3158 33 : if (fk_left && is_join(rl->op) && !rel_is_ref(rl)) {
3159 12 : r->l = rel_simplify_count_fk_join(sql, rl, gexps, gcols, changes);
3160 12 : if (rl != r->l)
3161 9 : rel_destroy(rl);
3162 : }
3163 39 : if (!fk_left && is_join(rr->op) && !rel_is_ref(rr)) {
3164 2 : r->r = rel_simplify_count_fk_join(sql, rr, gexps, gcols, changes);
3165 2 : if (rr != r->r)
3166 2 : rel_destroy(rr);
3167 : }
3168 :
3169 39 : if (!check_projection_on_foreignside(r, gcols, fk_left))
3170 : return r;
3171 :
3172 : /* rewrite, ie remove pkey side if possible */
3173 37 : le = (sql_exp*)je->l, re = (sql_exp*)je->l;
3174 :
3175 : /* both have NULL and there are semantics, the join cannot be removed */
3176 37 : if (is_semantics(je) && has_nil(le) && has_nil(re))
3177 : return r;
3178 :
3179 37 : (*changes)++;
3180 : /* if the foreign key column doesn't have NULL values, then return it */
3181 37 : if (!has_nil(le) || is_full(r->op) || (fk_left && is_left(r->op)) || (!fk_left && is_right(r->op))) {
3182 31 : if (fk_left) {
3183 25 : nr = rel_dup(r->l);
3184 : } else {
3185 6 : nr = rel_dup(r->r);
3186 : }
3187 31 : return nr;
3188 : }
3189 :
3190 : /* remove NULL values, ie generate a select not null */
3191 6 : nje = exp_compare(sql->sa, exp_ref(sql, le), exp_atom(sql->sa, atom_general(sql->sa, exp_subtype(le), NULL, 0)), cmp_equal);
3192 6 : set_anti(nje);
3193 6 : set_has_no_nil(nje);
3194 6 : set_semantics(nje);
3195 6 : if (fk_left) {
3196 6 : nr = rel_dup(r->l);
3197 : } else {
3198 0 : nr = rel_dup(r->r);
3199 : }
3200 6 : nr = rel_select(sql->sa, nr, nje);
3201 6 : set_processed(nr);
3202 6 : return nr;
3203 : }
3204 :
3205 : /*
3206 : * Handle (left/right/outer/natural) join fk-pk rewrites
3207 : * 1 group by ( fk-pk-join () ) [ count(*) ] -> group by ( fk )
3208 : * 2 project ( fk-pk-join () ) [ fk-column ] -> project (fk table)[ fk-column ]
3209 : * 3 project ( fk1-pk1-join( fk2-pk2-join()) [ fk-column, pk1 column ] -> project (fk1-pk1-join)[ fk-column, pk1 column ]
3210 : */
3211 : static inline sql_rel *
3212 3642354 : rel_simplify_fk_joins(visitor *v, sql_rel *rel)
3213 : {
3214 3642354 : sql_rel *r = NULL;
3215 :
3216 3642354 : if (is_simple_project(rel->op))
3217 1194614 : r = rel->l;
3218 :
3219 3642391 : while (is_simple_project(rel->op) && r && list_length(r->exps) == 1 && (is_join(r->op) || r->op == op_semi) && !(rel_is_ref(r))) {
3220 237704 : sql_rel *or = r;
3221 :
3222 237704 : r = rel_simplify_project_fk_join(v->sql, r, rel->exps, rel->r, &v->changes);
3223 237704 : if (r == or)
3224 : return rel;
3225 37 : rel_destroy(rel->l);
3226 37 : rel->l = r;
3227 : }
3228 :
3229 3404687 : if (!is_groupby(rel->op))
3230 : return rel;
3231 :
3232 86687 : r = rel->l;
3233 130803 : while(r && is_simple_project(r->op))
3234 44116 : r = r->l;
3235 :
3236 101097 : while (is_groupby(rel->op) && !rel_is_ref(rel) && r && (is_join(r->op) || r->op == op_semi) && list_length(r->exps) == 1 && !(rel_is_ref(r)) &&
3237 : /* currently only single count aggregation is handled, no other projects or aggregation */
3238 105129 : list_length(rel->exps) == 1 && exp_aggr_is_count(rel->exps->h->data)) {
3239 1839 : sql_rel *or = r;
3240 :
3241 1839 : r = rel_simplify_count_fk_join(v->sql, r, rel->exps, rel->r, &v->changes);
3242 1839 : if (r == or)
3243 : return rel;
3244 26 : rel_destroy(rel->l);
3245 26 : rel->l = r;
3246 : }
3247 : return rel;
3248 : }
3249 :
3250 : /*
3251 : * Gets the column expressions of a diff function and adds them to "columns".
3252 : * The diff function has two possible argument types: either a sql_exp representing a column
3253 : * or a sql_exp representing another diff function, therefore this function is recursive.
3254 : */
3255 : static void
3256 168 : get_diff_function_columns(sql_exp *diffExp, list *columns) {
3257 168 : list *args = diffExp->l;
3258 :
3259 370 : for (node *arg = args->h; arg; arg = arg->next) {
3260 202 : sql_exp *exp = arg->data;
3261 :
3262 : // diff function
3263 202 : if (exp->type == e_func) {
3264 34 : get_diff_function_columns(exp, columns);
3265 : }
3266 : // column
3267 : else {
3268 168 : list_append(columns, exp);
3269 : }
3270 : }
3271 168 : }
3272 :
3273 : /*
3274 : * Builds a list of aggregation key columns to be used by the select push down algorithm, namely for
3275 : * window functions. Returns NULL if the window function does not partition by any column
3276 : */
3277 : static list *
3278 3623 : get_aggregation_key_columns(allocator *sa, sql_rel *r) {
3279 34495 : for (node* n = r->exps->h; n; n = n->next) {
3280 31096 : sql_exp *e = n->data;
3281 :
3282 31096 : if (e->type == e_func) {
3283 7923 : sql_subfunc *f = e->f;
3284 :
3285 : // aggregation function
3286 7923 : if (!strcmp(f->func->base.name, "rank")) {
3287 224 : list* rankArguments = e->l;
3288 : // the partition key is the second argument
3289 224 : sql_exp *partitionExp = rankArguments->h->next->data;
3290 :
3291 : // check if the key contains any columns, i.e., is a diff function
3292 224 : if (partitionExp->type == e_func) {
3293 : // get columns to list
3294 134 : list *aggColumns = sa_list(sa);
3295 134 : get_diff_function_columns(partitionExp, aggColumns);
3296 134 : return aggColumns;
3297 : }
3298 : // the function has no aggregation columns (e_atom of boolean)
3299 : else {
3300 : return NULL;
3301 : }
3302 :
3303 : }
3304 : }
3305 : }
3306 : return NULL;
3307 : }
3308 :
3309 : /*
3310 : * Checks if a filter column is also used as an aggregation key, so it can be later safely pushed down.
3311 : */
3312 : static int
3313 187 : filter_column_in_aggregation_columns(sql_exp *column, list *aggColumns) {
3314 : /* check if it is a column or an e_convert, and get the actual column if it is the latter */
3315 187 : if (column->type == e_convert) {
3316 1 : column = column->l;
3317 : }
3318 :
3319 187 : char *tableName = column->l;
3320 187 : char *columnName = column->r;
3321 :
3322 404 : for (node *n = aggColumns->h; n; n = n->next) {
3323 226 : sql_exp *aggCol = n->data;
3324 226 : char *aggColTableName = aggCol->l;
3325 226 : char *aggColColumnName = aggCol->r;
3326 :
3327 226 : if (!strcmp(tableName, aggColTableName) && !strcmp(columnName, aggColColumnName)) {
3328 : /* match */
3329 : return 1;
3330 : }
3331 : }
3332 :
3333 : /* no matches found */
3334 : return 0;
3335 : }
3336 :
3337 :
3338 : /*
3339 : * Push select down, pushes the selects through (simple) projections. Also
3340 : * it cleans up the projections which become useless.
3341 : *
3342 : * WARNING - Make sure to call try_remove_empty_select macro before returning so we ensure
3343 : * possible generated empty selects won't never be generated
3344 : */
3345 : static sql_rel *
3346 7211230 : rel_push_select_down(visitor *v, sql_rel *rel)
3347 : {
3348 7211230 : list *exps = NULL;
3349 7211230 : sql_rel *r = NULL;
3350 7211230 : node *n;
3351 :
3352 7211230 : if (rel_is_ref(rel)) {
3353 374244 : if (is_select(rel->op) && !list_empty(rel->exps)) {
3354 : /* add inplace empty select */
3355 1732 : sql_rel *l = rel_select(v->sql->sa, rel->l, NULL);
3356 :
3357 1732 : l->exps = rel->exps;
3358 1732 : set_processed(l);
3359 1732 : rel->exps = NULL;
3360 1732 : rel->l = l;
3361 1732 : v->changes++;
3362 : }
3363 374244 : return rel;
3364 : }
3365 :
3366 : /* don't make changes for empty selects */
3367 6836986 : if (is_select(rel->op) && list_empty(rel->exps))
3368 10 : return try_remove_empty_select(v, rel);
3369 :
3370 : /* merge 2 selects */
3371 6836976 : r = rel->l;
3372 6836976 : if (is_select(rel->op) && r && r->exps && is_select(r->op) && !(rel_is_ref(r)) && !exps_have_func(rel->exps)) {
3373 27 : (void)list_merge(r->exps, rel->exps, (fdup)NULL);
3374 27 : rel->l = NULL;
3375 27 : rel_destroy(rel);
3376 27 : v->changes++;
3377 27 : return try_remove_empty_select(v, r);
3378 : }
3379 : /*
3380 : * Push select through semi/anti join
3381 : * select (semi(A,B)) == semi(select(A), B)
3382 : */
3383 6836949 : if (is_select(rel->op) && r && is_semi(r->op) && !(rel_is_ref(r))) {
3384 305 : rel->l = r->l;
3385 305 : r->l = rel;
3386 305 : v->changes++;
3387 : /*
3388 : * if A has 2 references (ie used on both sides of
3389 : * the semi join), we also push the select into A.
3390 : */
3391 305 : if (rel_is_ref(rel->l) && rel->l == rel_find_ref(r->r)){
3392 0 : sql_rel *lx = rel->l;
3393 0 : sql_rel *rx = r->r;
3394 0 : if (lx->ref.refcnt == 2 && !rel_is_ref(rx)) {
3395 0 : while (rx->l && !rel_is_ref(rx->l) &&
3396 0 : (is_project(rx->op) ||
3397 : is_select(rx->op) ||
3398 : is_join(rx->op)))
3399 : rx = rx->l;
3400 : /* probably we need to introduce a project */
3401 0 : rel_destroy(rel->l);
3402 0 : lx = rel_project(v->sql->sa, rel, rel_projections(v->sql, rel, NULL, 1, 1));
3403 0 : r->l = lx;
3404 0 : rx->l = rel_dup(lx);
3405 : }
3406 : }
3407 305 : return r;
3408 : }
3409 6836644 : exps = rel->exps;
3410 :
3411 : /* push select through join */
3412 6836644 : if (is_select(rel->op) && r && is_join(r->op) && !rel_is_ref(r) && !is_single(r)){
3413 22194 : sql_rel *jl = r->l, *ojl = jl, *jr = r->r, *ojr = jr;
3414 22194 : int left = r->op == op_join || r->op == op_left;
3415 22194 : int right = r->op == op_join || r->op == op_right;
3416 :
3417 22194 : if (r->op == op_full)
3418 : return rel;
3419 :
3420 : /* introduce selects under the join (if needed) */
3421 22168 : set_processed(jl);
3422 22168 : set_processed(jr);
3423 48707 : for (n = exps->h; n;) {
3424 26539 : node *next = n->next;
3425 26539 : sql_exp *e = n->data;
3426 :
3427 26539 : if (!exp_unsafe(e, false, true)) {
3428 26333 : if (left && rel_rebind_exp(v->sql, jl, e)) {
3429 13335 : if (!is_select(jl->op) || rel_is_ref(jl))
3430 11146 : r->l = jl = rel_select(v->sql->sa, jl, NULL);
3431 13335 : rel_select_add_exp(v->sql->sa, jl, e);
3432 13335 : list_remove_node(exps, NULL, n);
3433 13335 : v->changes++;
3434 12998 : } else if (right && rel_rebind_exp(v->sql, jr, e)) {
3435 5158 : if (!is_select(jr->op) || rel_is_ref(jr))
3436 4207 : r->r = jr = rel_select(v->sql->sa, jr, NULL);
3437 5158 : rel_select_add_exp(v->sql->sa, jr, e);
3438 5158 : list_remove_node(exps, NULL, n);
3439 5158 : v->changes++;
3440 : }
3441 : }
3442 : n = next;
3443 : }
3444 22168 : if (ojl != jl)
3445 11146 : set_processed(jl);
3446 22168 : if (ojr != jr)
3447 4207 : set_processed(jr);
3448 : }
3449 :
3450 : /* merge select and cross product ? */
3451 6836618 : if (is_select(rel->op) && r && r->op == op_join && !rel_is_ref(r) && !is_single(r) && !exps_have_unsafe(exps, false, true)) {
3452 14531 : for (n = exps->h; n;) {
3453 1913 : node *next = n->next;
3454 1913 : sql_exp *e = n->data;
3455 :
3456 1913 : if (exp_is_join(e, NULL) == 0) {
3457 693 : if (!r->exps)
3458 109 : r->exps = sa_list(v->sql->sa);
3459 693 : append(r->exps, e);
3460 693 : list_remove_node(exps, NULL, n);
3461 693 : v->changes++;
3462 : }
3463 : n = next;
3464 : }
3465 : }
3466 :
3467 6836618 : if (is_select(rel->op) && r && (is_simple_project(r->op) || (is_groupby(r->op) && !list_empty(r->r))) && !rel_is_ref(r) && !is_single(r)){
3468 70240 : sql_rel *pl = r->l, *opl = pl;
3469 : /* we cannot push through window functions (for safety I disabled projects over DDL too) */
3470 70240 : if (pl && pl->op != op_ddl && !exps_have_unsafe(r->exps, false, false)) {
3471 : /* introduce selects under the project (if needed) */
3472 48704 : set_processed(pl);
3473 48704 : if (!pl->exps)
3474 411 : pl->exps = sa_list(v->sql->sa);
3475 103718 : for (n = exps->h; n;) {
3476 55014 : node *next = n->next;
3477 55014 : sql_exp *e = n->data, *ne = NULL;
3478 :
3479 : /* can we move it down */
3480 55014 : if (e->type == e_cmp && (ne = exp_push_down_prj(v->sql, e, r, pl)) && ne != e) {
3481 25408 : if (!(is_select(pl->op) && is_join(pl->op) && is_semi(pl->op)) || rel_is_ref(pl))
3482 25408 : r->l = pl = rel_select(v->sql->sa, pl, NULL);
3483 25408 : rel_select_add_exp(v->sql->sa, pl, ne);
3484 25408 : list_remove_node(exps, NULL, n);
3485 25408 : v->changes++;
3486 : }
3487 : n = next;
3488 : }
3489 48704 : if (opl != pl)
3490 18329 : set_processed(pl);
3491 : }
3492 :
3493 : /* push filters if they match the aggregation key on a window function */
3494 3623 : else if (pl && pl->op != op_ddl && exps_have_unsafe(r->exps, false, false)) {
3495 3623 : set_processed(pl);
3496 : /* list of aggregation key columns */
3497 3623 : list *aggColumns = get_aggregation_key_columns(v->sql->sa, r);
3498 :
3499 : /* aggregation keys found, check if any filter matches them */
3500 3623 : if (aggColumns) {
3501 325 : for (n = exps->h; n;) {
3502 191 : node *next = n->next;
3503 191 : sql_exp *e = n->data, *ne = NULL;
3504 :
3505 191 : if (e->type == e_cmp) {
3506 : /* simple comparison filter */
3507 191 : if (e->flag == cmp_gt || e->flag == cmp_gte || e->flag == cmp_lte || e->flag == cmp_lt
3508 191 : || e->flag == cmp_equal || e->flag == cmp_notequal || e->flag == cmp_in || e->flag == cmp_notin
3509 5 : || (e->flag == cmp_filter && ((list*)e->l)->cnt == 1)) {
3510 187 : sql_exp* column;
3511 : /* the column in 'like' filters is stored inside a list */
3512 187 : if (e->flag == cmp_filter) {
3513 1 : column = ((list*)e->l)->h->data;
3514 : } else {
3515 186 : column = e->l;
3516 : }
3517 :
3518 : /* check if the expression matches any aggregation key, meaning we can
3519 : try to safely push it down */
3520 187 : if (filter_column_in_aggregation_columns(column, aggColumns)) {
3521 9 : ne = exp_push_down_prj(v->sql, e, r, pl);
3522 :
3523 : /* can we move it down */
3524 9 : if (ne && ne != e && pl->exps) {
3525 9 : if (!is_select(pl->op) || rel_is_ref(pl))
3526 8 : r->l = pl = rel_select(v->sql->sa, pl, NULL);
3527 9 : rel_select_add_exp(v->sql->sa, pl, ne);
3528 9 : list_remove_node(exps, NULL, n);
3529 9 : v->changes++;
3530 : }
3531 : }
3532 : }
3533 : }
3534 : n = next;
3535 : }
3536 :
3537 : /* cleanup list */
3538 134 : list_destroy(aggColumns);
3539 : }
3540 : }
3541 : }
3542 :
3543 : /* try push select under set relation */
3544 6836618 : if (is_select(rel->op) && r && is_set(r->op) && !list_empty(r->exps) && !rel_is_ref(r) && !is_single(r) && !list_empty(exps)) {
3545 19 : sql_rel *u = r, *ul = u->l, *ur = u->r;
3546 :
3547 19 : ul = rel_dup(ul);
3548 19 : ur = rel_dup(ur);
3549 19 : if (!is_project(ul->op))
3550 0 : ul = rel_project(v->sql->sa, ul,
3551 : rel_projections(v->sql, ul, NULL, 1, 1));
3552 19 : if (!is_project(ur->op))
3553 0 : ur = rel_project(v->sql->sa, ur,
3554 : rel_projections(v->sql, ur, NULL, 1, 1));
3555 19 : rel_rename_exps(v->sql, u->exps, ul->exps);
3556 19 : rel_rename_exps(v->sql, u->exps, ur->exps);
3557 :
3558 : /* introduce selects under the set */
3559 19 : ul = rel_select(v->sql->sa, ul, NULL);
3560 19 : ul->exps = exps_copy(v->sql, exps);
3561 19 : set_processed(ul);
3562 19 : ur = rel_select(v->sql->sa, ur, NULL);
3563 19 : ur->exps = exps_copy(v->sql, exps);
3564 19 : set_processed(ur);
3565 :
3566 19 : rel = rel_inplace_setop(v->sql, rel, ul, ur, u->op, rel_projections(v->sql, rel, NULL, 1, 1));
3567 19 : if (need_distinct(u))
3568 0 : set_distinct(rel);
3569 19 : v->changes++;
3570 : }
3571 6836618 : if (is_select(rel->op) && r && is_munion(r->op) && !list_empty(r->exps) && !rel_is_ref(r) && !is_single(r) && !list_empty(exps)) {
3572 4868 : sql_rel *u = r;
3573 4868 : list *rels = u->l, *nrels = sa_list(v->sql->sa);
3574 14630 : for(node *n = rels->h; n; n = n->next) {
3575 9762 : sql_rel *ul = n->data;
3576 9762 : ul = rel_dup(ul);
3577 9762 : if (!is_project(ul->op))
3578 8 : ul = rel_project(v->sql->sa, ul,
3579 : rel_projections(v->sql, ul, NULL, 1, 1));
3580 9762 : rel_rename_exps(v->sql, u->exps, ul->exps);
3581 :
3582 : /* introduce selects under the set */
3583 9762 : ul = rel_select(v->sql->sa, ul, NULL);
3584 9762 : ul->exps = exps_copy(v->sql, exps);
3585 9762 : set_processed(ul);
3586 9762 : nrels = append(nrels, ul);
3587 : }
3588 :
3589 4868 : rel = rel_inplace_setop_n_ary(v->sql, rel, nrels, u->op, rel_projections(v->sql, rel, NULL, 1, 1));
3590 4868 : if (need_distinct(u))
3591 6 : set_distinct(rel);
3592 4868 : v->changes++;
3593 : }
3594 :
3595 6836618 : return try_remove_empty_select(v, rel);
3596 : }
3597 :
3598 : static int
3599 12055 : index_exp(sql_exp *e, sql_idx *i)
3600 : {
3601 12055 : if (e->type == e_cmp && !is_complex_exp(e->flag)) {
3602 6798 : switch(i->type) {
3603 6798 : case hash_idx:
3604 : case oph_idx:
3605 6798 : if (e->flag == cmp_equal)
3606 : return 0;
3607 : /* fall through */
3608 : case join_idx:
3609 : default:
3610 1252 : return -1;
3611 : }
3612 : }
3613 : return -1;
3614 : }
3615 :
3616 : /* find column for the select/join expression */
3617 : static sql_column *
3618 5546 : sjexp_col(sql_exp *e, sql_rel *r)
3619 : {
3620 5546 : sql_column *res = NULL;
3621 :
3622 5546 : if (e->type == e_cmp && !is_complex_exp(e->flag)) {
3623 5546 : res = exp_find_column(r, e->l, -2);
3624 5546 : if (!res)
3625 486 : res = exp_find_column(r, e->r, -2);
3626 : }
3627 5546 : return res;
3628 : }
3629 :
3630 : static sql_idx *
3631 1772548 : find_index(allocator *sa, sql_rel *rel, sql_rel *sub, list **EXPS)
3632 : {
3633 1772548 : node *n;
3634 :
3635 : /* any (partial) match of the expressions with the index columns */
3636 : /* Depending on the index type we may need full matches and only
3637 : limited number of cmp types (hash only equality etc) */
3638 : /* Depending on the index type we should (in the rel_bin) generate
3639 : more code, ie for spatial index add post filter etc, for hash
3640 : compute hash value and use index */
3641 :
3642 1772548 : if (sub->exps && rel->exps)
3643 7115318 : for(n = sub->exps->h; n; n = n->next) {
3644 5525632 : prop *p;
3645 5525632 : sql_exp *e = n->data;
3646 :
3647 5525632 : if ((p = find_prop(e->p, PROP_HASHIDX)) != NULL) {
3648 9419 : list *exps, *cols;
3649 9419 : sql_idx *i = p->value.pval;
3650 9419 : fcmp cmp = (fcmp)&sql_column_kc_cmp;
3651 :
3652 : /* join indices are only interesting for joins */
3653 9419 : if (i->type == join_idx || list_length(i->columns) <= 1)
3654 0 : continue;
3655 : /* based on the index type, find qualifying exps */
3656 9419 : exps = list_select(rel->exps, i, (fcmp) &index_exp, (fdup)NULL);
3657 9419 : if (list_empty(exps))
3658 4848 : continue;
3659 : /* now we obtain the columns, move into sql_column_kc_cmp! */
3660 4571 : cols = list_map(exps, sub, (fmap) &sjexp_col);
3661 :
3662 : /* TODO check that at most 2 relations are involved */
3663 :
3664 : /* Match the index columns with the expression columns.
3665 : TODO, Allow partial matches ! */
3666 4571 : if (list_match(cols, i->columns, cmp) == 0) {
3667 : /* re-order exps in index order */
3668 297 : node *n, *m;
3669 297 : list *es = sa_list(sa);
3670 :
3671 991 : for(n = i->columns->h; n; n = n->next) {
3672 694 : int i = 0;
3673 2185 : for(m = cols->h; m; m = m->next, i++) {
3674 2185 : if (cmp(m->data, n->data) == 0){
3675 694 : sql_exp *e = list_fetch(exps, i);
3676 694 : list_append(es, e);
3677 694 : break;
3678 : }
3679 : }
3680 : }
3681 : /* fix the destroy function */
3682 297 : cols->destroy = NULL;
3683 297 : *EXPS = es;
3684 297 : e->used = 1;
3685 297 : return i;
3686 : }
3687 4274 : cols->destroy = NULL;
3688 : }
3689 : }
3690 : return NULL;
3691 : }
3692 :
3693 : static inline sql_rel *
3694 1158187 : rel_use_index(visitor *v, sql_rel *rel)
3695 : {
3696 1158187 : list *exps = NULL;
3697 1158187 : sql_idx *i = find_index(v->sql->sa, rel, rel->l, &exps);
3698 1158187 : int left = 1;
3699 :
3700 1158187 : assert(is_select(rel->op) || is_join(rel->op));
3701 1158187 : if (!i && is_join(rel->op)) {
3702 614361 : left = 0;
3703 614361 : i = find_index(v->sql->sa, rel, rel->r, &exps);
3704 : }
3705 :
3706 1158007 : if (i) {
3707 297 : prop *p;
3708 297 : node *n;
3709 297 : int single_table = 1;
3710 297 : sql_exp *re = NULL;
3711 :
3712 989 : for( n = exps->h; n && single_table; n = n->next) {
3713 692 : sql_exp *e = n->data, *nre = e->l;
3714 :
3715 692 : if (!is_compare(e->type) || is_anti(e) || e->flag != cmp_equal)
3716 : return rel;
3717 692 : if (is_join(rel->op) && ((left && !rel_find_exp(rel->l, nre)) || (!left && rel_find_exp(rel->r, nre))))
3718 113 : nre = e->r;
3719 692 : single_table = (!re || (exp_relname(nre) && exp_relname(re) && strcmp(exp_relname(nre), exp_relname(re)) == 0));
3720 692 : re = nre;
3721 : }
3722 297 : if (single_table) { /* add PROP_HASHCOL to all column exps */
3723 965 : for( n = exps->h; n; n = n->next) {
3724 676 : sql_exp *e = n->data;
3725 :
3726 : /* swapped ? */
3727 676 : if (is_join(rel->op) && ((left && !rel_find_exp(rel->l, e->l)) || (!left && !rel_find_exp(rel->r, e->l)))) {
3728 185 : exp_swap(e);
3729 : }
3730 676 : p = find_prop(e->p, PROP_HASHCOL);
3731 676 : if (!p)
3732 305 : e->p = p = prop_create(v->sql->sa, PROP_HASHCOL, e->p);
3733 676 : p->value.pval = i;
3734 : }
3735 : }
3736 : /* add the remaining exps to the new exp list */
3737 297 : if (list_length(rel->exps) > list_length(exps)) {
3738 108 : for( n = rel->exps->h; n; n = n->next) {
3739 81 : sql_exp *e = n->data;
3740 81 : if (!list_find(exps, e, (fcmp)&exp_cmp))
3741 27 : list_append(exps, e);
3742 : }
3743 : }
3744 297 : rel->exps = exps;
3745 : }
3746 : return rel;
3747 : }
3748 :
3749 : static sql_rel *
3750 3642354 : rel_select_leftgroup_2_semi(visitor *v, sql_rel *rel)
3751 : {
3752 3642354 : if (rel_is_ref(rel) || !is_select(rel->op) || list_empty(rel->exps))
3753 3108808 : return rel;
3754 533546 : sql_rel *l = rel->l;
3755 :
3756 533546 : if (!l || rel_is_ref(l) || !is_left(l->op) || list_empty(l->attr))
3757 532560 : return rel;
3758 :
3759 1964 : for(node *n = rel->exps->h; n; n = n->next) {
3760 986 : sql_exp *e = n->data;
3761 :
3762 986 : if (e->type == e_cmp && !is_semantics(e) && !e->f) {
3763 973 : list *attrs = l->attr;
3764 973 : sql_exp *a = attrs->h->data;
3765 :
3766 973 : if (exps_find_exp(l->attr, e->l) && exp_is_true(e->r) && e->flag == cmp_equal /*&& exp_is_true(a)*/) {
3767 : // printf("# optimize select leftgroup -> semi\n");
3768 8 : if (!list_empty(l->exps)) {
3769 13 : for(node *m = l->exps->h; m; m = m->next) {
3770 7 : sql_exp *j = m->data;
3771 7 : reset_any(j);
3772 : }
3773 : }
3774 8 : l->attr = NULL;
3775 8 : l->op = exp_is_true(a)?op_semi:op_anti;
3776 8 : list_remove_node(rel->exps, NULL, n);
3777 8 : rel = rel_project(v->sql->sa, rel, rel_projections(v->sql, rel, NULL, 1, 1));
3778 8 : list_append(rel->exps, attrs->h->data);
3779 8 : v->changes++;
3780 8 : return rel;
3781 : }
3782 : }
3783 : }
3784 : return rel;
3785 : }
3786 :
3787 : static sql_rel *
3788 3642354 : rel_optimize_select_and_joins_topdown_(visitor *v, sql_rel *rel)
3789 : {
3790 : /* push_join_down introduces semijoins */
3791 3642354 : uint8_t cycle = *(uint8_t*) v->data;
3792 3642354 : if (cycle <= 0) {
3793 1586870 : rel = rel_semijoin_use_fk(v, rel);
3794 1586870 : rel = rel_push_join_down(v, rel);
3795 : }
3796 :
3797 3642354 : rel = rel_simplify_fk_joins(v, rel);
3798 3642354 : rel = rel_push_select_down(v, rel);
3799 3642354 : rel = rel_select_leftgroup_2_semi(v, rel);
3800 3642354 : if (rel && rel->l && (is_select(rel->op) || is_join(rel->op)))
3801 1158187 : rel = rel_use_index(v, rel);
3802 3642354 : return rel;
3803 : }
3804 :
3805 : static sql_rel *
3806 126483 : rel_optimize_select_and_joins_topdown(visitor *v, global_props *gp, sql_rel *rel)
3807 : {
3808 126483 : v->data = &gp->opt_cycle;
3809 126483 : rel = rel_visitor_topdown(v, rel, &rel_optimize_select_and_joins_topdown_);
3810 126483 : v->data = gp;
3811 126483 : return rel;
3812 : }
3813 :
3814 : run_optimizer
3815 623304 : bind_optimize_select_and_joins_topdown(visitor *v, global_props *gp)
3816 : {
3817 623304 : int flag = v->sql->sql_optimizer;
3818 623104 : return gp->opt_level == 1 && (gp->cnt[op_join] || gp->cnt[op_left] || gp->cnt[op_right]
3819 538955 : || gp->cnt[op_full] || gp->cnt[op_semi] || gp->cnt[op_anti] ||
3820 1246408 : gp->cnt[op_select]) && (flag & optimize_select_and_joins_topdown) ? rel_optimize_select_and_joins_topdown : NULL;
3821 : }
3822 :
3823 :
3824 : static int
3825 1243178 : can_push_func(sql_exp *e, sql_rel *rel, int *must, int depth)
3826 : {
3827 1287404 : switch(e->type) {
3828 1089443 : case e_cmp: {
3829 1089443 : sql_exp *l = e->l, *r = e->r, *f = e->f;
3830 :
3831 : /* don't push down functions inside attribute joins */
3832 1089443 : if (e->flag == cmp_or || e->flag == cmp_in || e->flag == cmp_notin || e->flag == cmp_filter || (is_join(rel->op) && is_any(e)))
3833 : return 0;
3834 1033839 : if (depth > 0) { /* for comparisons under the top ones, they become functions */
3835 58 : int lmust = 0;
3836 58 : int res = can_push_func(l, rel, &lmust, depth + 1) && can_push_func(r, rel, &lmust, depth + 1) &&
3837 20 : (!f || can_push_func(f, rel, &lmust, depth + 1));
3838 12 : if (res && !lmust)
3839 : return 1;
3840 54 : (*must) |= lmust;
3841 54 : return res;
3842 : } else {
3843 1033781 : int mustl = 0, mustr = 0, mustf = 0;
3844 1033781 : return ((l->type == e_column || can_push_func(l, rel, &mustl, depth + 1)) && (*must = mustl)) ||
3845 2060521 : ((r->type == e_column || can_push_func(r, rel, &mustr, depth + 1)) && (*must = mustr)) ||
3846 3644 : ((f && (f->type == e_column || can_push_func(f, rel, &mustf, depth + 1)) && (*must = mustf)));
3847 : }
3848 : }
3849 44226 : case e_convert:
3850 44226 : return can_push_func(e->l, rel, must, depth + 1);
3851 9457 : case e_aggr:
3852 : case e_func: {
3853 9457 : list *l = e->l;
3854 9457 : int res = 1, lmust = 0;
3855 :
3856 9457 : if (exp_unsafe(e, false, false))
3857 : return 0;
3858 26949 : if (l) for (node *n = l->h; n && res; n = n->next)
3859 17492 : res &= can_push_func(n->data, rel, &lmust, depth + 1);
3860 9457 : if (res && !lmust)
3861 : return 1;
3862 8610 : (*must) |= lmust;
3863 8610 : return res;
3864 : }
3865 50428 : case e_column:
3866 50428 : if (rel && !rel_find_exp(rel, e))
3867 : return 0;
3868 29700 : (*must) = 1;
3869 : /* fall through */
3870 : default:
3871 : return 1;
3872 : }
3873 : }
3874 :
3875 : static int
3876 903061 : exps_can_push_func(list *exps, sql_rel *rel)
3877 : {
3878 3926384 : for(node *n = exps->h; n; n = n->next) {
3879 3047160 : sql_exp *e = n->data;
3880 3047160 : int mustl = 0, mustr = 0;
3881 :
3882 3047160 : if ((is_joinop(rel->op) || is_select(rel->op)) && ((can_push_func(e, rel->l, &mustl, 0) && mustl)))
3883 23837 : return 1;
3884 3042896 : if (is_joinop(rel->op) && can_push_func(e, rel->r, &mustr, 0) && mustr)
3885 : return 1;
3886 : }
3887 : return 0;
3888 : }
3889 :
3890 : static int
3891 82245 : exp_needs_push_down(sql_rel *rel, sql_exp *e)
3892 : {
3893 106734 : switch(e->type) {
3894 27607 : case e_cmp:
3895 : /* don't push down functions inside attribute joins */
3896 27607 : if (e->flag == cmp_or || e->flag == cmp_in || e->flag == cmp_notin || e->flag == cmp_filter || (is_join(rel->op) && is_any(e)))
3897 : return 0;
3898 26671 : return exp_needs_push_down(rel, e->l) || exp_needs_push_down(rel, e->r) || (e->f && exp_needs_push_down(rel, e->f));
3899 24489 : case e_convert:
3900 24489 : return exp_needs_push_down(rel, e->l);
3901 1922 : case e_aggr:
3902 : case e_func:
3903 1922 : if (!e->l || exps_are_atoms(e->l))
3904 20 : return 0;
3905 : return 1;
3906 1459 : case e_atom:
3907 1459 : if (!e->f || exps_are_atoms(e->f))
3908 1459 : return 0;
3909 : return 1;
3910 : case e_column:
3911 : default:
3912 : return 0;
3913 : }
3914 : }
3915 :
3916 : static int
3917 23837 : exps_need_push_down(sql_rel *rel, list *exps )
3918 : {
3919 49534 : for(node *n = exps->h; n; n = n->next)
3920 27599 : if (exp_needs_push_down(rel, n->data))
3921 : return 1;
3922 : return 0;
3923 : }
3924 :
3925 : static sql_exp *exp_push_single_func_down(visitor *v, sql_rel *rel, sql_rel *ol, sql_rel *or, sql_exp *e, int depth);
3926 :
3927 : static list *
3928 2414 : exps_push_single_func_down(visitor *v, sql_rel *rel, sql_rel *ol, sql_rel *or, list *exps, int depth)
3929 : {
3930 4828 : if (mvc_highwater(v->sql))
3931 : return exps;
3932 :
3933 6724 : for (node *n = exps->h; n; n = n->next)
3934 4310 : if ((n->data = exp_push_single_func_down(v, rel, ol, or, n->data, depth)) == NULL)
3935 : return NULL;
3936 : return exps;
3937 : }
3938 :
3939 : static sql_exp *
3940 15731 : exp_push_single_func_down(visitor *v, sql_rel *rel, sql_rel *ol, sql_rel *or, sql_exp *e, int depth)
3941 : {
3942 27503 : if (mvc_highwater(v->sql))
3943 : return e;
3944 :
3945 15731 : switch(e->type) {
3946 4247 : case e_cmp: {
3947 4247 : if (e->flag == cmp_or || e->flag == cmp_filter) {
3948 247 : if ((e->l = exps_push_single_func_down(v, rel, ol, or, e->l, depth + 1)) == NULL)
3949 : return NULL;
3950 247 : if ((e->r = exps_push_single_func_down(v, rel, ol, or, e->r, depth + 1)) == NULL)
3951 : return NULL;
3952 4000 : } else if (e->flag == cmp_in || e->flag == cmp_notin) {
3953 18 : if ((e->l = exp_push_single_func_down(v, rel, ol, or, e->l, depth + 1)) == NULL)
3954 : return NULL;
3955 18 : if ((e->r = exps_push_single_func_down(v, rel, ol, or, e->r, depth + 1)) == NULL)
3956 : return NULL;
3957 : } else {
3958 3982 : if ((e->l = exp_push_single_func_down(v, rel, ol, or, e->l, depth + 1)) == NULL)
3959 : return NULL;
3960 3982 : if ((e->r = exp_push_single_func_down(v, rel, ol, or, e->r, depth + 1)) == NULL)
3961 : return NULL;
3962 3982 : if (e->f && (e->f = exp_push_single_func_down(v, rel, ol, or, e->f, depth + 1)) == NULL)
3963 : return NULL;
3964 : }
3965 : } break;
3966 2004 : case e_convert:
3967 2004 : if ((e->l = exp_push_single_func_down(v, rel, ol, or, e->l, depth + 1)) == NULL)
3968 : return NULL;
3969 : break;
3970 3982 : case e_aggr:
3971 : case e_func: {
3972 3982 : sql_rel *l = rel->l, *r = rel->r;
3973 3982 : int must = 0, mustl = 0, mustr = 0;
3974 :
3975 3982 : if (exp_unsafe(e, false, false))
3976 23 : return e;
3977 3982 : if (!e->l || exps_are_atoms(e->l))
3978 23 : return e;
3979 3959 : if ((is_joinop(rel->op) && ((can_push_func(e, l, &mustl, depth + 1) && mustl) || (can_push_func(e, r, &mustr, depth + 1) && mustr))) ||
3980 2977 : (is_select(rel->op) && can_push_func(e, l, &must, depth + 1) && must)) {
3981 3946 : exp_label(v->sql->sa, e, ++v->sql->label);
3982 : /* we need a full projection, group by's and unions cannot be extended with more expressions */
3983 3946 : if (mustr) {
3984 352 : if (r == or) /* don't project twice */
3985 329 : rel->r = r = rel_project(v->sql->sa, r, rel_projections(v->sql, r, NULL, 1, 1));
3986 352 : list_append(r->exps, e);
3987 : } else {
3988 3594 : if (l == ol) /* don't project twice */
3989 1838 : rel->l = l = rel_project(v->sql->sa, l, rel_projections(v->sql, l, NULL, 1, 1));
3990 3594 : list_append(l->exps, e);
3991 : }
3992 3946 : e = exp_ref(v->sql, e);
3993 3946 : v->changes++;
3994 : }
3995 3959 : } break;
3996 1156 : case e_atom: {
3997 1156 : if (e->f && (e->f = exps_push_single_func_down(v, rel, ol, or, e->f, depth + 1)) == NULL)
3998 : return NULL;
3999 : } break;
4000 : case e_column:
4001 : case e_psm:
4002 : break;
4003 : }
4004 : return e;
4005 : }
4006 :
4007 : static inline sql_rel *
4008 3568876 : rel_push_func_down(visitor *v, sql_rel *rel)
4009 : {
4010 3568876 : if ((is_select(rel->op) || is_joinop(rel->op)) && rel->l && rel->exps && !(rel_is_ref(rel))) {
4011 1083488 : int changes = v->changes;
4012 1083488 : sql_rel *l = rel->l, *r = rel->r;
4013 :
4014 : /* only push down when is useful */
4015 1083488 : if ((is_select(rel->op) && list_length(rel->exps) <= 1) || rel_is_ref(l) || (is_joinop(rel->op) && rel_is_ref(r)))
4016 532997 : return rel;
4017 550491 : if (exps_can_push_func(rel->exps, rel) && exps_need_push_down(rel, rel->exps) && !exps_push_single_func_down(v, rel, l, r, rel->exps, 0))
4018 : return NULL;
4019 550491 : if (v->changes > changes) /* once we get a better join order, we can try to remove this projection */
4020 1902 : return rel_project(v->sql->sa, rel, rel_projections(v->sql, rel, NULL, 1, 1));
4021 : }
4022 3033977 : if (is_simple_project(rel->op) && rel->l && rel->exps) {
4023 1126933 : sql_rel *pl = rel->l;
4024 :
4025 1126933 : if (is_joinop(pl->op) && exps_can_push_func(rel->exps, rel)) {
4026 0 : sql_rel *l = pl->l, *r = pl->r, *ol = l, *or = r;
4027 :
4028 0 : for (node *n = rel->exps->h; n; ) {
4029 0 : node *next = n->next;
4030 0 : sql_exp *e = n->data;
4031 0 : int mustl = 0, mustr = 0;
4032 :
4033 0 : if ((can_push_func(e, l, &mustl, 0) && mustl) || (can_push_func(e, r, &mustr, 0) && mustr)) {
4034 0 : if (mustl) {
4035 0 : if (l == ol) /* don't project twice */
4036 0 : pl->l = l = rel_project(v->sql->sa, l, rel_projections(v->sql, l, NULL, 1, 1));
4037 0 : list_append(l->exps, e);
4038 0 : list_remove_node(rel->exps, NULL, n);
4039 0 : v->changes++;
4040 : } else {
4041 0 : if (r == or) /* don't project twice */
4042 0 : pl->r = r = rel_project(v->sql->sa, r, rel_projections(v->sql, r, NULL, 1, 1));
4043 0 : list_append(r->exps, e);
4044 0 : list_remove_node(rel->exps, NULL, n);
4045 0 : v->changes++;
4046 : }
4047 : }
4048 0 : n = next;
4049 : }
4050 : }
4051 : }
4052 : return rel;
4053 : }
4054 :
4055 : static sql_rel *
4056 3568876 : rel_push_func_and_select_down_(visitor *v, sql_rel *rel)
4057 : {
4058 3568876 : if (rel)
4059 3568876 : rel = rel_push_func_down(v, rel);
4060 3568876 : if (rel)
4061 3568876 : rel = rel_push_select_down(v, rel);
4062 3568876 : return rel;
4063 : }
4064 :
4065 : static sql_rel *
4066 126483 : rel_push_func_and_select_down(visitor *v, global_props *gp, sql_rel *rel)
4067 : {
4068 126483 : (void) gp;
4069 126483 : return rel_visitor_topdown(v, rel, &rel_push_func_and_select_down_);
4070 : }
4071 :
4072 : run_optimizer
4073 623038 : bind_push_func_and_select_down(visitor *v, global_props *gp)
4074 : {
4075 623038 : int flag = v->sql->sql_optimizer;
4076 622832 : return gp->opt_level == 1 && (gp->cnt[op_join] || gp->cnt[op_left] || gp->cnt[op_right]
4077 538917 : || gp->cnt[op_full] || gp->cnt[op_semi] || gp->cnt[op_anti] || gp->cnt[op_select])
4078 749097 : && (flag & push_func_and_select_down) ? rel_push_func_and_select_down : NULL;
4079 : }
|