Line data Source code
1 : /*
2 : * SPDX-License-Identifier: MPL-2.0
3 : *
4 : * This Source Code Form is subject to the terms of the Mozilla Public
5 : * License, v. 2.0. If a copy of the MPL was not distributed with this
6 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
7 : *
8 : * Copyright 2024 MonetDB Foundation;
9 : * Copyright August 2008 - 2023 MonetDB B.V.;
10 : * Copyright 1997 - July 2008 CWI.
11 : */
12 :
13 : #include "monetdb_config.h"
14 : #include "opt_commonTerms.h"
15 : #include "mal_exception.h"
16 : /*
17 : * Caveat. A lot of time was lost due to constants that are indistinguisable
18 : * at the surface level. It requires the constant optimizer to be ran first.
19 : */
20 :
21 : /* The key for finding common terms is that they share variables.
22 : * Therefore we skip all constants, except for a constant only situation.
23 : */
24 :
25 : /*
26 : * Speed up simple insert operations by skipping the common terms.
27 : */
28 :
29 : static inline bool __attribute__((__pure__))
30 240403 : isProjectConst(const InstrRecord *p)
31 : {
32 240403 : return (getModuleId(p) == algebraRef && getFunctionId(p) == projectRef);
33 : }
34 :
35 : static int __attribute__((__pure__))
36 13210535 : hashInstruction(const MalBlkRecord *mb, const InstrRecord *p)
37 : {
38 13210535 : int i;
39 35099956 : for (i = p->argc - 1; i >= p->retc; i--)
40 34757489 : if (!isVarConstant(mb, getArg(p, i)))
41 12868068 : return getArg(p, i);
42 342467 : if (isVarConstant(mb, getArg(p, p->retc)))
43 342467 : return p->retc;
44 : return -1;
45 : }
46 :
47 : str
48 483559 : OPTcommonTermsImplementation(Client cntxt, MalBlkPtr mb, MalStkPtr stk,
49 : InstrPtr pci)
50 : {
51 483559 : int i, j, k, barrier = 0, bailout = 0;
52 483559 : InstrPtr p, q;
53 483559 : int actions = 0;
54 483559 : int limit, slimit;
55 483559 : int duplicate;
56 483559 : int *alias = NULL;
57 483559 : int *hash = NULL, h;
58 483559 : int *list = NULL;
59 483559 : str msg = MAL_SUCCEED;
60 :
61 483559 : InstrPtr *old = NULL;
62 :
63 : /* catch simple insert operations */
64 483559 : if (isSimpleSQL(mb)) {
65 283882 : goto wrapup;
66 : }
67 :
68 199698 : (void) cntxt;
69 199698 : (void) stk;
70 199698 : alias = (int *) GDKzalloc(sizeof(int) * mb->vtop);
71 199703 : list = (int *) GDKzalloc(sizeof(int) * mb->stop);
72 199708 : hash = (int *) GDKzalloc(sizeof(int) * mb->vtop);
73 199707 : if (alias == NULL || list == NULL || hash == NULL) {
74 0 : msg = createException(MAL, "optimizer.commonTerms",
75 : SQLSTATE(HY013) MAL_MALLOC_FAIL);
76 0 : goto wrapup;
77 : }
78 :
79 199707 : old = mb->stmt;
80 199707 : limit = mb->stop;
81 199707 : slimit = mb->ssize;
82 199707 : if (newMalBlkStmt(mb, mb->ssize) < 0) {
83 0 : msg = createException(MAL, "optimizer.commonTerms",
84 : SQLSTATE(HY013) MAL_MALLOC_FAIL);
85 0 : old = NULL;
86 0 : goto wrapup;
87 : }
88 :
89 16450649 : for (i = 0; mb->errors == NULL && i < limit; i++) {
90 16450649 : p = old[i];
91 16450649 : duplicate = 0;
92 :
93 90375960 : for (k = 0; k < p->argc; k++)
94 73925311 : if (alias[getArg(p, k)])
95 300585 : getArg(p, k) = alias[getArg(p, k)];
96 :
97 16450649 : if (p->token == ENDsymbol) {
98 199652 : pushInstruction(mb, p);
99 199680 : old[i] = NULL;
100 199680 : break;
101 : }
102 : /*
103 : * Any barrier block signals the end of this optimizer,
104 : * because the impact of the block can affect the common code eliminated.
105 : */
106 32501994 : barrier |= (p->barrier == BARRIERsymbol || p->barrier == CATCHsymbol
107 16250997 : || p->barrier == RETURNsymbol);
108 : /*
109 : * Also block further optimization when you have seen an assert().
110 : * This works particularly for SQL, because it is not easy to track
111 : * the BAT identifier aliases to look for updates. The sql.assert
112 : * at least tells us that an update is planned.
113 : * Like all optimizer decisions, it is safe to stop.
114 : */
115 16250997 : barrier |= getFunctionId(p) == assertRef;
116 16250997 : if (barrier || p->token == ASSIGNsymbol) {
117 257590 : TRC_DEBUG(MAL_OPTIMIZER, "Skipped[%d]: %d %d\n", i, barrier,
118 : p->retc == p->argc);
119 257590 : pushInstruction(mb, p);
120 257611 : old[i] = NULL;
121 257611 : continue;
122 : }
123 :
124 : /* when we enter a barrier block, we should ditch all previous instructions from consideration */
125 15993407 : if (p->barrier == BARRIERsymbol || p->barrier == CATCHsymbol
126 15993407 : || p->barrier == RETURNsymbol) {
127 0 : memset(list, 0, sizeof(int) * mb->stop);
128 0 : memset(hash, 0, sizeof(int) * mb->vtop);
129 : }
130 : /* side-effect producing operators can never be replaced */
131 : /* the same holds for function calls without an argument, it is
132 : * unclear where the results comes from (e.g. clock()) */
133 15993407 : if (mayhaveSideEffects(cntxt, mb, p, TRUE) || p->argc == p->retc) {
134 1667795 : TRC_DEBUG(MAL_OPTIMIZER, "Skipped[%d] side-effect: %d\n", i,
135 : p->retc == p->argc);
136 1667795 : pushInstruction(mb, p);
137 1667815 : old[i] = NULL;
138 1667815 : continue;
139 : }
140 : /* simple SQL bind operations need not be merged, they are cheap
141 : * and/or can be duplicated eliminated elsewhere cheaper */
142 14326438 : if (getModuleId(p) == sqlRef && (getFunctionId(p) != tidRef && getFunctionId(p) != bindRef)) {
143 909223 : pushInstruction(mb, p);
144 909223 : old[i] = NULL;
145 909223 : continue;
146 : }
147 13417215 : if (getModuleId(p) == matRef) { /* mat.packIncrement has requirement on number of instructions (or that needs an update */
148 207154 : pushInstruction(mb, p);
149 207154 : old[i] = NULL;
150 207154 : continue;
151 : }
152 :
153 : /* from here we have a candidate to look for a match */
154 :
155 13210061 : h = hashInstruction(mb, p);
156 :
157 13210061 : TRC_DEBUG(MAL_OPTIMIZER, "Candidate[%d] look at list[%d] => %d\n", i, h,
158 : hash[h]);
159 13210061 : traceInstruction(MAL_OPTIMIZER, mb, 0, p, LIST_MAL_ALL);
160 :
161 13210188 : if (h < 0) {
162 0 : pushInstruction(mb, p);
163 0 : old[i] = NULL;
164 0 : continue;
165 : }
166 :
167 13210188 : bailout = 1024; // don't run over long collision list
168 : /* Look into the hash structure for matching instructions */
169 112398427 : for (j = hash[h]; j > 0 && bailout-- > 0; j = list[j]) {
170 99409836 : if ((q = getInstrPtr(mb, j))
171 99409836 : && getFunctionId(q) == getFunctionId(p)
172 73912875 : && getModuleId(q) == getModuleId(p)) {
173 73916462 : TRC_DEBUG(MAL_OPTIMIZER,
174 : "Candidate[%d->%d] %d %d :%d %d %d=%d %d %d %d\n", j,
175 : list[j], hasSameSignature(mb, p, q),
176 : hasSameArguments(mb, p, q), q->token != ASSIGNsymbol,
177 : list[getArg(q, q->argc - 1)], i, !hasCommonResults(p,
178 : q),
179 : !isUnsafeFunction(q), !isUpdateInstruction(q),
180 : isLinearFlow(q));
181 73916462 : traceInstruction(MAL_OPTIMIZER, mb, 0, q, LIST_MAL_ALL);
182 :
183 : /*
184 : * Simple assignments are not replaced either. They should be
185 : * handled by the alias removal part. All arguments should
186 : * be assigned their value before instruction p.
187 : */
188 73921230 : if (hasSameArguments(mb, p, q)
189 240421 : && hasSameSignature(mb, p, q)
190 240403 : && !hasCommonResults(p, q)
191 240403 : && !isUnsafeFunction(q)
192 240403 : && !isUpdateInstruction(q)
193 240403 : && !isProjectConst(q) && /* disable project(x,val), as its used for the result of case statements */
194 219320 : isLinearFlow(q)) {
195 219320 : if (safetyBarrier(p, q)) {
196 0 : TRC_DEBUG(MAL_OPTIMIZER, "Safety barrier reached\n");
197 : break;
198 : }
199 219320 : duplicate = 1;
200 219320 : clrFunction(p);
201 219320 : p->argc = p->retc;
202 450494 : for (k = 0; k < q->retc; k++) {
203 231174 : alias[getArg(p, k)] = getArg(q, k);
204 : /* we know the arguments fit so the instruction can safely be patched */
205 231174 : p = pushArgument(mb, p, getArg(q, k));
206 : }
207 :
208 219320 : TRC_DEBUG(MAL_OPTIMIZER, "Modified expression %d -> %d ",
209 : getArg(p, 0), getArg(p, 1));
210 219320 : traceInstruction(MAL_OPTIMIZER, mb, 0, p, LIST_MAL_ALL);
211 :
212 219320 : actions++;
213 219320 : break; /* end of search */
214 : }
215 25493374 : } else if (isUpdateInstruction(p)) {
216 0 : TRC_DEBUG(MAL_OPTIMIZER, "Skipped: %d %d\n",
217 : mayhaveSideEffects(cntxt, mb, q, TRUE),
218 : isUpdateInstruction(p));
219 0 : traceInstruction(MAL_OPTIMIZER, mb, 0, q, LIST_MAL_ALL);
220 : }
221 : }
222 :
223 13207911 : if (duplicate) {
224 219320 : pushInstruction(mb, p);
225 219320 : old[i] = NULL;
226 219320 : continue;
227 : }
228 : /* update the hash structure with another candidate for reuse */
229 12988591 : TRC_DEBUG(MAL_OPTIMIZER,
230 : "Update hash[%d] - look at arg '%d' hash '%d' list '%d'\n", i,
231 : getArg(p, p->argc - 1), h, hash[h]);
232 12988591 : traceInstruction(MAL_OPTIMIZER, mb, 0, p, LIST_MAL_ALL);
233 :
234 12989189 : if (!mayhaveSideEffects(cntxt, mb, p, TRUE) && p->argc != p->retc
235 25979419 : && isLinearFlow(p) && !isUnsafeFunction(p)
236 12989792 : && !isUpdateInstruction(p)) {
237 12989543 : list[i] = hash[h];
238 12989543 : hash[h] = i;
239 12989543 : pushInstruction(mb, p);
240 12990114 : old[i] = NULL;
241 : }
242 : }
243 46188226 : for (; i < slimit; i++)
244 45988519 : if (old[i])
245 5056967 : pushInstruction(mb, old[i]);
246 : /* Defense line against incorrect plans */
247 199707 : if (actions > 0) {
248 10069 : msg = chkTypes(cntxt->usermodule, mb, FALSE);
249 10069 : if (!msg)
250 10069 : msg = chkFlow(mb);
251 10069 : if (!msg)
252 10069 : msg = chkDeclarations(mb);
253 : }
254 189638 : wrapup:
255 : /* keep actions taken as a fake argument */
256 483589 : (void) pushInt(mb, pci, actions);
257 :
258 483589 : if (alias)
259 199707 : GDKfree(alias);
260 483589 : if (list)
261 199707 : GDKfree(list);
262 483596 : if (hash)
263 199714 : GDKfree(hash);
264 483569 : if (old)
265 199687 : GDKfree(old);
266 483596 : return msg;
267 : }
|