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 : /* The SQL code generator can not always look ahead to avoid
14 : * generation of intermediates.
15 : * Some of these patterns are captured in a postfix optimalisation.
16 : */
17 : #include "monetdb_config.h"
18 : #include "mal_instruction.h"
19 : #include "opt_postfix.h"
20 :
21 : #define isCandidateList(M,P,I) ((M)->var[getArg(P,I)].id[0]== 'C')
22 : str
23 477408 : OPTpostfixImplementation(Client cntxt, MalBlkPtr mb, MalStkPtr stk,
24 : InstrPtr pci)
25 : {
26 477408 : int i, slimit, actions = 0;
27 477408 : str msg = MAL_SUCCEED;
28 477408 : InstrPtr p;
29 :
30 477408 : (void) cntxt;
31 477408 : (void) stk;
32 :
33 477408 : slimit = mb->stop;
34 477408 : setVariableScope(mb);
35 : /* Remove the result from any join/group instruction when it is not used later on */
36 28227621 : for (i = 0; i < slimit; i++) {
37 : /* POSTFIX ACTION FOR THE JOIN CASE */
38 27272778 : p = getInstrPtr(mb, i);
39 27272778 : if (getModuleId(p) == algebraRef && p->retc == 2) {
40 438956 : if (getFunctionId(p) == leftjoinRef || /*getFunctionId(p) == outerjoinRef || */
41 438308 : getFunctionId(p) == bandjoinRef
42 438308 : || getFunctionId(p) == rangejoinRef
43 438166 : || getFunctionId(p) == likejoinRef) {
44 851 : if (getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
45 46 : delArgument(p, p->retc - 1);
46 46 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
47 46 : actions++;
48 46 : continue;
49 : }
50 438105 : } else if (getFunctionId(p) == semijoinRef
51 437859 : || getFunctionId(p) == joinRef
52 102027 : || getFunctionId(p) == thetajoinRef
53 51224 : || /*getFunctionId(p) == outerjoinRef || */
54 51224 : getFunctionId(p) == crossRef) {
55 432609 : int is_first_ret_not_used = getVarEolife(mb, getArg(p, p->retc - 2)) == i;
56 432609 : int is_second_ret_not_used = getVarEolife(mb, getArg(p, p->retc - 1)) == i;
57 :
58 432609 : if (getFunctionId(p) == semijoinRef
59 246 : &&
60 : ((is_first_ret_not_used
61 58 : && getVarConstant(mb, getArg(p, 7)).val.btval != 1 /*not single */ )
62 246 : || is_second_ret_not_used)) {
63 : /* Can't swap arguments on single semijoins */
64 85 : if (is_first_ret_not_used) {
65 : /* semijoin with just the right output is a join */
66 0 : getArg(p, 2) ^= getArg(p, 3); /* swap join inputs */
67 0 : getArg(p, 3) ^= getArg(p, 2);
68 0 : getArg(p, 2) ^= getArg(p, 3);
69 :
70 0 : getArg(p, 4) ^= getArg(p, 5); /* swap candidate lists */
71 0 : getArg(p, 5) ^= getArg(p, 4);
72 0 : getArg(p, 4) ^= getArg(p, 5);
73 0 : setFunctionId(p, joinRef);
74 0 : delArgument(p, 7); /* delete 'max_one' argument */
75 : } else {
76 : /* semijoin with just the left output is an intersection */
77 85 : setFunctionId(p, intersectRef);
78 : }
79 :
80 170 : delArgument(p,
81 85 : is_second_ret_not_used ? p->retc - 1 : p->retc -
82 : 2);
83 85 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
84 85 : actions++;
85 85 : continue;
86 432524 : } else if (is_second_ret_not_used) {
87 53924 : delArgument(p, p->retc - 1);
88 53924 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
89 53924 : actions++;
90 53924 : continue;
91 378600 : } else if (is_first_ret_not_used &&
92 32135 : (getFunctionId(p) == joinRef
93 13716 : || (getFunctionId(p) == thetajoinRef
94 13341 : && isVarConstant(mb, getArg(p, 6)))
95 375 : || (getFunctionId(p) == crossRef
96 317 : && getVarConstant(mb, getArg(p, 4)).val.btval != 1 /*not single */ ))) {
97 : /* Can't swap arguments on single cross products */
98 : /* swap join inputs */
99 32073 : getArg(p, 2) ^= getArg(p, 3);
100 32073 : getArg(p, 3) ^= getArg(p, 2);
101 32073 : getArg(p, 2) ^= getArg(p, 3);
102 :
103 32073 : if (getFunctionId(p) != crossRef) { /* swap candidate lists */
104 31760 : getArg(p, 4) ^= getArg(p, 5);
105 31760 : getArg(p, 5) ^= getArg(p, 4);
106 31760 : getArg(p, 4) ^= getArg(p, 5);
107 31760 : if (getFunctionId(p) == thetajoinRef) { /* swap the comparison */
108 13341 : ValRecord *x = &getVarConstant(mb, getArg(p, 6)),
109 13341 : cst = {
110 : .vtype = TYPE_int
111 : };
112 13341 : switch (x->val.ival) {
113 13269 : case JOIN_LT:
114 13269 : cst.val.ival = JOIN_GT;
115 13269 : break;
116 17 : case JOIN_LE:
117 17 : cst.val.ival = JOIN_GE;
118 17 : break;
119 47 : case JOIN_GT:
120 47 : cst.val.ival = JOIN_LT;
121 47 : break;
122 8 : case JOIN_GE:
123 8 : cst.val.ival = JOIN_LE;
124 8 : break;
125 0 : default:
126 0 : cst.val.ival = x->val.ival;
127 : }
128 13341 : setArg(p, 6, defConstant(mb, TYPE_int, &cst));
129 : }
130 : }
131 32073 : delArgument(p, p->retc - 2);
132 32073 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
133 32073 : actions++;
134 32073 : continue;
135 : }
136 : }
137 : }
138 : /* POSTFIX ACTION FOR THE EXTENT CASE */
139 27186650 : if (getModuleId(p) == groupRef && getFunctionId(p) == groupRef
140 19908 : && getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
141 19869 : delArgument(p, p->retc - 1);
142 19869 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
143 19869 : actions++;
144 19869 : continue;
145 : }
146 27166781 : if (getModuleId(p) == groupRef && getFunctionId(p) == subgroupRef
147 4577 : && getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
148 4577 : delArgument(p, p->retc - 1);
149 4577 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
150 4577 : actions++;
151 4577 : continue;
152 : }
153 27162204 : if (getModuleId(p) == groupRef && getFunctionId(p) == subgroupdoneRef
154 20060 : && getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
155 19804 : delArgument(p, p->retc - 1);
156 19804 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
157 19804 : actions++;
158 19804 : continue;
159 : }
160 27142400 : if (getModuleId(p) == groupRef && getFunctionId(p) == groupdoneRef
161 40633 : && getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
162 31302 : delArgument(p, p->retc - 1);
163 31302 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
164 31302 : actions++;
165 31302 : continue;
166 : }
167 : /* POSTFIX ACTION FOR SORT, could be dropping the last two */
168 27111098 : if (getModuleId(p) == algebraRef && getFunctionId(p) == sortRef
169 22271 : && getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
170 9778 : delArgument(p, p->retc - 1);
171 9778 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
172 9778 : actions++;
173 9778 : if (getModuleId(p) == algebraRef && getFunctionId(p) == sortRef
174 9778 : && getVarEolife(mb, getArg(p, p->retc - 1)) == i) {
175 3406 : delArgument(p, p->retc - 1);
176 3406 : typeChecker(cntxt->usermodule, mb, p, i, TRUE);
177 3406 : actions++;
178 : }
179 9778 : continue;
180 : }
181 : }
182 : /* Defense line against incorrect plans */
183 477435 : if (actions) {
184 : // msg = chkTypes(cntxt->usermodule, mb, FALSE);
185 : // if (!msg)
186 : // msg = chkFlow(mb);
187 : // if (!msg)
188 : // msg = chkDeclarations(mb);
189 477435 : }
190 : /* keep actions taken as a fake argument */
191 477435 : (void) pushInt(mb, pci, actions);
192 477443 : return msg;
193 : }
|