LCOV - code coverage report
Current view: top level - monetdb5/mal - mal_function.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 309 421 73.4 %
Date: 2025-03-25 21:27:32 Functions: 12 17 70.6 %

          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, 2025 MonetDB Foundation;
       9             :  * Copyright August 2008 - 2023 MonetDB B.V.;
      10             :  * Copyright 1997 - July 2008 CWI.
      11             :  */
      12             : 
      13             : /*
      14             :  * (author) M. Kersten
      15             :  * For documentation see website
      16             :  */
      17             : #include "monetdb_config.h"
      18             : #include "mal_function.h"
      19             : #include "mal_resolve.h"              /* for isPolymorphic() & chkProgram() */
      20             : #include "mal_interpreter.h"  /* for showErrors() */
      21             : #include "mal_listing.h"
      22             : #include "mal_namespace.h"
      23             : #include "mal_private.h"
      24             : 
      25             : Symbol
      26       50642 : newFunctionArgs(const char *mod, const char *nme, int kind, int args)
      27             : {
      28       50642 :         Symbol s;
      29             : 
      30       50642 :         if (mod == NULL || nme == NULL)
      31             :                 return NULL;
      32             : 
      33       50642 :         s = newSymbol(nme, kind);
      34       50642 :         if (s == NULL)
      35             :                 return NULL;
      36             : 
      37       50642 :         if (kind == FUNCTIONsymbol) {
      38       50629 :                 int varid = newVariable(s->def, nme, strlen(nme), TYPE_any);
      39       50622 :                 if (varid < 0) {
      40           0 :                         freeSymbol(s);
      41           0 :                         return NULL;
      42             :                 }
      43             : 
      44       50622 :                 if (args > 0) {
      45       50611 :                         InstrPtr p = newInstructionArgs(NULL, mod, nme, args);
      46       50616 :                         if (p == NULL) {
      47           0 :                                 freeSymbol(s);
      48           0 :                                 return NULL;
      49             :                         }
      50       50616 :                         p->token = kind;
      51       50616 :                         p->barrier = 0;
      52       50616 :                         setDestVar(p, varid);
      53       50616 :                         pushInstruction(s->def, p);
      54       50612 :                         if (s->def->errors) {
      55           0 :                                 freeSymbol(s);
      56           0 :                                 return NULL;
      57             :                         }
      58             :                 }
      59             :         }
      60             :         return s;
      61             : }
      62             : 
      63             : Symbol
      64       49407 : newFunction(const char *mod, const char *nme, int kind)
      65             : {
      66       49407 :         return newFunctionArgs(mod, nme, kind, MAXARG);
      67             : }
      68             : 
      69             : int
      70      174899 : getPC(MalBlkPtr mb, InstrPtr p)
      71             : {
      72      174899 :         int i;
      73     1955517 :         for (i = 0; i < mb->stop; i++)
      74     1955524 :                 if (getInstrPtr(mb, i) == p)
      75      174906 :                         return i;
      76             :         return -1;
      77             : }
      78             : 
      79             : /*
      80             :  * Checking the control flow structure is done by a single pass over the
      81             :  * MAL program after the program has been type-checked.
      82             :  * It should inspect all BARRIER and CATCH blocks for proper structure.
      83             :  * If the flow is correct and not dependent on an undefined typed instruction
      84             :  * we avoid doing this check any further.
      85             :  */
      86             : #define DEPTH 128
      87             : 
      88             : str
      89     3601364 : chkFlow(MalBlkPtr mb)
      90             : {
      91     3601364 :         int i, j, k, v, lastInstruction;
      92     3601364 :         int pc[DEPTH];
      93     3601364 :         int var[DEPTH];
      94     3601364 :         InstrPtr stmt[DEPTH];
      95     3601364 :         int btop = 0;
      96     3601364 :         int endseen = 0, retseen = 0;
      97     3601364 :         InstrPtr p, sig;
      98     3601364 :         str msg = MAL_SUCCEED;
      99     3601364 :         char name[IDLENGTH];
     100             : 
     101     3601364 :         if (mb->errors != MAL_SUCCEED)
     102             :                 return mb->errors;
     103     3603143 :         sig = getInstrPtr(mb, 0);
     104     3603143 :         lastInstruction = mb->stop - 1;
     105   244378608 :         for (i = 0; i < mb->stop; i++) {
     106   240775478 :                 p = getInstrPtr(mb, i);
     107             :                 /* we have to keep track on the maximal arguments/block
     108             :                    because it is needed by the interpreter */
     109   240775478 :                 switch (p->barrier) {
     110      278277 :                 case BARRIERsymbol:
     111             :                 case CATCHsymbol:
     112      278277 :                         if (btop == DEPTH)
     113           0 :                                 throw(MAL, "chkFlow", "%s.%s Too many nested MAL blocks",
     114             :                                           getModuleId(sig), getFunctionId(sig));
     115      278277 :                         pc[btop] = i;
     116      278277 :                         v = var[btop] = getDestVar(p);
     117      278277 :                         stmt[btop] = p;
     118             : 
     119      278999 :                         for (j = btop - 1; j >= 0; j--)
     120         722 :                                 if (v == var[j])
     121           0 :                                         throw(MAL, "chkFlow",
     122             :                                                   "%s.%s recursive %s[%d] shields %s[%d]",
     123             :                                                   getModuleId(sig), getFunctionId(sig), getVarNameIntoBuffer(mb, v, name),
     124           0 :                                                   pc[j], getFcnName(mb), pc[i]);
     125             : 
     126      278277 :                         btop++;
     127      278277 :                         break;
     128      278433 :                 case EXITsymbol:
     129      278433 :                         v = getDestVar(p);
     130      278433 :                         if (btop > 0 && var[btop - 1] != v) {
     131           3 :                                 char name2[IDLENGTH];
     132           3 :                                 throw(MAL, "chkFlow",
     133             :                                           "%s.%s exit-label '%s' does not match '%s'",
     134             :                                           getModuleId(sig), getFunctionId(sig), getVarNameIntoBuffer(mb, v, name),
     135             :                                           getVarNameIntoBuffer(mb, var[btop - 1], name2));
     136             :                         }
     137      278430 :                         if (btop == 0)
     138           0 :                                 throw(MAL, "chkFlow",
     139             :                                           "%s.%s exit-label '%s' without begin-label",
     140             :                                           getModuleId(sig), getFunctionId(sig), getVarNameIntoBuffer(mb, v, name));
     141             :                         /* search the matching block */
     142      278430 :                         for (j = btop - 1; j >= 0; j--)
     143      278382 :                                 if (var[j] == v)
     144             :                                         break;
     145      278430 :                         if (j >= 0)
     146             :                                 btop = j;
     147             :                         else
     148           0 :                                 btop--;
     149             : 
     150             :                         /* retrofit LEAVE/REDO instructions */
     151      278430 :                         stmt[btop]->jump = i;
     152    28246357 :                         for (k = pc[btop]; k < i; k++) {
     153    27967927 :                                 InstrPtr p1 = getInstrPtr(mb, k);
     154    27967927 :                                 if (getDestVar(p1) == v) {
     155             :                                         /* handle assignments with leave/redo option */
     156      287729 :                                         if (p1->barrier == LEAVEsymbol)
     157         607 :                                                 p1->jump = i;
     158      287729 :                                         if (p1->barrier == REDOsymbol)
     159        9001 :                                                 p1->jump = pc[btop] + 1;
     160             :                                 }
     161             :                         }
     162             :                         break;
     163        9622 :                 case LEAVEsymbol:
     164             :                 case REDOsymbol:
     165        9622 :                         v = getDestVar(p);
     166        9631 :                         for (j = btop - 1; j >= 0; j--)
     167        9623 :                                 if (var[j] == v)
     168             :                                         break;
     169             :                         if (j < 0) {
     170           8 :                                 throw(MAL, "chkFlow", "%s.%s label '%s' not in guarded block",
     171             :                                           getModuleId(sig), getFunctionId(sig),
     172             :                                           getVarNameIntoBuffer(mb, v, name));
     173             :                         }
     174             :                         break;
     175        3634 :                 case RETURNsymbol: {
     176        3634 :                         InstrPtr ps = getInstrPtr(mb, 0);
     177        3634 :                         int e;
     178        3634 :                         if (ps->retc != p->retc) {
     179           2 :                                 throw(MAL, "chkFlow", "%s.%s invalid return target!",
     180             :                                           getModuleId(sig), getFunctionId(sig));
     181        3632 :                         } else if (ps->typeresolved)
     182       22005 :                                 for (e = 0; e < p->retc; e++) {
     183       18382 :                                         if (resolvedType(getArgType(mb, ps, e), getArgType(mb, p, e)) < 0) {
     184           0 :                                                 str tpname = getTypeName(getArgType(mb, p, e));
     185           0 :                                                 msg = createException(MAL,
     186             :                                                                                           "%s.%s RETURN type mismatch at type '%s'\n",
     187           0 :                                                                                           getModuleId(p) ? getModuleId(p) :
     188             :                                                                                           "",
     189           0 :                                                                                           getFunctionId(p) ?
     190             :                                                                                           getFunctionId(p) : "", tpname);
     191           0 :                                                 GDKfree(tpname);
     192           0 :                                                 return msg;
     193             :                                         }
     194             :                                 }
     195             :                         //if (btop == 0)
     196             :                         retseen = 1;
     197             :                         break;
     198             :                 }
     199             :                 case RAISEsymbol:
     200             :                         endseen = 1;
     201             :                         break;
     202             :                 case ENDsymbol:
     203             :                         endseen = 1;
     204             :                         break;
     205   240204869 :                 default:
     206   240204869 :                         if (isaSignature(p)) {
     207    33847478 :                                 if (p->token == REMsymbol) {
     208             :                                         /* do nothing */
     209     3602530 :                                 } else if (i) {
     210           0 :                                         str l = instruction2str(mb, 0, p, TRUE);
     211           0 :                                         msg = createException(MAL, "%s.%s signature misplaced\n!%s",
     212             :                                                                                   getModuleId(p), getFunctionId(p), l);
     213           0 :                                         GDKfree(l);
     214           0 :                                         return msg;
     215             :                                 }
     216             :                         }
     217             :                 }
     218             :         }
     219             : 
     220     3603130 :         if (lastInstruction < mb->stop - 1)
     221           0 :                 throw(MAL, "chkFlow", "%s.%s instructions after END", getModuleId(sig),
     222             :                           getFunctionId(sig));
     223             : 
     224     3603130 :         if (endseen && btop > 0)
     225           0 :                 throw(MAL, "chkFlow", "barrier '%s' without exit in %s[%d]",
     226           0 :                           getVarNameIntoBuffer(mb, var[btop - 1], name), getFcnName(mb), i);
     227     3603130 :         p = getInstrPtr(mb, 0);
     228     3603130 :         if (!isaSignature(p))
     229           0 :                 throw(MAL, "chkFlow", "%s.%s signature missing", getModuleId(sig),
     230             :                           getFunctionId(sig));
     231     3603130 :         if (retseen == 0) {
     232     3600503 :                 if (getArgType(mb, p, 0) != TYPE_void && (p->token == FUNCTIONsymbol))
     233           0 :                         throw(MAL, "chkFlow", "%s.%s RETURN missing", getModuleId(sig),
     234             :                                   getFunctionId(sig));
     235             :         }
     236             :         return MAL_SUCCEED;
     237             : }
     238             : 
     239             : /*
     240             :  * A code may contain temporary names for marking barrier blocks.
     241             :  * Since they are introduced by the compiler, the parser should locate
     242             :  * them itself when encountering the LEAVE,EXIT,REDO.
     243             :  * The starting position is mostly the last statement entered.
     244             :  * Purposely, the nameless envelops searches the name of the last
     245             :  * unclosed block. All others are ignored.
     246             :  */
     247             : int
     248           4 : getBarrierEnvelop(MalBlkPtr mb)
     249             : {
     250           4 :         int pc;
     251           4 :         InstrPtr p;
     252          10 :         for (pc = mb->stop - 2; pc >= 0; pc--) {
     253           9 :                 p = getInstrPtr(mb, pc);
     254           9 :                 if (blockExit(p)) {
     255           1 :                         int l = p->argv[0];
     256           4 :                         for (; pc >= 0; pc--) {
     257           4 :                                 p = getInstrPtr(mb, pc);
     258           4 :                                 if (blockStart(p) && p->argv[0] == l)
     259             :                                         break;
     260             :                         }
     261           1 :                         continue;
     262             :                 }
     263           8 :                 if (blockStart(p))
     264           3 :                         return p->argv[0];
     265             :         }
     266           1 :         return newTmpVariable(mb, TYPE_any);
     267             : }
     268             : 
     269             : static void
     270          14 : replaceTypeVar(MalBlkPtr mb, InstrPtr p, int v, malType t)
     271             : {
     272          83 :         for (int j = 0; j < mb->stop; j++) {
     273          69 :                 p = getInstrPtr(mb, j);
     274          69 :                 if (p->polymorphic) {
     275          94 :                         for (int i = 0; i < p->argc; i++) {
     276          68 :                                 int x = getArgType(mb, p, i);
     277          68 :                                 if (isPolymorphic(x) && getTypeIndex(x) == v) {
     278          33 :                                         if (isaBatType(x)) {
     279          17 :                                                 int tail = newBatType(t);
     280          17 :                                                 setArgType(mb, p, i, tail);
     281             :                                         } else {
     282          16 :                                                 setArgType(mb, p, i, t);
     283             :                                         }
     284             :                                 }
     285             :                         }
     286             :                 }
     287             :         }
     288          14 : }
     289             : 
     290             : /* insert a symbol into the symbol table just before the symbol
     291             :  * "before". */
     292             : static void
     293          11 : insertSymbolBefore(Module scope, Symbol prg, Symbol before)
     294             : {
     295          11 :         int t;
     296          11 :         Symbol s;
     297             : 
     298          11 :         assert(strcmp(prg->name, before->name) == 0);
     299          11 :         t = getSymbolIndex(prg->name);
     300          11 :         assert(scope->space != NULL);
     301          11 :         assert(scope->space[t] != NULL);
     302          11 :         s = scope->space[t];
     303          11 :         prg->skip = before->skip;
     304          11 :         prg->peer = before;
     305          11 :         if (s == before) {
     306           9 :                 scope->space[t] = prg;
     307             :         } else {
     308           3 :                 for (;;) {
     309           3 :                         assert(s != NULL);
     310           3 :                         if (s->skip == before) {
     311           0 :                                 s->skip = prg;
     312             :                         }
     313           3 :                         if (s->peer == before) {
     314           2 :                                 s->peer = prg;
     315           2 :                                 break;
     316             :                         }
     317             :                         s = s->peer;
     318             :                 }
     319             :         }
     320          11 : }
     321             : 
     322             : /*
     323             :  * Upon cloning a function we should remove all the polymorphic flags.
     324             :  * Otherwise we may end up with a recursive clone.
     325             :  */
     326             : Symbol
     327          11 : cloneFunction(Module scope, Symbol proc, MalBlkPtr mb, InstrPtr p)
     328             : {
     329          11 :         Symbol new;
     330          11 :         int i, v;
     331          11 :         InstrPtr pp;
     332          11 :         str msg = MAL_SUCCEED;
     333             : 
     334          11 :         new = newFunctionArgs(scope->name, proc->name, proc->kind, -1);
     335          11 :         if (new == NULL) {
     336             :                 return NULL;
     337             :         }
     338          11 :         freeMalBlk(new->def);
     339          11 :         if ((new->def = copyMalBlk(proc->def)) == NULL) {
     340           0 :                 freeSymbol(new);
     341           0 :                 return NULL;
     342             :         }
     343             :         /* now change the definition of the original proc */
     344             :         /* check for errors after fixation , TODO */
     345          11 :         pp = getSignature(new);
     346          43 :         for (i = 0; i < pp->argc; i++)
     347          32 :                 if (isPolymorphic(v = getArgType(new->def, pp, i))) {
     348          14 :                         int t = getArgType(mb, p, i);
     349             : 
     350          14 :                         if (v == TYPE_any) {
     351             :                                 assert(0);
     352             :                                 replaceTypeVar(new->def, pp, v, t);
     353             :                         }
     354          14 :                         if (isaBatType(v)) {
     355          14 :                                 if (getTypeIndex(v))
     356          14 :                                         replaceTypeVar(new->def, pp, getTypeIndex(v), getBatType(t));
     357             :                         } else
     358           0 :                                 replaceTypeVar(new->def, pp, getTypeIndex(v), t);
     359             :                 }
     360             :         /* include the function at the proper place in the scope */
     361          11 :         insertSymbolBefore(scope, new, proc);
     362             :         /* clear polymorphic and type to force analysis */
     363          76 :         for (i = 0; i < new->def->stop; i++) {
     364          54 :                 pp = getInstrPtr(new->def, i);
     365          54 :                 pp->typeresolved = false;
     366          54 :                 pp->polymorphic = 0;
     367             :         }
     368             :         /* clear type fixations */
     369          85 :         for (i = 0; i < new->def->vtop; i++)
     370          74 :                 clrVarFixed(new->def, i);
     371             : 
     372             : 
     373             :         /* check for errors after fixation , TODO */
     374             :         /* beware, we should now ignore any cloning */
     375          11 :         if (proc->def->errors == 0) {
     376          11 :                 msg = chkProgram(scope, new->def);
     377          11 :                 if (msg)
     378           3 :                         mb->errors = msg;
     379           8 :                 else if (new->def->errors) {
     380           0 :                         assert(mb->errors == NULL);
     381           0 :                         mb->errors = new->def->errors;
     382           0 :                         mb->errors = createMalException(mb, 0, TYPE, "Error in cloned function");
     383           0 :                         new->def->errors = 0;
     384             :                 }
     385             :         }
     386             : 
     387             :         return new;
     388             : }
     389             : 
     390             : /*
     391             :  * For commands we do not have to clone the routine. We merely have to
     392             :  * assure that the type-constraints are obeyed. The resulting type
     393             :  * is returned.
     394             :  */
     395             : void
     396           0 : debugFunction(stream *fd, MalBlkPtr mb, MalStkPtr stk, int flg, int first,
     397             :                           int step)
     398             : {
     399           0 :         int i, j;
     400           0 :         str ps;
     401           0 :         InstrPtr p;
     402             : 
     403           0 :         if (mb == NULL) {
     404           0 :                 mnstr_printf(fd, "# function definition missing\n");
     405           0 :                 return;
     406             :         }
     407           0 :         if (flg == 0 || step < 0 || first < 0)
     408             :                 return;
     409             : 
     410           0 :         if (mb->errors)
     411           0 :                 mnstr_printf(fd, "#errors seen: %s\n", mb->errors);
     412           0 :         for (i = first; i < first + step && i < mb->stop; i++) {
     413           0 :                 ps = instruction2str(mb, stk, (p = getInstrPtr(mb, i)), flg);
     414           0 :                 if (ps) {
     415           0 :                         if (p->token == REMsymbol)
     416           0 :                                 mnstr_printf(fd, "%-40s\n", ps);
     417             :                         else {
     418           0 :                                 mnstr_printf(fd, "%-40s\t#[%d] %s ", ps, i,
     419           0 :                                                          (p->blk ? p->blk->binding : ""));
     420           0 :                                 if (flg & LIST_MAL_FLOW) {
     421           0 :                                         for (j = 0; j < p->retc; j++)
     422           0 :                                                 mnstr_printf(fd, "%d ", getArg(p, j));
     423           0 :                                         if (p->argc - p->retc > 0)
     424           0 :                                                 mnstr_printf(fd, "<- ");
     425           0 :                                         for (; j < p->argc; j++)
     426           0 :                                                 mnstr_printf(fd, "%d ", getArg(p, j));
     427             :                                 }
     428           0 :                                 mnstr_printf(fd, "\n");
     429             :                         }
     430           0 :                         GDKfree(ps);
     431             :                 } else
     432           0 :                         mnstr_printf(fd, "#failed instruction2str()\n");
     433             :         }
     434             : }
     435             : 
     436             : void
     437          58 : listFunction(stream *fd, MalBlkPtr mb, MalStkPtr stk, int flg, int first,
     438             :                          int size)
     439             : {
     440          58 :         int i;
     441          58 :         int sample = 256;
     442             : 
     443          58 :         if (mb == NULL) {
     444           0 :                 mnstr_printf(fd, "# function definition missing\n");
     445           0 :                 return;
     446             :         }
     447          58 :         if (flg == 0)
     448             :                 return;
     449             : 
     450          58 :         assert(size >= 0);
     451          58 :         assert(first >= 0 && first < mb->stop);
     452          58 :         if (flg & LIST_MAL_MAPI) {
     453          54 :                 size_t len = 0;
     454          54 :                 str ps;
     455          54 :                 mnstr_printf(fd, "&1 0 %d 1 %d\n",        /* type id rows columns tuples */
     456             :                                          mb->stop, mb->stop);
     457          54 :                 mnstr_printf(fd, "%% .explain # table_name\n");
     458          54 :                 mnstr_printf(fd, "%% mal # name\n");
     459          54 :                 mnstr_printf(fd, "%% clob # type\n");
     460        2759 :                 for (i = first; i < first + size && i < mb->stop && sample-- > 0; i++) {
     461        2651 :                         ps = instruction2str(mb, stk, getInstrPtr(mb, i), flg);
     462        2651 :                         if (ps) {
     463        2651 :                                 size_t l = strlen(ps);
     464        2651 :                                 if (l > len)
     465             :                                         len = l;
     466        2651 :                                 GDKfree(ps);
     467             :                         } else
     468           0 :                                 mnstr_printf(fd, "#failed instruction2str()\n");
     469             :                 }
     470          54 :                 mnstr_printf(fd, "%% %zu # length\n", len);
     471             :         }
     472        2749 :         for (i = first; i < first + size && i < mb->stop; i++)
     473        2691 :                 printInstruction(fd, mb, stk, getInstrPtr(mb, i), flg);
     474             : }
     475             : 
     476             : 
     477             : void
     478          58 : printFunction(stream *fd, MalBlkPtr mb, MalStkPtr stk, int flg)
     479             : {
     480          58 :         int i, j;
     481          58 :         InstrPtr p;
     482             : 
     483             : 
     484             :         // Set the used bits properly
     485        5953 :         for (i = 0; i < mb->vtop; i++)
     486        5895 :                 clrVarUsed(mb, i);
     487             : 
     488             : 
     489        2749 :         for (i = 0; i < mb->stop; i++) {
     490        2691 :                 p = getInstrPtr(mb, i);
     491        8883 :                 for (j = p->retc; j < p->argc; j++)
     492        6192 :                         setVarUsed(mb, getArg(p, j));
     493        2691 :                 if (p->barrier)
     494          24 :                         for (j = 0; j < p->retc; j++)
     495          12 :                                 setVarUsed(mb, getArg(p, j));
     496             :         }
     497          58 :         listFunction(fd, mb, stk, flg, 0, mb->stop);
     498          58 : }
     499             : 
     500             : /* initialize the static scope boundaries for all variables */
     501             : void
     502     2884404 : setVariableScope(MalBlkPtr mb)
     503             : {
     504     2884404 :         int pc, k, depth = 0, dflow = -1;
     505     2884404 :         InstrPtr p;
     506             : 
     507             :         /* reset the scope admin */
     508   318358678 :         for (k = 0; k < mb->vtop; k++)
     509   315474274 :                 if (isVarConstant(mb, k)) {
     510    85362156 :                         setVarScope(mb, k, 0);
     511    85362156 :                         setVarDeclared(mb, k, 0);
     512    85362156 :                         setVarUpdated(mb, k, 0);
     513    85362156 :                         setVarEolife(mb, k, mb->stop);
     514             :                 } else {
     515   230112118 :                         setVarScope(mb, k, 0);
     516   230112118 :                         setVarDeclared(mb, k, 0);
     517   230112118 :                         setVarUpdated(mb, k, 0);
     518   230112118 :                         setVarEolife(mb, k, 0);
     519             :                 }
     520             : 
     521   166533266 :         for (pc = 0; pc < mb->stop; pc++) {
     522   163648862 :                 p = getInstrPtr(mb, pc);
     523             : 
     524   163648862 :                 if (blockStart(p)) {
     525      492927 :                         if (getModuleId(p) && getFunctionId(p)
     526      488376 :                                 && strcmp(getModuleId(p), "language") == 0
     527      475704 :                                 && strcmp(getFunctionId(p), "dataflow") == 0) {
     528      475728 :                                 if (dflow != -1)
     529           0 :                                         addMalException(mb,
     530             :                                                                         "setLifeSpan nested dataflow blocks not allowed");
     531             :                                 dflow = depth;
     532             :                         } else
     533       17199 :                                 depth++;
     534             :                 }
     535             : 
     536   727891239 :                 for (k = 0; k < p->argc; k++) {
     537   564242377 :                         int v = getArg(p, k);
     538   564242377 :                         if (isVarConstant(mb, v) && getVarUpdated(mb, v) == 0)
     539    77660219 :                                 setVarUpdated(mb, v, pc);
     540             : 
     541   564242377 :                         if (getVarDeclared(mb, v) == 0) {
     542   244292646 :                                 setVarDeclared(mb, v, pc);
     543   244292646 :                                 setVarScope(mb, v, depth);
     544             :                         }
     545   564242377 :                         if (k < p->retc)
     546   167355962 :                                 setVarUpdated(mb, v, pc);
     547   564242377 :                         if (getVarScope(mb, v) == depth)
     548   564123588 :                                 setVarEolife(mb, v, pc);
     549             : 
     550   564242377 :                         if (k >= p->retc && getVarScope(mb, v) < depth)
     551       73723 :                                 setVarEolife(mb, v, -1);
     552             :                 }
     553             :                 /*
     554             :                  * At a block exit we can finalize all variables defined within that block.
     555             :                  * This does not hold for dataflow blocks. They merely direct the execution
     556             :                  * thread, not the syntactic scope.
     557             :                  */
     558   163648862 :                 if (blockExit(p)) {
     559   162058635 :                         for (k = 0; k < mb->vtop; k++)
     560   161565757 :                                 if (getVarEolife(mb, k) == 0 && getVarScope(mb, k) == depth)
     561    50683501 :                                         setVarEolife(mb, k, pc);
     562   110882256 :                                 else if (getVarEolife(mb, k) == -1)
     563       44950 :                                         setVarEolife(mb, k, pc);
     564             : 
     565      492878 :                         if (dflow == depth)
     566             :                                 dflow = -1;
     567             :                         else
     568       17116 :                                 depth--;
     569             :                 }
     570   163648862 :                 if (blockReturn(p)) {
     571       26640 :                         for (k = 0; k < p->argc; k++)
     572       23994 :                                 setVarEolife(mb, getArg(p, k), pc);
     573             :                 }
     574             :         }
     575   318598641 :         for (k = 0; k < mb->vtop; k++)
     576   315714237 :                 if (getVarEolife(mb, k) == 0)
     577    27454297 :                         setVarEolife(mb, k, mb->stop - 1);
     578     2884404 : }
     579             : 
     580             : int
     581           0 : isLoopBarrier(MalBlkPtr mb, int pc)
     582             : {
     583           0 :         InstrPtr p;
     584           0 :         int varid;
     585           0 :         p = getInstrPtr(mb, pc);
     586           0 :         if (p->barrier != BARRIERsymbol)
     587             :                 return 0;
     588           0 :         varid = getDestVar(p);
     589           0 :         for (pc++; pc < mb->stop; pc++) {
     590           0 :                 p = getInstrPtr(mb, pc);
     591           0 :                 if (p->barrier == REDOsymbol && getDestVar(p) == varid)
     592             :                         return 1;
     593           0 :                 if (p->barrier == EXITsymbol && getDestVar(p) == varid)
     594             :                         break;
     595             :         }
     596             :         return 0;
     597             : }
     598             : 
     599             : int
     600           0 : getBlockBegin(MalBlkPtr mb, int pc)
     601             : {
     602           0 :         InstrPtr p;
     603           0 :         int varid = 0, i;
     604             : 
     605           0 :         for (i = pc; i < mb->stop; i++) {
     606           0 :                 p = getInstrPtr(mb, i);
     607           0 :                 if (p->barrier == EXITsymbol) {
     608           0 :                         varid = getDestVar(p);
     609           0 :                         break;
     610             :                 }
     611             :         }
     612           0 :         if (i == mb->stop)
     613             :                 return 0;
     614             : 
     615           0 :         for (; pc > 0; pc--) {
     616           0 :                 p = getInstrPtr(mb, pc);
     617           0 :                 if ((p->barrier == BARRIERsymbol || p->barrier == CATCHsymbol) &&
     618           0 :                         getDestVar(p) == varid)
     619           0 :                         return pc;
     620             :         }
     621             :         return 0;
     622             : }
     623             : 
     624             : int
     625           0 : getBlockExit(MalBlkPtr mb, int pc)
     626             : {
     627           0 :         InstrPtr p;
     628           0 :         int varid;
     629           0 :         p = getInstrPtr(mb, pc);
     630           0 :         if (p->barrier != BARRIERsymbol && p->barrier != CATCHsymbol)
     631             :                 return 0;
     632           0 :         varid = getDestVar(p);
     633           0 :         for (pc++; pc < mb->stop; pc++) {
     634           0 :                 p = getInstrPtr(mb, pc);
     635           0 :                 if (p->barrier == EXITsymbol && getDestVar(p) == varid)
     636           0 :                         return pc;
     637             :         }
     638             :         return 0;
     639             : }
     640             : 
     641             : /*
     642             :  * Variable declaration
     643             :  * Variables are implicitly declared upon first use.
     644             :  * This feature may become a source of runtime errors and
     645             :  * complicates the analyse during optimization.
     646             :  * Therefore, in line with the flow of control check,
     647             :  * we make sure that all variables are properly initialized
     648             :  * before being used. Since barrier blocks may be skipped at
     649             :  * runtime, they actually introduce a separate scope.
     650             :  * Variables declared within a block may not be used outside it.
     651             :  * Variables can only be declared once.
     652             :  *
     653             :  * In many situation chkFlow and chkDeclarations should be called
     654             :  * together. Moreover, an erroneous chkFlow most likely implies
     655             :  * errors in the declarations as well.
     656             :  *
     657             :  * Since in interactive mode each statement is handled separately,
     658             :  * we have to remember the scope assigned to a variable.
     659             :  */
     660             : void
     661           0 : clrDeclarations(MalBlkPtr mb)
     662             : {
     663           0 :         int i;
     664           0 :         for (i = 0; i < mb->vtop; i++) {
     665           0 :                 clrVarInit(mb, i);
     666           0 :                 clrVarUsed(mb, i);
     667           0 :                 clrVarDisabled(mb, i);
     668             :         }
     669           0 : }
     670             : 
     671             : str
     672     3603484 : chkDeclarations(MalBlkPtr mb)
     673             : {
     674     3603484 :         int pc, i, k, l;
     675     3603484 :         InstrPtr p, sig;
     676     3603484 :         short blks[MAXDEPTH], top = 0, blkId = 1;
     677     3603484 :         int dflow = -1;
     678     3603484 :         str msg = MAL_SUCCEED;
     679     3603484 :         char name[IDLENGTH];
     680             : 
     681     3603484 :         if (mb->errors)
     682           0 :                 return GDKstrdup(mb->errors);
     683     3603484 :         blks[top] = blkId;
     684             : 
     685             :         /* initialize the scope */
     686   460136288 :         for (i = 0; i < mb->vtop; i++)
     687   456532804 :                 setVarScope(mb, i, 0);
     688             : 
     689             :         /* all signature variables are declared at outer level */
     690     3603484 :         sig = getInstrPtr(mb, 0);
     691     7226612 :         for (k = 0; k < sig->argc; k++)
     692     3623128 :                 setVarScope(mb, getArg(sig, k), blkId);
     693             : 
     694   240541767 :         for (pc = 1; pc < mb->stop; pc++) {
     695   236938288 :                 p = getInstrPtr(mb, pc);
     696   236938288 :                 if (p->token == REMsymbol)
     697    30248576 :                         continue;
     698             :                 /* check correct use of the arguments */
     699   813092348 :                 for (k = p->retc; k < p->argc; k++) {
     700   606402641 :                         l = getArg(p, k);
     701   606402641 :                         if (l < 0)
     702           0 :                                 throw(MAL, "chkFlow",
     703             :                                           "%s.%s Non-declared variable: pc=%d, var= %d",
     704             :                                           getModuleId(sig), getFunctionId(sig), pc, k);
     705   606402641 :                         setVarUsed(mb, l);
     706   606402641 :                         if (getVarScope(mb, l) == 0) {
     707             :                                 /*
     708             :                                  * The problem created here is that only variables are
     709             :                                  * recognized that are declared through instructions.
     710             :                                  * For interactive code, and code that is based on a global
     711             :                                  * stack this is insufficient. In those cases, the variable
     712             :                                  * can be defined in a previous execution.
     713             :                                  * We have to recognize if the declaration takes place
     714             :                                  * in the context of a global stack.
     715             :                                  */
     716   249573228 :                                 if (p->barrier == CATCHsymbol) {
     717           0 :                                         setVarScope(mb, l, blks[0]);
     718   249573228 :                                 } else if (!(isVarConstant(mb, l) || isVarTypedef(mb, l))
     719       12615 :                                                    && !isVarInit(mb, l)) {
     720           5 :                                         throw(MAL, "chkFlow",
     721             :                                                   "%s.%s '%s' may not be used before being initialized",
     722             :                                                   getModuleId(sig), getFunctionId(sig), getVarNameIntoBuffer(mb, l, name));
     723             :                                 }
     724   356829413 :                         } else if (!isVarInit(mb, l)) {
     725             :                                 /* is the block still active ? */
     726       10803 :                                 for (i = 0; i <= top; i++)
     727       10803 :                                         if (blks[i] == getVarScope(mb, l))
     728             :                                                 break;
     729       10803 :                                 if (i > top || blks[i] != getVarScope(mb, l))
     730           0 :                                         throw(MAL, "chkFlow", "%s.%s '%s' used outside scope",
     731             :                                                   getModuleId(sig), getFunctionId(sig), getVarNameIntoBuffer(mb, l, name));
     732             :                         }
     733   606402636 :                         if (blockCntrl(p) || blockStart(p))
     734      345170 :                                 setVarInit(mb, l);
     735             :                 }
     736             :                 /* define variables */
     737   424488733 :                 for (k = 0; k < p->retc; k++) {
     738   217799026 :                         l = getArg(p, k);
     739   217799026 :                         if (isVarInit(mb, l) && getVarScope(mb, l) == 0) {
     740             :                                 /* first time we see this variable and it is already
     741             :                                  * initialized: assume it exists globally */
     742   170646265 :                                 setVarScope(mb, l, blks[0]);
     743             :                         }
     744   217799026 :                         setVarInit(mb, l);
     745   217799026 :                         if (getVarScope(mb, l) == 0) {
     746             :                                 /* variable has not been defined yet */
     747             :                                 /* exceptions are always declared at level 1 */
     748    46889415 :                                 if (p->barrier == CATCHsymbol)
     749         202 :                                         setVarScope(mb, l, blks[0]);
     750             :                                 else
     751    46889213 :                                         setVarScope(mb, l, blks[top]);
     752             :                         }
     753   217799026 :                         if (blockCntrl(p) || blockStart(p))
     754      322912 :                                 setVarUsed(mb, l);
     755             :                 }
     756   206689707 :                 if (p->barrier && msg == MAL_SUCCEED) {
     757      570586 :                         if (blockStart(p)) {
     758      278313 :                                 if (top == MAXDEPTH - 2)
     759           0 :                                         throw(MAL, "chkFlow",
     760             :                                                   "%s.%s too deeply nested  MAL program",
     761             :                                                   getModuleId(sig), getFunctionId(sig));
     762      278313 :                                 blkId++;
     763      278313 :                                 if (getModuleId(p) && getFunctionId(p)
     764      272437 :                                         && strcmp(getModuleId(p), "language") == 0
     765      264051 :                                         && strcmp(getFunctionId(p), "dataflow") == 0) {
     766      264060 :                                         if (dflow != -1)
     767           0 :                                                 throw(MAL, "chkFlow",
     768             :                                                           "%s.%s setLifeSpan nested dataflow blocks not allowed",
     769             :                                                           getModuleId(sig), getFunctionId(sig));
     770      264060 :                                         dflow = blkId;
     771             :                                 }
     772      278313 :                                 blks[++top] = blkId;
     773             :                         }
     774      570586 :                         if (blockExit(p) && top > 0) {
     775      278393 :                                 if (dflow == blks[top]) {
     776             :                                         dflow = -1;
     777             :                                 } else
     778             :                                         /*
     779             :                                          * At the end of the block we should reset the status of all variables
     780             :                                          * defined within the block. For, the block could have been skipped
     781             :                                          * leading to uninitialized variables.
     782             :                                          */
     783    10805071 :                                         for (l = 0; l < mb->vtop; l++)
     784    10790769 :                                                 if (getVarScope(mb, l) == blks[top]) {
     785       69226 :                                                         setVarScope(mb, l, 0);
     786       69226 :                                                         clrVarInit(mb, l);
     787             :                                                 }
     788      278393 :                                 top--;
     789             :                         }
     790             :                 }
     791             :         }
     792             :         return msg;
     793             : }

Generated by: LCOV version 1.14