LCOV - code coverage report
Current view: top level - sql/server - rel_select.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 3684 4060 90.7 %
Date: 2025-03-25 20:06:35 Functions: 113 114 99.1 %

          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             : #include "monetdb_config.h"
      14             : #include "rel_select.h"
      15             : #include "sql_tokens.h"
      16             : #include "sql_privileges.h"
      17             : #include "sql_env.h"
      18             : #include "sql_decimal.h"
      19             : #include "sql_qc.h"
      20             : #include "rel_rel.h"
      21             : #include "rel_basetable.h"
      22             : #include "rel_exp.h"
      23             : #include "rel_xml.h"
      24             : #include "rel_dump.h"
      25             : #include "rel_prop.h"
      26             : #include "rel_psm.h"
      27             : #include "rel_schema.h"
      28             : #include "rel_unnest.h"
      29             : #include "rel_sequence.h"
      30             : #include "rel_file_loader.h"
      31             : #include "rel_proto_loader.h"
      32             : #include "rel_optimizer_private.h"
      33             : 
      34             : #define VALUE_FUNC(f) (f->func->type == F_FUNC || f->func->type == F_FILT)
      35             : #define check_card(card,f) ((card == card_none && !f->res) || (CARD_VALUE(card) && f->res && VALUE_FUNC(f)) || card == card_loader || (card == card_relation && f->func->type == F_UNION))
      36             : 
      37             : /* return all expressions, with table name == tname */
      38             : static list *
      39       58251 : rel_table_projections( mvc *sql, sql_rel *rel, char *tname, int level )
      40             : {
      41       58779 :         list *exps;
      42             : 
      43       58779 :         if (mvc_highwater(sql))
      44           0 :                 return sql_error(sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
      45             : 
      46       58779 :         if (!rel)
      47             :                 return NULL;
      48             : 
      49       58779 :         if (!tname)
      50       42863 :                 return _rel_projections(sql, rel, NULL, 1, 0, 1);
      51             : 
      52       15916 :         switch(rel->op) {
      53         541 :         case op_join:
      54             :         case op_left:
      55             :         case op_right:
      56             :         case op_full:
      57         541 :                 exps = rel_table_projections( sql, rel->l, tname, level+1);
      58         541 :                 if (exps)
      59             :                         return exps;
      60          20 :                 return rel_table_projections( sql, rel->r, tname, level+1);
      61         508 :         case op_semi:
      62             :         case op_anti:
      63             :         case op_select:
      64         508 :                 return rel_table_projections( sql, rel->l, tname, level+1);
      65             : 
      66         137 :         case op_topn:
      67             :         case op_sample:
      68             :         case op_groupby:
      69             :         case op_union:
      70             :         case op_except:
      71             :         case op_inter:
      72             :         case op_project:
      73         137 :                 if (!is_processed(rel) && level == 0)
      74           0 :                         return rel_table_projections( sql, rel->l, tname, level+1);
      75             :                 /* fall through */
      76             :         case op_munion:
      77         146 :                 if (!is_processed(rel) && level == 0) {
      78           0 :                         node *n = ((list*)rel->l)->h;
      79           0 :                         if (n)
      80           0 :                                 return rel_table_projections(sql, n->data, tname, level+1);
      81             :                 }
      82             :         /* fall through */
      83             :         case op_table:
      84             :         case op_basetable:
      85       14867 :                 if (is_basetable(rel->op) && !rel->exps)
      86       14213 :                         return rel_base_project_all(sql, rel, tname);
      87         654 :                 if (rel->exps) {
      88         654 :                         int rename = 0;
      89         654 :                         node *en;
      90             : 
      91             :                         /* first check alias */
      92         654 :                         if (!is_base(rel->op) && !level) {
      93         129 :                                 list *exps = sa_list(sql->sa);
      94             : 
      95         647 :                                 for (en = rel->exps->h; en && !rename; en = en->next) {
      96         518 :                                         sql_exp *e = en->data;;
      97             : 
      98         518 :                                         if ((is_basecol(e) && exp_relname(e) && strcmp(exp_relname(e), tname) == 0) ||
      99         283 :                                             (is_basecol(e) && !exp_relname(e) && e->l && strcmp(e->l, tname) == 0)) {
     100         235 :                                                 if (exp_name(e) && exps_bind_column2(exps, tname, exp_name(e), NULL))
     101             :                                                         rename = 1;
     102             :                                                 else
     103         234 :                                                         append(exps, e);
     104             :                                         }
     105             :                                 }
     106             :                         }
     107             : 
     108         654 :                         exps = new_exp_list(sql->sa);
     109        5270 :                         for (en = rel->exps->h; en; en = en->next) {
     110        4616 :                                 sql_exp *e = en->data;
     111        4616 :                                 if (is_basecol(e) && exp_relname(e) && strcmp(exp_relname(e), tname) == 0) {
     112        4323 :                                         if (rename)
     113           2 :                                                 append(exps, exp_alias_ref(sql, e));
     114             :                                         else {
     115        4321 :                                                 sql_exp *ne = exp_ref(sql, e);
     116        4321 :                                                 exp_setname(sql, ne, tname, exp_name(e));
     117        4321 :                                                 append(exps, ne);
     118             :                                         }
     119             :                                 }
     120        4616 :                                 if (is_basecol(e) && !exp_relname(e) && e->l && strcmp(e->l, tname) == 0) {
     121           0 :                                         if (rename)
     122           0 :                                                 append(exps, exp_alias_ref(sql, e));
     123             :                                         else {
     124           0 :                                                 sql_exp *ne = exp_ref(sql, e);
     125           0 :                                                 exp_setname(sql, ne, tname, exp_name(e));
     126           0 :                                                 append(exps, ne);
     127             :                                         }
     128             :                                 }
     129             : 
     130             :                         }
     131         654 :                         if (exps && list_length(exps))
     132             :                                 return exps;
     133             :                 }
     134             :                 /* fall through */
     135             :         default:
     136             :                 return NULL;
     137             :         }
     138             : }
     139             : 
     140             : static sql_exp *
     141           0 : rel_lastexp(mvc *sql, sql_rel *rel )
     142             : {
     143           0 :         sql_exp *e;
     144             : 
     145           0 :         if (!is_processed(rel) || is_topn(rel->op) || is_sample(rel->op))
     146           0 :                 rel = rel_parent(rel);
     147           0 :         assert(list_length(rel->exps));
     148           0 :         if (rel->op == op_project) {
     149           0 :                 list_hash_clear(rel->exps);
     150           0 :                 return exp_ref(sql, rel->exps->t->data);
     151             :         }
     152           0 :         assert(is_project(rel->op));
     153           0 :         e = rel->exps->t->data;
     154           0 :         return exp_ref(sql, e);
     155             : }
     156             : 
     157             : static sql_rel *
     158       37435 : rel_orderby(mvc *sql, sql_rel *l)
     159             : {
     160       37435 :         sql_rel *rel = rel_create(sql->sa);
     161       37435 :         if (!rel)
     162             :                 return NULL;
     163             : 
     164       37435 :         assert(l->op == op_project && !l->r);
     165       37435 :         rel->l = l;
     166       37435 :         rel->r = NULL;
     167       37435 :         rel->op = op_project;
     168       37435 :         rel->exps = rel_projections(sql, l, NULL, 1, 0);
     169       37435 :         rel->card = l->card;
     170       37435 :         rel->nrcols = l->nrcols;
     171       37435 :         return rel;
     172             : }
     173             : 
     174             : /* forward refs */
     175             : static sql_rel * rel_setquery(sql_query *query, symbol *sq);
     176             : static sql_rel * rel_joinquery(sql_query *query, symbol *sq, list *refs);
     177             : 
     178             : static sql_rel *
     179      211629 : rel_table_optname(mvc *sql, sql_rel *sq, symbol *optname, list *refs)
     180             : {
     181      211629 :         sql_rel *osq = sq;
     182      211629 :         node *ne;
     183             : 
     184      211629 :         if (optname && optname->token == SQL_NAME) {
     185       17338 :                 dlist *columnrefs = NULL;
     186       17338 :                 char *tname = optname->data.lval->h->data.sval;
     187       17338 :                 list *l = sa_list(sql->sa);
     188             : 
     189       17338 :                 columnrefs = optname->data.lval->h->next->data.lval;
     190       17338 :                 if (is_topn(sq->op) || is_sample(sq->op) || ((is_simple_project(sq->op) || is_groupby(sq->op)) && sq->r) || is_base(sq->op)) {
     191         138 :                         sq = rel_project(sql->sa, sq, rel_projections(sql, sq, NULL, 1, 0));
     192         138 :                         osq = sq;
     193             :                 }
     194       17338 :                 if (columnrefs && dlist_length(columnrefs) != list_length(sq->exps))
     195           6 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: The number of aliases don't match the number of columns (%d != %d)", dlist_length(columnrefs), sq->nrcols);
     196        3961 :                 if (columnrefs && sq->exps) {
     197        3961 :                         dnode *d = columnrefs->h;
     198             : 
     199        3961 :                         ne = sq->exps->h;
     200        3961 :                         list_hash_clear(sq->exps);
     201       17737 :                         for (; d && ne; d = d->next, ne = ne->next) {
     202        9816 :                                 sql_exp *e = ne->data;
     203             : 
     204        9816 :                                 if (exps_bind_column2(l, tname, d->data.sval, NULL))
     205           1 :                                         return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "SELECT: Duplicate column name '%s.%s'", tname, d->data.sval);
     206        9815 :                                 exp_setname(sql, e, tname, d->data.sval );
     207        9815 :                                 if (!is_intern(e))
     208        9815 :                                         set_basecol(e);
     209        9815 :                                 append(l, e);
     210             :                         }
     211             :                 }
     212       17331 :                 if (!columnrefs && sq->exps) {
     213       13371 :                         ne = sq->exps->h;
     214       13371 :                         list_hash_clear(sq->exps);
     215      127504 :                         for (; ne; ne = ne->next) {
     216      100762 :                                 sql_exp *e = ne->data;
     217      100762 :                                 char *name = NULL;
     218             : 
     219      100762 :                                 if (!is_intern(e)) {
     220      100762 :                                         if (!exp_name(e))
     221         109 :                                                 name = make_label(sql->sa, ++sql->label);
     222      100762 :                                         noninternexp_setname(sql, e, tname, name);
     223      100762 :                                         set_basecol(e);
     224             :                                 }
     225             :                         }
     226             :                 }
     227       17331 :                 if (refs) { /* if this relation is under a FROM clause, check for duplicate names */
     228       17281 :                         if (list_find(refs, tname, (fcmp) &strcmp))
     229           6 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: relation name \"%s\" specified more than once", tname);
     230       17275 :                         assert(tname);
     231       17275 :                         list_append(refs, tname);
     232             :                 }
     233             :         } else {
     234      194291 :                 if (!is_project(sq->op) || is_topn(sq->op) || is_sample(sq->op) || ((is_simple_project(sq->op) || is_groupby(sq->op)) && sq->r)) {
     235       59912 :                         sq = rel_project(sql->sa, sq, rel_projections(sql, sq, NULL, 1, 1));
     236       59912 :                         osq = sq;
     237             :                 }
     238     2428293 :                 for (ne = osq->exps->h; ne; ne = ne->next) {
     239     2234002 :                         sql_exp *e = ne->data;
     240             : 
     241     2234002 :                         if (!is_intern(e))
     242     2101890 :                                 set_basecol(e);
     243             :                 }
     244             :         }
     245             :         return osq;
     246             : }
     247             : 
     248             : static sql_rel *
     249      102959 : rel_subquery_optname(sql_query *query, symbol *ast, list *refs)
     250             : {
     251      102959 :         mvc *sql = query->sql;
     252      102959 :         SelectNode *sn = (SelectNode *) ast;
     253      102959 :         exp_kind ek = {type_value, card_relation, TRUE};
     254      102959 :         sql_rel *sq = rel_subquery(query, ast, ek);
     255             : 
     256      102959 :         assert(ast->token == SQL_SELECT);
     257      102959 :         if (!sq)
     258             :                 return NULL;
     259             : 
     260      102937 :         return rel_table_optname(sql, sq, sn->name, refs);
     261             : }
     262             : 
     263             : static void
     264       10177 : rel_rename(mvc *sql, sql_rel *nrel, char *rname, sql_rel *brel)
     265             : {
     266       10177 :         assert(is_project(nrel->op));
     267       10177 :         if (brel) {
     268         156 :                 if (is_project(nrel->op) && nrel->exps) {
     269         368 :                         for (node *ne = nrel->exps->h, *be = brel->exps->h; ne && be; ne = ne->next, be = be->next) {
     270         212 :                                 sql_exp *e = ne->data;
     271         212 :                                 sql_exp *b = be->data;
     272         212 :                                 char *name = NULL;
     273             : 
     274         212 :                                 if (!is_intern(e)) {
     275         212 :                                         if (!exp_name(e))
     276           0 :                                                 name = make_label(sql->sa, ++sql->label);
     277         212 :                                         noninternexp_setname(sql, e, rname, name);
     278         212 :                                         set_basecol(e);
     279         212 :                                         e->alias.label = b->alias.label;
     280             :                                 }
     281             :                         }
     282             :                 }
     283         156 :                 list_hash_clear(nrel->exps);
     284       10021 :         } else if (is_project(nrel->op) && nrel->exps) {
     285       10021 :                 node *ne = nrel->exps->h;
     286             : 
     287       41404 :                 for (; ne; ne = ne->next) {
     288       31383 :                         sql_exp *e = ne->data;
     289       31383 :                         char *name = NULL;
     290             : 
     291       31383 :                         if (!is_intern(e)) {
     292       31383 :                                 if (!exp_name(e))
     293          17 :                                         name = make_label(sql->sa, ++sql->label);
     294       31383 :                                 noninternexp_setname(sql, e, rname, name);
     295       31383 :                                 set_basecol(e);
     296             :                         }
     297             :                 }
     298       10021 :                 list_hash_clear(nrel->exps);
     299             :         }
     300       10177 : }
     301             : 
     302             : sql_rel *
     303        5937 : rel_with_query(sql_query *query, symbol *q )
     304             : {
     305        5937 :         mvc *sql = query->sql;
     306        5937 :         dnode *d = q->data.lval->h;
     307        5937 :         symbol *next = d->next->data.sym;
     308        5937 :         bool recursive = d->next->next->data.i_val;
     309        5937 :         sql_rel *rel;
     310             : 
     311        5937 :         if (!stack_push_frame(sql, NULL))
     312           0 :                 return sql_error(sql, 02, SQLSTATE(HY013) MAL_MALLOC_FAIL);
     313             :         /* first handle all with's (ie inlined views) */
     314       16036 :         for (d = d->data.lval->h; d; d = d->next) {
     315       10119 :                 symbol *sym = d->data.sym;
     316       10119 :                 dnode *dn = sym->data.lval->h->next;
     317       10119 :                 char *rname = qname_schema_object(dn->data.lval);
     318       10119 :                 sql_rel *nrel, *base_rel = NULL;
     319       10119 :                 symbol *recursive_part = NULL;
     320       10119 :                 sql_rel_view *recursive_union = NULL;
     321       10119 :                 int recursive_distinct = 0;
     322             : 
     323       10119 :                 if (frame_find_rel_view(sql, rname)) {
     324           3 :                         stack_pop_frame(sql);
     325           3 :                         return sql_error(sql, 01, SQLSTATE(42000) "View '%s' already declared", rname);
     326             :                 }
     327       10116 :                 if (recursive) {
     328         103 :                         symbol *union_stmt = dn->next->next->data.sym;
     329         103 :                         if (union_stmt->token == SQL_UNION) { /* split in base and recursive part */
     330          78 :                                 dnode *n = union_stmt->data.lval->h;
     331          78 :                                 symbol *base = n->data.sym;
     332          78 :                                 recursive_distinct = n->next->data.i_val;
     333          78 :                                 dlist *corresponding = n->next->next->data.lval;
     334          78 :                                 recursive_part = n->next->next->next->data.sym;
     335          78 :                                 if (corresponding)
     336           0 :                                         return sql_error(sql, 01, SQLSTATE(42000) "Recursive with corresponding is not supported");
     337          78 :                                 dn->next->next->data.sym = base;
     338             :                         }
     339             :                 }
     340       10116 :                 nrel = rel_semantic(query, sym);
     341       10116 :                 if (!nrel) {
     342          17 :                         stack_pop_frame(sql);
     343          17 :                         return NULL;
     344             :                 }
     345       10099 :                 if (!(recursive_union = stack_push_rel_view(sql, rname, nrel))) {
     346           0 :                         stack_pop_frame(sql);
     347           0 :                         return sql_error(sql, 02, SQLSTATE(HY013) MAL_MALLOC_FAIL);
     348             :                 }
     349       10099 :                 if (recursive && recursive_part) {
     350          78 :                         base_rel = nrel;
     351          78 :                         rel_rename(sql, base_rel, rname, base_rel);
     352          78 :                         dn->next->next->data.sym = recursive_part;
     353          78 :                         set_processed(nrel);
     354          78 :                         nrel = rel_semantic(query, sym);
     355          78 :                         if (!nrel) {
     356           0 :                                 stack_pop_frame(sql);
     357           0 :                                 return NULL;
     358             :                         }
     359          78 :                         list *ls = rel_projections(sql, base_rel, NULL, 0, 1);
     360          78 :                         list *rs = rel_projections(sql, nrel, NULL, 0, 1);
     361             : 
     362          78 :                         if (!rel_is_ref(base_rel)) { /* not recursive */
     363           3 :                                 nrel = rel_setop_n_ary_check_types(sql, base_rel, nrel, ls, rs, op_munion);
     364             :                         } else {
     365          75 :                                 base_rel->used |= statistics_gathered;
     366          75 :                                 prop *p = base_rel->p = prop_create(sql->sa, PROP_COUNT, base_rel->p);
     367          75 :                                 p->value.lval = 1000000; /* random ? */
     368             : 
     369             :                                 /* down cast the recursive side (on errors users should add casts on the base side) */
     370          75 :                                 list *nrs = new_exp_list(sql->sa);
     371          75 :                                 if(!nrs)
     372             :                                         return NULL;
     373             : 
     374         174 :                                 for (node *n = ls->h, *m = rs->h; n && m; n = n->next, m = m->next) {
     375          99 :                                         sql_subtype *t = exp_subtype(n->data);
     376          99 :                                         append(nrs, exp_check_type(sql, t, nrel, m->data, type_equal));
     377             :                                 }
     378          75 :                                 nrel = rel_project(sql->sa, nrel, nrs);
     379          75 :                                 nrel = rel_setop_n_ary(sql->sa, append(append(sa_list(sql->sa), base_rel), nrel), op_munion);
     380          75 :                                 set_recursive(nrel);
     381             :                         }
     382          78 :                         if (recursive_distinct)
     383          34 :                                 set_distinct(nrel);
     384          78 :                         rel_setop_n_ary_set_exps(sql, nrel, rel_projections(sql, nrel, NULL, 0, 1), false);
     385          78 :                         set_processed(nrel);
     386          78 :                         recursive_union->rel_view = nrel;
     387             :                 }
     388       10099 :                 if (!is_project(nrel->op)) {
     389           0 :                         if (is_topn(nrel->op) || is_sample(nrel->op)) {
     390           0 :                                 nrel = rel_project(sql->sa, nrel, rel_projections(sql, nrel, NULL, 1, 1));
     391             :                         } else {
     392           0 :                                 stack_pop_frame(sql);
     393           0 :                                 return NULL;
     394             :                         }
     395             :                 }
     396       10099 :                 rel_rename(sql, nrel, rname, base_rel);
     397             :         }
     398        5917 :         rel = rel_semantic(query, next);
     399        5917 :         stack_pop_frame(sql);
     400        5917 :         return rel;
     401             : }
     402             : 
     403             : static sql_rel *
     404       61027 : query_exp_optname(sql_query *query, symbol *q, list *refs)
     405             : {
     406       61027 :         mvc *sql = query->sql;
     407       61027 :         switch (q->token) {
     408         995 :         case SQL_WITH:
     409             :         {
     410         995 :                 sql_rel *tq = rel_with_query(query, q);
     411             : 
     412         995 :                 if (!tq)
     413             :                         return NULL;
     414         995 :                 if (q->data.lval->t->type == type_symbol)
     415         995 :                         return rel_table_optname(sql, tq, q->data.lval->t->data.sym, refs);
     416             :                 return tq;
     417             :         }
     418       60032 :         case SQL_JOIN:
     419             :         {
     420       60032 :                 sql_rel *tq = rel_joinquery(query, q, refs);
     421             : 
     422       60032 :                 if (!tq)
     423             :                         return NULL;
     424       59993 :                 return rel_table_optname(sql, tq, q->data.lval->t->data.sym, NULL);
     425             :         }
     426           0 :         default:
     427           0 :                 (void) sql_error(sql, 02, SQLSTATE(42000) "case %d %s", (int) q->token, token2string(q->token));
     428             :         }
     429           0 :         return NULL;
     430             : }
     431             : 
     432             : static sql_subfunc *
     433      588699 : bind_func_(mvc *sql, char *sname, char *fname, list *ops, sql_ftype type, bool private, bool *found, bool exact)
     434             : {
     435      588699 :         sql_subfunc *sf = NULL;
     436             : 
     437      589162 :         if (sql->forward && strcmp(fname, sql->forward->base.name) == 0 &&
     438         465 :             list_cmp(sql->forward->ops, ops, (fcmp)&arg_subtype_cmp) == 0 &&
     439           4 :             execute_priv(sql, sql->forward) && type == sql->forward->type)
     440           2 :                 return sql_dup_subfunc(sql->sa, sql->forward, NULL, NULL);
     441      588697 :         sf = sql_bind_func_(sql, sname, fname, ops, type, private, exact);
     442      588696 :         if (found)
     443      588149 :                 *found |= sf != NULL;
     444      588696 :         if (sf && execute_priv(sql, sf->func))
     445             :                 return sf;
     446             :         return NULL;
     447             : }
     448             : 
     449             : static sql_subfunc *
     450      220229 : bind_func(mvc *sql, char *sname, char *fname, sql_subtype *t1, sql_subtype *t2, int nr, sql_ftype type, bool private, bool *found, bool exact)
     451             : {
     452      220229 :         list *tl = sa_list(sql->sa);
     453      220229 :         assert(nr >= 1 && nr <= 2);
     454      220229 :         append(tl, t1);
     455      220229 :         if (nr == 2)
     456      184505 :                 append(tl, t2);
     457      220229 :         sql_subfunc *sf = NULL;
     458             : 
     459      220229 :         if (sql->forward) {
     460       22910 :                 if (execute_priv(sql, sql->forward) &&
     461       22910 :                     strcmp(fname, sql->forward->base.name) == 0 &&
     462           0 :                    ((!t1 && list_length(sql->forward->ops) == 0) ||
     463           0 :                     (!t2 && list_length(sql->forward->ops) == 1 && arg_subtype_cmp(sql->forward->ops->h->data, t1) == 0) ||
     464           0 :                     (list_length(sql->forward->ops) == 2 &&
     465           0 :                         arg_subtype_cmp(sql->forward->ops->h->data, t1) == 0 &&
     466           0 :                         arg_subtype_cmp(sql->forward->ops->h->next->data, t2) == 0)) && type == sql->forward->type) {
     467           0 :                         return sql_dup_subfunc(sql->sa, sql->forward, NULL, NULL);
     468             :                 }
     469             :         }
     470      220229 :         sf = sql_bind_func_(sql, sname, fname, tl, type, private, exact);
     471      220229 :         if (found)
     472      220229 :                 *found |= sf != NULL;
     473      220229 :         if (sf && execute_priv(sql, sf->func))
     474             :                 return sf;
     475             :         return NULL;
     476             : }
     477             : 
     478             : static sql_subfunc *
     479      693717 : find_func(mvc *sql, char *sname, char *fname, int len, sql_ftype type, bool private, sql_subfunc *prev, bool *found)
     480             : {
     481      693717 :         sql_subfunc *sf = NULL;
     482             : 
     483      693717 :         if (sql->forward && strcmp(fname, sql->forward->base.name) == 0 && list_length(sql->forward->ops) == len && execute_priv(sql, sql->forward) && type == sql->forward->type)
     484           0 :                 return sql_dup_subfunc(sql->sa, sql->forward, NULL, NULL);
     485      693717 :         sf = sql_find_func(sql, sname, fname, len, type, private, prev);
     486      693717 :         if (found)
     487           0 :                 *found |= sf != NULL;
     488      693717 :         if (sf && execute_priv(sql, sf->func))
     489             :                 return sf;
     490             :         return NULL;
     491             : }
     492             : 
     493             : static sql_exp *
     494      806736 : exp_fix_scale(mvc *sql, sql_subtype *ct, sql_exp *e)
     495             : {
     496      806736 :         sql_subtype *et = exp_subtype(e);
     497             : 
     498      806736 :         if (ct->type->scale == SCALE_FIX && et->type->scale == SCALE_FIX) {
     499      139309 :                 int scale_diff = ((int) ct->scale - (int) et->scale);
     500             : 
     501      139309 :                 if (scale_diff) {
     502         132 :                         if (scale_diff < 0)
     503             :                                 return e;
     504         102 :                         sql_subtype st;
     505         102 :                         int scale = ct->scale;
     506         102 :                         int digits = ((et->type->eclass == EC_NUM)?bits2digits(et->digits):et->digits)-et->scale+scale;
     507         102 :                         (void)sql_find_subtype(&st, ct->type->base.name, digits, scale);
     508         102 :                         return exp_convert(sql, e, et, &st);
     509             :                 }
     510             :         }
     511             :         return e;
     512             : }
     513             : 
     514             : static lng
     515        9835 : rel_get_count(sql_rel *rel)
     516             : {
     517       31648 :         if (!rel)
     518             :                 return 0;
     519       31644 :         prop *p = NULL;
     520       31644 :         if (rel->p && (p = find_prop(rel->p, PROP_COUNT)) != NULL)
     521        3106 :                 return p->value.lval;
     522       28538 :         else if(is_munion(rel->op)) {
     523         174 :                 lng cnt = 0;
     524         174 :                 list *l = rel->l;
     525         522 :                 for (node *n = l->h; n; n = n->next) {
     526         348 :                         lng lcnt = rel_get_count(n->data);
     527         348 :                         if (lcnt == BUN_MAX)
     528             :                                 return BUN_MAX;
     529         348 :                         cnt += lcnt;
     530             :                 }
     531         174 :                 return cnt;
     532       28364 :         } else if(rel->l) {
     533       25650 :                 if (is_select(rel->op) || is_project(rel->op))
     534             :                         return rel_get_count(rel->l);
     535             :         }
     536             :         return 0;
     537             : }
     538             : 
     539             : #define is_sum_aggr(f) (f->type == F_AGGR && strcmp(f->base.name, "sum") == 0)
     540             : 
     541             : list *
     542      821232 : check_arguments_and_find_largest_any_type(mvc *sql, sql_rel *rel, list *exps, sql_subfunc *sf, int maybe_zero_or_one, bool internal)
     543             : {
     544      821232 :         list *nexps = new_exp_list(sql->sa);
     545      821230 :         sql_subtype *atp = NULL, super, *res = !list_empty(sf->res) ? sf->res->h->data: NULL;
     546      821230 :         unsigned int digits = 0, scale = 0;
     547             : 
     548             :         /* find largest any type argument */
     549     2395367 :         for (node *n = exps->h, *m = sf->func->ops->h; n && m; n = n->next, m = m->next) {
     550     1574137 :                 sql_arg *a = m->data;
     551     1574137 :                 sql_exp *e = n->data;
     552     1574137 :                 sql_subtype *t = exp_subtype(e);
     553             : 
     554     1574137 :                 if (a->type.type->eclass == EC_ANY) {
     555      291472 :                         if (t && atp) {
     556      129027 :                                 supertype(&super, t, atp);
     557      129027 :                                 atp = &super;
     558      162445 :                         } else if (t) {
     559             :                                 atp = t;
     560             :                         }
     561             :                 }
     562     1574134 :                 if (t && sf->func->fix_scale == SCALE_FIX && t->type->eclass == EC_DEC) {
     563        1260 :                         if (digits < t->digits)
     564             :                                 digits = t->digits;
     565        1260 :                         if (scale < t->scale)
     566             :                                 scale = t->scale;
     567             :                 }
     568             :         }
     569      821230 :         if (!atp && !list_empty(exps))
     570      656914 :                 atp = exp_subtype(exps->h->data);
     571             : 
     572      821229 :         if ((atp && atp->type->localtype == TYPE_void) || !atp) /* NULL */
     573       12057 :                 atp = sql_bind_localtype("str");
     574             : 
     575      821229 :         node *n, *m;
     576      821229 :         sql_arg *last = NULL;
     577     3969593 :         for (n = exps->h, m = sf->func->ops->h; n && ((sf->func->vararg && last) || m); n = n->next, m = m?m->next:NULL) {
     578     1574184 :                 sql_arg *a = m?m->data:last;
     579     1574184 :                 sql_exp *e = n->data;
     580     1574184 :                 sql_subtype *ntp = &a->type, *t = exp_subtype(e);
     581             : 
     582     1574182 :                 last = a;
     583     1574182 :                 if (!t) {
     584          37 :                         if (a->type.type->eclass == EC_ANY && atp)
     585           3 :                                 ntp = sql_create_subtype(sql->sa, atp->type, atp->digits, atp->scale);
     586          59 :                         rel_set_type_param(sql, ntp, rel, e, sf->func->fix_scale != INOUT && !UDF_LANG(sf->func->lang));
     587     1574145 :                 } else if (a->type.type->eclass == EC_ANY && atp) {
     588      291516 :                         ntp = sql_create_subtype(sql->sa, atp->type, atp->digits, atp->scale);
     589     1282629 :                 } else if (t && ntp->digits == 0 && EC_VARCHAR(a->type.type->eclass)) {
     590      533974 :                         ntp = sql_create_subtype(sql->sa, a->type.type, type_digits_to_char_digits(t), 0);
     591      748655 :                 } else if (t && ntp->digits > 0 && a->type.type->eclass == EC_NUM && t->type->eclass == EC_NUM) {
     592      400974 :                         ntp = sql_create_subtype(sql->sa, a->type.type, t->digits, 0);
     593      347681 :                 } else if (t && ntp->scale == 0 && ntp->type->eclass == EC_DEC && EC_VARCHAR(t->type->eclass)) {
     594           0 :                         sql_subtype *res = SA_NEW(sql->sa, sql_subtype);
     595           0 :                         int digits = t->digits?t->digits+3:ntp->digits;
     596           0 :                         (void)sql_find_subtype(res, a->type.type->base.name, digits, 3);
     597           0 :                         ntp = res;
     598      347681 :                 } else if (t && ntp->scale == 0 && ntp->type->eclass == EC_DEC) {
     599        4914 :                         ntp = sql_create_subtype(sql->sa, a->type.type, t->type->eclass == EC_NUM?bits2digits(t->digits):t->digits, t->scale);
     600      342767 :                 } else if (t->type == ntp->type) {
     601      340656 :                         ntp = t;
     602             :                 }
     603     1574182 :                 if (!(e = exp_check_type(sql, ntp, rel, e, type_equal)))
     604             :                         return NULL;
     605     1574185 :                 if (sf->func->fix_scale == SCALE_FIX) {
     606      592898 :                         ntp = sql_create_subtype(sql->sa, a->type.type->localtype?a->type.type:t?t->type:atp->type, digits, scale);
     607      592898 :                         e = exp_fix_scale(sql, ntp, e);
     608      981287 :                 } else if (sf->func->fix_scale == SCALE_EQ) {
     609      213838 :                         e = exp_fix_scale(sql, &a->type, e);
     610             :                 }
     611     1574185 :                 if (maybe_zero_or_one && e->card > CARD_ATOM) {
     612          22 :                         sql_subfunc *zero_or_one = sql_bind_func(sql, "sys", "zero_or_one", exp_subtype(e), NULL, F_AGGR, true, false);
     613          22 :                         e = exp_aggr1(sql->sa, e, zero_or_one, 0, 0, CARD_ATOM, has_nil(e));
     614             :                 }
     615     1574185 :                 append(nexps, e);
     616             :         }
     617             :         /* handle any extra arguments for rel_dump/analytic funcs */
     618      821449 :         for ( ; n; n = n->next)
     619         222 :                 append(nexps, n->data);
     620      821227 :         if (sf->func->fix_scale == SCALE_FIX || IS_ANALYTIC(sf->func)) {
     621      299011 :                 exps_scale_fix(sf, nexps, atp);
     622      522216 :         } else if (sf->func->fix_scale == MAX_BITS) {
     623      114559 :                 exps_max_bits(sf, nexps);
     624      407657 :         } else if (sf->func->fix_scale == SCALE_MUL) {
     625       25384 :                 exps_sum_scales(sf, nexps);
     626      382273 :         } else if (!internal && sf->func->fix_scale == SCALE_DIV) {
     627        2625 :                 if (!exps_scale_algebra(sql, sf, rel, nexps))
     628             :                         return NULL;
     629      379648 :         } else if (sf->func->fix_scale == DIGITS_ADD) {
     630      188947 :                 exps_digits_add(sf, nexps);
     631      190701 :         } else if (sf->func->fix_scale == INOUT) {
     632       19792 :                 exps_inout(sf, nexps);
     633      170909 :         } else if (is_sum_aggr(sf->func))
     634        9487 :                 exps_largest_int(sf, nexps, rel_get_count(rel));
     635             : 
     636             :         /* dirty hack */
     637      821225 :         if (sf->func->type != F_PROC && sf->func->type != F_UNION && sf->func->type != F_LOADER && res) {
     638      782551 :                 if (res->type->eclass == EC_ANY && atp)
     639          73 :                         sf->res->h->data = sql_create_subtype(sql->sa, atp->type, atp->digits, atp->scale);
     640             :         }
     641             :         return nexps;
     642             : }
     643             : 
     644             : static char *
     645         286 : nary_function_arg_types_2str(mvc *sql, list* types, int N)
     646             : {
     647         286 :         char *arg_list = NULL;
     648         286 :         int i = 0;
     649             : 
     650         770 :         for (node *n = types->h; n && i < N; n = n->next) {
     651         484 :                 sql_subtype *t = (sql_subtype *) n->data;
     652         484 :                 char *tpe = t ? sql_subtype_string(sql->ta, t) : "?";
     653             : 
     654         484 :                 if (arg_list) {
     655         215 :                         arg_list = sa_message(sql->ta, "%s, %s", arg_list, tpe);
     656             :                 } else {
     657             :                         arg_list = tpe;
     658             :                 }
     659         484 :                 i++;
     660             :         }
     661         286 :         return arg_list;
     662             : }
     663             : 
     664             : static char *
     665          35 : file_loader_add_table_column_types(mvc *sql, sql_subfunc *f, list *exps, list *res_exps, char *tname)
     666             : {
     667          35 :         sql_exp *file = exps->h->data;
     668          35 :         if (!exp_is_atom(file))
     669             :                 return "Filename missing";
     670             : 
     671          35 :         atom *a = file->l;
     672          35 :         if (a->data.vtype != TYPE_str || !a->data.val.sval)
     673             :                 return "Filename missing";
     674             : 
     675          35 :         char *filename = a->data.val.sval;
     676          35 :         if (strcmp(filename, "") == 0)
     677             :                 return "Filename missing";
     678             : 
     679          33 :         char *ext = strrchr(filename, '.'), *ep = ext;
     680             : 
     681          33 :         if (ext) {
     682          30 :                 ext = ext + 1;
     683          30 :                 ext = mkLower(sa_strdup(sql->sa, ext));
     684             :         }
     685          30 :         if (!ext)
     686           3 :                 return "Filename extension missing";
     687             : 
     688          30 :         file_loader_t *fl = fl_find(ext);
     689          30 :         if (!fl) {
     690             :                 /* maybe compressed */
     691          10 :                 char *p = ep - 1;
     692          66 :                 while (p > filename && *p != '.')
     693          56 :                         p--;
     694          10 :                 if (p != filename) {
     695           8 :                         ext = p + 1;
     696           8 :                         ext = sa_strdup(sql->sa, ext);
     697           8 :                         char *d = strchr(ext, '.');
     698           8 :                         assert(d);
     699           8 :                         *d = 0;
     700           8 :                         fl = fl_find(ext);
     701             :                 }
     702          10 :                 if (!fl) /* fallback */
     703           2 :                         fl = fl_find("csv");
     704           2 :                 if (!fl) /* not expected */
     705           0 :                         return sa_message(sql->ta, "Filename extension '%s' missing", ext?ext:"");
     706             :         }
     707          30 :         str err = fl->add_types(sql, f, filename, res_exps, tname);
     708          30 :         if (err)
     709             :                 return err;
     710          16 :         sql_subtype *st = sql_bind_localtype("str");
     711          16 :         sql_exp *ext_exp = exp_atom(sql->sa, atom_string(sql->sa, st, ext));
     712          16 :         if (!ext_exp)
     713             :                 return MAL_MALLOC_FAIL;
     714          16 :         append(exps, ext_exp);
     715          16 :         return NULL;
     716             : }
     717             : 
     718             : static sql_rel *
     719          35 : rel_file_loader(mvc *sql, list *exps, list *tl, char *tname)
     720             : {
     721          35 :         sql_subfunc *f = NULL;
     722          35 :         bool found = false;
     723             : 
     724          35 :         if ((f = bind_func_(sql, NULL, "file_loader", tl, F_UNION, true, &found, false))) {
     725          35 :                 list *nexps = exps;
     726          35 :                 if (list_empty(tl) || (nexps = check_arguments_and_find_largest_any_type(sql, NULL, exps, f, 1, false))) {
     727          35 :                         list *res_exps = sa_list(sql->sa);
     728          35 :                         if (list_length(exps) == 1 && f && f->func->varres && strlen(f->func->mod) == 0 && strlen(f->func->imp) == 0) {
     729          35 :                                 char *err = file_loader_add_table_column_types(sql, f, nexps, res_exps, tname);
     730          35 :                                 if (err)
     731          19 :                                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: file_loader function failed '%s'", err);
     732             :                         }
     733          16 :                         sql_exp *e = exp_op(sql->sa, nexps, f);
     734          16 :                         sql_rel *rel = rel_table_func(sql->sa, NULL, e, res_exps, TABLE_PROD_FUNC);
     735          16 :                         if (rel)
     736          16 :                                 rel = rel_project(sql->sa, rel, exps_alias(sql, res_exps));
     737          16 :                         return rel;
     738             :                 }
     739             :         }
     740             :         return NULL;
     741             : }
     742             : 
     743             : static char *
     744          71 : proto_loader_add_table_column_types(mvc *sql, sql_subfunc *f, list *exps, list *res_exps, char *tname)
     745             : {
     746          71 :         sql_exp *uri = exps->h->data;
     747          71 :         if (!exp_is_atom(uri))
     748             :                 return "URI missing";
     749             : 
     750          71 :         atom *a = uri->l;
     751          71 :         if (a->data.vtype != TYPE_str || !a->data.val.sval)
     752             :                 return "URI missing";
     753             : 
     754          71 :         char *uristr = a->data.val.sval;
     755          71 :         if (strcmp(uristr, "") == 0)
     756             :                 return "URI missing";
     757             : 
     758          71 :         char *proto = uristr;
     759          71 :         char *ep = strchr(uristr, ':');
     760          71 :         if (ep) {
     761          70 :                 *ep = 0;
     762          70 :                 proto = mkLower(sa_strdup(sql->sa, proto));
     763          70 :                 *ep = ':';
     764             :         } else {
     765             :                 return "Missing ':' separator to determine the URI scheme";
     766             :         }
     767             : 
     768          70 :         if (!proto)
     769             :                 return "URI scheme missing";
     770             : 
     771             :         // find uri scheme in registered protocols (e.g. is: 'file' or 'monetdb' or 'odbc')
     772          70 :         proto_loader_t *pl = pl_find(proto);
     773          70 :         if (!pl)
     774           0 :                 return sa_message(sql->ta, "URI protocol '%s' not supported", proto?proto:"");
     775             : 
     776          70 :         str err = pl->add_types(sql, f, uristr, res_exps, tname);
     777          70 :         if (err)
     778             :                 return err;
     779             : 
     780          53 :         sql_subtype *st = sql_bind_localtype("str");
     781          53 :         sql_exp *proto_exp = exp_atom(sql->sa, atom_string(sql->sa, st, proto));
     782          53 :         if (!proto_exp)
     783             :                 return MAL_MALLOC_FAIL;
     784             : 
     785          53 :         append(exps, proto_exp);
     786          53 :         return NULL;
     787             : }
     788             : 
     789             : static sql_rel *
     790          72 : rel_proto_loader(mvc *sql, list *exps, list *tl, char *tname)
     791             : {
     792          72 :         sql_subfunc *f = NULL;
     793          72 :         bool found = false;
     794             : 
     795          72 :         if ((f = bind_func_(sql, NULL, "proto_loader", tl, F_UNION, true, &found, false))) {
     796          71 :                 list *nexps = exps;
     797          71 :                 if (list_empty(tl) || f->func->vararg || (nexps = check_arguments_and_find_largest_any_type(sql, NULL, exps, f, 1, false))) {
     798          71 :                         list *res_exps = sa_list(sql->sa);
     799          71 :                         if (list_length(exps) == 1 && f && f->func->varres && strlen(f->func->mod) == 0 && strlen(f->func->imp) == 0) {
     800          71 :                                 char *err = proto_loader_add_table_column_types(sql, f, nexps, res_exps, tname);
     801          71 :                                 if (err)
     802          18 :                                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: proto_loader function failed '%s'", err);
     803             :                         }
     804          53 :                         sql_exp *e = exp_op(sql->sa, nexps, f);
     805          53 :                         sql_rel *rel = rel_table_func(sql->sa, NULL, e, res_exps, TABLE_PROD_FUNC);
     806          53 :                         if (rel)
     807          53 :                                 rel = rel_project(sql->sa, rel, exps_alias(sql, res_exps));
     808          53 :                         return rel;
     809             :                 }
     810             :         }
     811             :         return NULL;
     812             : }
     813             : 
     814             : sql_exp *
     815       10999 : find_table_function(mvc *sql, char *sname, char *fname, list *exps, list *tl, sql_ftype type)
     816             : {
     817       10999 :         bool found = false;
     818       10999 :         sql_subfunc *f = NULL;
     819             : 
     820       10999 :         assert(type == F_UNION || type == F_LOADER);
     821       10999 :         if ((f = bind_func_(sql, sname, fname, tl, type, false, &found, false))) {
     822       10976 :                 list *nexps = exps;
     823       10976 :                 if (list_empty(tl) || (nexps = check_arguments_and_find_largest_any_type(sql, NULL, exps, f, 1, false)))
     824       10976 :                         return exp_op(sql->sa, nexps, f);
     825           0 :                 found = false;
     826             :         }
     827          23 :         char *arg_list = list_length(tl) ? nary_function_arg_types_2str(sql, tl, list_length(tl)) : NULL;
     828          61 :         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: %s %s function %s%s%s'%s'(%s)",
     829          23 :                                          found ? "insufficient privileges for" : "no such", type == F_UNION ? "table returning" : "loader", sname ? "'":"", sname ? sname : "",
     830             :                                          sname ? "'.":"", fname, arg_list ? arg_list : "");
     831             : }
     832             : 
     833             : static sql_rel *
     834       11087 : rel_named_table_function(sql_query *query, sql_rel *rel, symbol *ast, int lateral, list *refs)
     835             : {
     836       11087 :         mvc *sql = query->sql;
     837       11087 :         list *exps = NULL, *tl;
     838       11087 :         node *m;
     839       11087 :         exp_kind ek = {type_value, card_relation, TRUE};
     840       11087 :         sql_rel *sq = NULL, *outer = NULL;
     841       11087 :         sql_exp *e = NULL;
     842       11087 :         sql_subfunc *sf = NULL;
     843       11087 :         symbol *sym = ast->data.lval->h->data.sym, *subquery = NULL;
     844       11087 :         dnode *l = sym->data.lval->h, *n;
     845       11087 :         char *tname = NULL;
     846       11087 :         char *fname = qname_schema_object(l->data.lval);
     847       11087 :         char *sname = qname_schema(l->data.lval);
     848             : 
     849       11087 :         tl = sa_list(sql->sa);
     850       11087 :         exps = sa_list(sql->sa);
     851       11087 :         if (l->next)
     852       11087 :                 l = l->next; /* skip distinct */
     853       11087 :         if (l->next) { /* table call with subquery */
     854       11087 :                 int is_value = 1;
     855       11087 :                 if (l->next->type == type_symbol || l->next->type == type_list) {
     856       11087 :                         exp_kind iek = {type_value, card_set, TRUE};
     857       11087 :                         int count = 0;
     858             : 
     859       11087 :                         if (l->next->type == type_symbol)
     860             :                                 n = l->next;
     861             :                         else
     862       11087 :                                 n = l->next->data.lval?l->next->data.lval->h:NULL;
     863             : 
     864       13873 :                         for (dnode *m = n; m; m = m->next) {
     865        2786 :                                 if (m->type == type_symbol && m->data.sym->token == SQL_SELECT)
     866        2786 :                                         subquery = m->data.sym;
     867        2786 :                                 count++;
     868             :                         }
     869       11087 :                         if (subquery && count > 1)
     870          17 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: The input for the table returning function %s%s%s'%s' must be either a single sub query, or a list of values",
     871             :                                                                  sname ? "'":"", sname ? sname : "", sname ? "'.":"", fname);
     872             : 
     873       11082 :                         if (subquery) {
     874          40 :                                 if (!(sq = rel_subquery(query, subquery, ek)))
     875             :                                         return NULL;
     876             :                                 is_value = 0;
     877             :                         } else {
     878       13767 :                                 for ( ; n; n = n->next) {
     879        2731 :                                         sql_exp *e = rel_value_exp(query, &outer, n->data.sym, sql_sel | sql_from, iek);
     880             : 
     881        2731 :                                         if (!e)
     882             :                                                 return NULL;
     883        2725 :                                         append(exps, e);
     884        2725 :                                         is_value &= exp_is_atom(e);
     885             :                                 }
     886       11036 :                                 if (!is_value || (lateral && outer))
     887          14 :                                         sq = rel_project(sql->sa, NULL, exps);
     888          14 :                                 if (lateral && outer) {
     889           0 :                                         sq = rel_crossproduct(sql->sa, sq, outer, op_join);
     890           0 :                                         set_dependent(sq);
     891           0 :                                         set_processed(sq);
     892             :                                 }
     893             :                         }
     894             :                 }
     895       11075 :                 if (!is_value && (!sq || (!lateral && outer)))
     896           0 :                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: no such table returning function %s%s%s'%s'", sname ? "'":"", sname ? sname : "", sname ? "'.":"", fname);
     897       11075 :                 if (!is_value) {
     898          53 :                         if (list_length(exps))
     899          14 :                                 exps = sa_list(sql->sa);
     900         171 :                         for (node *en = sq->exps->h; en; en = en->next) {
     901         118 :                                 sql_exp *e = en->data;
     902             : 
     903         118 :                                 if (!e->alias.label)
     904          28 :                                         exp_label(sql->sa, e, ++sql->label);
     905         118 :                                 sql_exp *ne = exp_ref(sql, e);
     906             :                                 /* allow for table functions with table input */
     907         118 :                                 ne->card = CARD_ATOM;
     908         118 :                                 exp_setname(sql, ne, tname, exp_name(e));
     909         118 :                                 append(exps, ne);
     910         118 :                                 append(tl, exp_subtype(e));
     911             :                         }
     912             :                 } else {
     913       13722 :                         for (node *en = exps->h; en; en = en->next)
     914        2700 :                                 append(tl, exp_subtype(en->data));
     915             :                 }
     916             :         }
     917             : 
     918       11075 :         rel = NULL;
     919       11075 :         if (ast->data.lval->t->type == type_symbol && ast->data.lval->t->data.sym)
     920        1234 :                 tname = ast->data.lval->t->data.sym->data.lval->h->data.sval;
     921             :         else
     922        9841 :                 tname = make_label(sql->sa, ++sql->label);
     923             : 
     924       11075 :         if (!sname && strcmp(fname, "proto_loader") == 0) {
     925          72 :                 rel = rel_proto_loader(sql, exps, tl, tname);
     926          72 :                 if (!rel)
     927             :                         return NULL;
     928         795 :         } else if (!sname && strcmp(fname, "file_loader") == 0) {
     929          35 :                 rel = rel_file_loader(sql, exps, tl, tname);
     930          35 :                 if (!rel)
     931             :                         return NULL;
     932       20914 :         } else if (!(e = find_table_function(sql, sname, fname, list_empty(exps) ? NULL : exps, tl, F_UNION)))
     933             :                 return NULL;
     934             : 
     935       10946 :         if (!rel) {
     936       10946 :                 rel = sq;
     937             : 
     938             :                 /* column or table function */
     939       10946 :                 sf = e->f;
     940       10946 :                 if (e->type != e_func || sf->func->type != F_UNION)
     941           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: '%s' does not return a table", exp_func_name(e));
     942             : 
     943       10946 :                 if (sq) {
     944         171 :                         for (node *n = sq->exps->h, *m = sf->func->ops->h ; n && m ; n = n->next, m = m->next) {
     945         118 :                                 sql_exp *e = (sql_exp*) n->data;
     946         118 :                                 sql_arg *a = (sql_arg*) m->data;
     947         118 :                                 if (!exp_subtype(e) && rel_set_type_param(sql, &(a->type), sq, e, 0) < 0)
     948             :                                         return NULL;
     949             :                         }
     950             :                 }
     951             : 
     952             :                 /* for each column add table.column name */
     953       10946 :                 exps = new_exp_list(sql->sa);
     954      107603 :                 for (m = sf->func->res->h; m; m = m->next) {
     955       96657 :                         sql_arg *a = m->data;
     956       96657 :                         sql_exp *e = exp_column(sql->sa, tname, a->name, &a->type, CARD_MULTI, 1, 0, 0);
     957       96657 :                         e->alias.label = -(sql->nid++);
     958             : 
     959       96657 :                         set_basecol(e);
     960       96657 :                         append(exps, e);
     961             :                 }
     962       21839 :                 rel = rel_table_func(sql->sa, rel, e, exps, (sq)?TABLE_FROM_RELATION:TABLE_PROD_FUNC);
     963             :         }
     964       11015 :         if (ast->data.lval->t->type == type_symbol && ast->data.lval->t->data.sym && ast->data.lval->t->data.sym->data.lval->h->next->data.lval) {
     965          49 :                 rel = rel_table_optname(sql, rel, ast->data.lval->t->data.sym, refs);
     966       10966 :         } else if (refs) { /* if this relation is under a FROM clause, check for duplicate names */
     967       10966 :                 if (list_find(refs, tname, (fcmp) &strcmp))
     968           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: relation name \"%s\" specified more than once", tname);
     969       10966 :                 list_append(refs, tname);
     970             :         }
     971             :         return rel;
     972             : }
     973             : 
     974             : static sql_exp*
     975        4210 : exp_tuples_set_supertype(mvc *sql, list *tuple_values, sql_exp *tuples)
     976             : {
     977        4210 :         assert(is_values(tuples));
     978        4210 :         list *vals = exp_get_values(tuples);
     979        4210 :         if (!vals || !vals->h)
     980             :                 return NULL;
     981             : 
     982        4210 :         int tuple_width = list_length(tuple_values), i;
     983        4210 :         sql_subtype *types = SA_NEW_ARRAY(sql->sa, sql_subtype, tuple_width);
     984        4210 :         bool *has_type = SA_NEW_ARRAY(sql->sa, bool, tuple_width);
     985        4210 :         node *n;
     986             : 
     987        4210 :         memset(has_type, 0, sizeof(bool)*tuple_width);
     988       13716 :         for(n = tuple_values->h, i = 0; n; n = n->next, i++) {
     989        9506 :                 sql_exp *e = n->data;
     990        9506 :                 if (exp_subtype(e)) {
     991        9504 :                         types[i] = *exp_subtype(e);
     992        9504 :                         has_type[i] = 1;
     993             :                 }
     994             :         }
     995             : 
     996        8422 :         for (node *m = vals->h; m; m = m->next) {
     997        4212 :                 sql_exp *tuple = m->data;
     998        4212 :                 if (is_values(tuple)) {
     999           0 :                         list *exps = tuple->f;
    1000           0 :                         for(n = exps->h, i = 0; n; n = n->next, i++) {
    1001           0 :                                 sql_subtype *tpe;
    1002           0 :                                 sql_exp *e = n->data;
    1003             : 
    1004           0 :                                 if (has_type[i] && e->type == e_atom && !e->l && !e->r && !e->f && !e->tpe.type) {
    1005           0 :                                         if (set_type_param(sql, types+i, e->flag) == 0)
    1006           0 :                                                 e->tpe = types[i];
    1007             :                                         else
    1008             :                                                 return NULL;
    1009             :                                 }
    1010           0 :                                 tpe = exp_subtype(e);
    1011           0 :                                 if (!tpe)
    1012             :                                         return NULL;
    1013           0 :                                 if (has_type[i] && tpe) {
    1014           0 :                                         supertype(types+i, types+i, tpe);
    1015             :                                 } else {
    1016           0 :                                         has_type[i] = 1;
    1017           0 :                                         types[i] = *tpe;
    1018             :                                 }
    1019             :                         }
    1020             :                 } else {
    1021        4212 :                         sql_rel *tuple_relation = exp_rel_get_rel(sql->sa, tuple);
    1022             : 
    1023       13722 :                         for(n = tuple_relation->exps->h, i = 0; n; n = n->next, i++) {
    1024        9510 :                                 sql_subtype *tpe;
    1025        9510 :                                 sql_exp *e = n->data;
    1026             : 
    1027        9510 :                                 if (has_type[i] && e->type == e_atom && !e->l && !e->r && !e->f && !e->tpe.type) {
    1028           5 :                                         if (set_type_param(sql, types+i, e->flag) == 0)
    1029           5 :                                                 e->tpe = types[i];
    1030             :                                         else
    1031             :                                                 return NULL;
    1032             :                                 }
    1033        9510 :                                 tpe = exp_subtype(e);
    1034        9510 :                                 if (!tpe)
    1035             :                                         return NULL;
    1036        9510 :                                 if (has_type[i] && tpe) {
    1037        9508 :                                         cmp_supertype(types+i, types+i, tpe);
    1038             :                                 } else {
    1039           2 :                                         has_type[i] = 1;
    1040           2 :                                         types[i] = *tpe;
    1041             :                                 }
    1042             :                         }
    1043             :                 }
    1044             :         }
    1045             : 
    1046        8422 :         for (node *m = vals->h; m; m = m->next) {
    1047        4212 :                 sql_exp *tuple = m->data;
    1048        4212 :                 if (is_values(tuple)) {
    1049           0 :                         list *exps = tuple->f;
    1050           0 :                         list *nexps = sa_list(sql->sa);
    1051           0 :                         for(n = exps->h, i = 0; n; n = n->next, i++) {
    1052           0 :                                 sql_exp *e = n->data;
    1053             : 
    1054           0 :                                 e = exp_check_type(sql, types+i, NULL, e, type_equal);
    1055           0 :                                 if (!e)
    1056             :                                         return NULL;
    1057           0 :                                 exp_label(sql->sa, e, ++sql->label);
    1058           0 :                                 append(nexps, e);
    1059             :                         }
    1060           0 :                         tuple->f = nexps;
    1061             :                 } else {
    1062        4212 :                         sql_rel *tuple_relation = exp_rel_get_rel(sql->sa, tuple);
    1063             : 
    1064        4212 :                         list *nexps = sa_list(sql->sa);
    1065       13722 :                         for(n = tuple_relation->exps->h, i = 0; n; n = n->next, i++) {
    1066        9510 :                                 sql_exp *e = n->data;
    1067             : 
    1068        9510 :                                 e = exp_check_type(sql, types+i, NULL, e, type_equal);
    1069        9510 :                                 if (!e)
    1070             :                                         return NULL;
    1071        9510 :                                 append(nexps, e);
    1072             :                         }
    1073        4212 :                         tuple_relation->exps = nexps;
    1074             :                 }
    1075             :         }
    1076             :         return tuples;
    1077             : }
    1078             : 
    1079             : static int
    1080        9506 : rel_binop_check_types(mvc *sql, sql_rel *rel, sql_exp *ls, sql_exp *rs, int upcast)
    1081             : {
    1082        9506 :         sql_subtype *t1 = exp_subtype(ls), *t2 = exp_subtype(rs);
    1083             : 
    1084        9506 :         if (!t1 || !t2) {
    1085           2 :                 if (t2 && !t1 && rel_set_type_param(sql, t2, rel, ls, upcast) < 0)
    1086             :                         return -1;
    1087           2 :                 if (t1 && !t2 && rel_set_type_param(sql, t1, rel, rs, upcast) < 0)
    1088             :                         return -1;
    1089             :         }
    1090        9506 :         if (!exp_subtype(ls) && !exp_subtype(rs)) {
    1091           0 :                 (void) sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) on both sides of an expression");
    1092           0 :                 return -1;
    1093             :         }
    1094             :         return 0;
    1095             : }
    1096             : 
    1097             : static list *
    1098        4210 : tuples_check_types(mvc *sql, list *tuple_values, sql_exp *tuples)
    1099             : {
    1100        4210 :         list *tuples_list = exp_get_values(tuples);
    1101        4210 :         sql_exp *first_tuple = tuples_list->h->data;
    1102        4210 :         list *nvalues = sa_list(sql->sa);
    1103        4210 :         if (is_values(first_tuple)) {
    1104           0 :                 list *vals = first_tuple->f;
    1105             : 
    1106           0 :                 for (node *n = tuple_values->h, *m = vals->h; n && m; n = n->next, m = m->next) {
    1107           0 :                         sql_exp *le = n->data, *re = m->data;
    1108             : 
    1109           0 :                         if (rel_binop_check_types(sql, NULL, le, re, 0) < 0)
    1110             :                                 return NULL;
    1111           0 :                         if ((le = exp_check_type(sql, exp_subtype(re), NULL, le, type_equal)) == NULL)
    1112             :                                 return NULL;
    1113           0 :                         append(nvalues, le);
    1114             :                 }
    1115             :                 return nvalues;
    1116             :         } else {
    1117        4210 :                 sql_rel *tuple_relation = exp_rel_get_rel(sql->sa, first_tuple);
    1118             : 
    1119        4210 :                 assert(list_length(tuple_values) == list_length(tuple_relation->exps));
    1120       13716 :                 for (node *n = tuple_values->h, *m = tuple_relation->exps->h; n && m; n = n->next, m = m->next) {
    1121        9506 :                         sql_exp *le = n->data, *re = m->data;
    1122             : 
    1123        9506 :                         if (rel_binop_check_types(sql, NULL, le, re, 0) < 0)
    1124             :                                 return NULL;
    1125        9506 :                         if ((le = exp_check_type(sql, exp_subtype(re), NULL, le, type_equal)) == NULL)
    1126             :                                 return NULL;
    1127        9506 :                         append(nvalues, le);
    1128             :                 }
    1129             :                 return nvalues;
    1130             :         }
    1131             :         return tuple_values;
    1132             : }
    1133             : 
    1134             : static sql_rel *
    1135        3941 : rel_values(sql_query *query, symbol *tableref, list *refs)
    1136             : {
    1137        3941 :         mvc *sql = query->sql;
    1138        3941 :         sql_rel *r = NULL;
    1139        3941 :         symbol *values = tableref;
    1140        3941 :         symbol *optname = NULL;
    1141        3941 :         if (tableref->token == SQL_TABLE) {
    1142        1410 :                 values = tableref->data.lval->h->data.sym;
    1143        1410 :                 optname = tableref->data.lval->t->type == type_symbol ? tableref->data.lval->t->data.sym : NULL;
    1144             :         }
    1145        3941 :         dlist *rowlist = values->data.lval;
    1146        3941 :         node *m;
    1147        3941 :         list *exps = sa_list(sql->sa);
    1148        3941 :         exp_kind ek = {type_value, card_value, TRUE};
    1149             : 
    1150       15679 :         for (dnode *o = rowlist->h; o; o = o->next) {
    1151       11748 :                 dlist *values = o->data.lval;
    1152             : 
    1153       11748 :                 if (!list_empty(exps) && list_length(exps) != dlist_length(values)) {
    1154           3 :                         return sql_error(sql, 02, SQLSTATE(42000) "VALUES: number of columns doesn't match between rows");
    1155             :                 } else {
    1156       11745 :                         dnode *n;
    1157             : 
    1158       11745 :                         if (list_empty(exps)) {
    1159       12485 :                                 for (n = values->h; n; n = n->next) {
    1160        8544 :                                         sql_exp *vals = exp_values(sql->sa, sa_list(sql->sa));
    1161             : 
    1162        8544 :                                         exp_label(sql->sa, vals, ++sql->label);
    1163        8544 :                                         list_append(exps, vals);
    1164             :                                 }
    1165             :                         }
    1166       37179 :                         for (n = values->h, m = exps->h; n && m; n = n->next, m = m->next) {
    1167       25441 :                                 sql_exp *vals = m->data;
    1168       25441 :                                 list *vals_list = vals->f;
    1169       25441 :                                 sql_exp *e = rel_value_exp(query, NULL, n->data.sym, sql_sel | sql_values, ek);
    1170       25441 :                                 if (!e)
    1171             :                                         return NULL;
    1172       25434 :                                 list_append(vals_list, e);
    1173             :                         }
    1174             :                 }
    1175             :         }
    1176             : 
    1177             :         /* loop to check types and cardinality */
    1178        3931 :         unsigned int card = exps->h && list_length(((sql_exp*)exps->h->data)->f) > 1 ? CARD_MULTI : CARD_ATOM;
    1179       12458 :         for (m = exps->h; m; m = m->next) {
    1180        8530 :                 sql_exp *e = m->data;
    1181             : 
    1182        8530 :                 if (!(e = exp_values_set_supertype(sql, e, NULL)))
    1183             :                         return NULL;
    1184        8527 :                 e->card = card;
    1185        8527 :                 m->data = e;
    1186             :         }
    1187             : 
    1188        3928 :         r = rel_project(sql->sa, NULL, exps);
    1189        3928 :         r->nrcols = list_length(exps);
    1190        3928 :         r->card = card;
    1191        3928 :         return rel_table_optname(sql, r, optname, refs);
    1192             : }
    1193             : 
    1194             : static int
    1195      513933 : check_is_lateral(symbol *tableref)
    1196             : {
    1197      513933 :         if (tableref->token == SQL_NAME || tableref->token == SQL_TABLE ||
    1198             :                 tableref->token == SQL_VALUES) {
    1199      462915 :                 if (dlist_length(tableref->data.lval) == 3)
    1200      462914 :                         return tableref->data.lval->h->next->data.i_val;
    1201             :                 return 0;
    1202             :         } else if (tableref->token == SQL_WITH) {
    1203         993 :                 if (dlist_length(tableref->data.lval) == 5)
    1204          16 :                         return tableref->data.lval->h->next->next->next->data.i_val;
    1205             :                 return 0;
    1206             :         } else if (tableref->token == SQL_SELECT) {
    1207        2327 :                 SelectNode *sn = (SelectNode *) tableref;
    1208        2327 :                 return sn->lateral;
    1209             :         } else if (tableref->token == SQL_EXCEPT || tableref->token == SQL_INTERSECT ||
    1210             :                            tableref->token == SQL_UNION) {
    1211       11116 :                 if (dlist_length(tableref->data.lval) == 6)
    1212        9361 :                         return tableref->data.lval->h->next->next->next->next->data.i_val;
    1213             :                 return 0;
    1214             :         } else {
    1215             :                 return 0;
    1216             :         }
    1217             : }
    1218             : 
    1219             : sql_rel *
    1220          25 : rel_reduce_on_column_privileges(mvc *sql, sql_rel *rel, sql_table *t)
    1221             : {
    1222          25 :         list *exps = sa_list(sql->sa);
    1223             : 
    1224          99 :         for (node *n = rel->exps->h, *m = ol_first_node(t->columns); n && m; n = n->next, m = m->next) {
    1225          74 :                 sql_exp *e = n->data;
    1226          74 :                 sql_column *c = m->data;
    1227             : 
    1228          74 :                 if (column_privs(sql, c, PRIV_SELECT))
    1229          12 :                         append(exps, e);
    1230             :         }
    1231          25 :         if (!list_empty(exps)) {
    1232           9 :                 rel->exps = exps;
    1233           9 :                 return rel;
    1234             :         }
    1235             :         return NULL;
    1236             : }
    1237             : 
    1238             : sql_rel *
    1239      704451 : table_ref(sql_query *query, symbol *tableref, int lateral, list *refs)
    1240             : {
    1241      704451 :         mvc *sql = query->sql;
    1242      704451 :         char *tname = NULL;
    1243      704451 :         sql_table *t = NULL;
    1244      704451 :         sql_rel *res = NULL;
    1245             : 
    1246      704451 :         if (tableref->token == SQL_NAME) {
    1247      484219 :                 dlist *name = tableref->data.lval->h->data.lval;
    1248      484219 :                 sql_rel *temp_table = NULL;
    1249      484219 :                 char *sname = qname_schema(name);
    1250      484219 :                 int allowed = 1;
    1251             : 
    1252      484219 :                 tname = qname_schema_object(name);
    1253             : 
    1254      484219 :                 if (dlist_length(name) > 2)
    1255           4 :                         return sql_error(sql, 02, SQLSTATE(3F000) "SELECT: only a schema and table name expected");
    1256             : 
    1257      484215 :                 if (!sname)
    1258      110111 :                         temp_table = stack_find_rel_view(sql, tname);
    1259      110111 :                 if (!temp_table)
    1260      471259 :                         t = find_table_or_view_on_scope(sql, NULL, sname, tname, "SELECT", false);
    1261      484215 :                 if (!t && !temp_table)
    1262             :                         return NULL;
    1263      484108 :                 if (!temp_table && !table_privs(sql, t, PRIV_SELECT))
    1264      484108 :                         allowed = 0;
    1265             : 
    1266      484108 :                 if (tableref->data.lval->t->type == type_symbol && tableref->data.lval->t->data.sym) /* AS */
    1267      286760 :                         tname = tableref->data.lval->t->data.sym->data.lval->h->data.sval;
    1268      484108 :                 if (temp_table && !t) {
    1269       12956 :                         node *n;
    1270       12956 :                         int needed = !is_simple_project(temp_table->op);
    1271             : 
    1272       12956 :                         if (is_basetable(temp_table->op) && !temp_table->exps) {
    1273           0 :                                 if (strcmp(rel_base_name(temp_table), tname) != 0)
    1274           0 :                                         rel_base_rename(temp_table, tname);
    1275             :                         } else {
    1276       42450 :                                 for (n = temp_table->exps->h; n && !needed; n = n->next) {
    1277       29494 :                                         sql_exp *e = n->data;
    1278             : 
    1279       29494 :                                         if (!exp_relname(e) || strcmp(exp_relname(e), tname) != 0)
    1280             :                                                 needed = 1;
    1281             :                                 }
    1282             : 
    1283       12956 :                                 if (needed) {
    1284        5824 :                                         list *exps = rel_projections(sql, temp_table, NULL, 1, 1);
    1285             : 
    1286        5824 :                                         temp_table = rel_project(sql->sa, temp_table, exps);
    1287       18731 :                                         for (n = exps->h; n; n = n->next) {
    1288       12907 :                                                 sql_exp *e = n->data;
    1289             : 
    1290       12907 :                                                 noninternexp_setname(sql, e, tname, NULL);
    1291       12907 :                                                 set_basecol(e);
    1292             :                                         }
    1293        5824 :                                         list_hash_clear(exps);
    1294             :                                 }
    1295             :                         }
    1296       12956 :                         if (temp_table && tableref->data.lval->t->type == type_symbol && tableref->data.lval->t->data.sym && tableref->data.lval->t->data.sym->data.lval->h->next->data.lval) /* AS with column aliases */
    1297          10 :                                 temp_table = rel_table_optname(sql, temp_table, tableref->data.lval->t->data.sym, refs);
    1298       12956 :                         if (allowed)
    1299             :                                 return temp_table;
    1300           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: access denied for %s to table '%s'", get_string_global_var(sql, "current_user"), tname);
    1301      471152 :                 } else if (isView(t)) {
    1302             :                         /* instantiate base view */
    1303       72117 :                         node *n,*m;
    1304       72117 :                         sql_rel *rel;
    1305             : 
    1306       72117 :                         if (sql->emode == m_deps) {
    1307       25335 :                                 rel = rel_basetable(sql, t, tname);
    1308       25335 :                                 if (!allowed)
    1309           0 :                                         rel_base_disallow(rel);
    1310             :                         } else {
    1311             :                                 /* when recreating a view, the view itself can't be found */
    1312       46782 :                                 if (sql->objid && sql->objid == t->base.id)
    1313           1 :                                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: attempting to recursively bind view '%s'.'%s'", t->s->base.name, tname);
    1314       46781 :                                 rel = rel_parse(sql, t->s, t->query, m_instantiate);
    1315       46781 :                                 if (rel && sql->emode == m_deps)
    1316           0 :                                         rel = rel_unnest(sql, rel);
    1317             :                         }
    1318             : 
    1319       72114 :                         if (!rel)
    1320           2 :                                 return NULL;
    1321             :                         /* Rename columns of the rel_parse relation */
    1322       72114 :                         if (sql->emode != m_deps) {
    1323       46779 :                                 assert(is_project(rel->op));
    1324       46779 :                                 set_processed(rel);
    1325       46779 :                                 if (is_mset(rel->op) || is_simple_project(rel->op) || (is_groupby(rel->op) && !list_empty(rel->r))) {
    1326             :                                         /* it's unsafe to set the projection names because of possible dependent sorting/grouping columns */
    1327       46779 :                                         rel = rel_project(sql->sa, rel, rel_projections(sql, rel, NULL, 0, 0));
    1328       46779 :                                         set_processed(rel);
    1329             :                                 }
    1330      420905 :                                 for (n = ol_first_node(t->columns), m = rel->exps->h; n && m; n = n->next, m = m->next) {
    1331      374126 :                                         sql_column *c = n->data;
    1332      374126 :                                         sql_exp *e = m->data;
    1333             : 
    1334      374126 :                                         exp_setname(sql, e, tname, c->base.name);
    1335      374126 :                                         set_basecol(e);
    1336             :                                 }
    1337       46779 :                                 list_hash_clear(rel->exps);
    1338             :                         }
    1339       72114 :                         if (rel && !allowed && t->query && (rel = rel_reduce_on_column_privileges(sql, rel, t)) == NULL)
    1340          16 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: access denied for %s to view '%s.%s'", get_string_global_var(sql, "current_user"), t->s->base.name, tname);
    1341       72098 :                         return rel;
    1342             :                 }
    1343      399035 :                 if ((isMergeTable(t) || isReplicaTable(t)) && list_length(t->members)==0)
    1344          14 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s '%s'.'%s' should have at least one table associated",
    1345          14 :                                                         TABLE_TYPE_DESCRIPTION(t->type, t->properties), t->s->base.name, tname);
    1346      399021 :                 res = rel_basetable(sql, t, tname);
    1347      399021 :                 if (!allowed) {
    1348          53 :                         rel_base_disallow(res);
    1349          53 :                         if (rel_base_has_column_privileges(sql, res) == 0)
    1350          62 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: access denied for %s to %s '%s.%s'", get_string_global_var(sql, "current_user"), isView(t) ? "view" : "table", t->s->base.name, tname);
    1351             :                 }
    1352      398990 :                 if (tableref->data.lval->t->type == type_symbol && tableref->data.lval->t->data.sym && tableref->data.lval->t->data.sym->data.lval->h->next->data.lval) { /* AS with column aliases */
    1353          12 :                         res = rel_table_optname(sql, res, tableref->data.lval->t->data.sym, refs);
    1354      398978 :                 } else if (refs) { /* if this relation is under a FROM clause, check for duplicate names */
    1355      398973 :                         if (list_find(refs, tname, (fcmp) &strcmp))
    1356           7 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: relation name \"%s\" specified more than once", tname);
    1357      398966 :                         assert(tname);
    1358      398966 :                         list_append(refs, tname);
    1359             :                 }
    1360      398983 :                 return res;
    1361      220232 :         } else if (tableref->token == SQL_VALUES || (tableref->token == SQL_TABLE && tableref->data.lval->h->data.sym->token == SQL_VALUES)) {
    1362        1447 :                 return rel_values(query, tableref, refs);
    1363      218785 :         } else if (tableref->token == SQL_TABLE) {
    1364       11087 :                 return rel_named_table_function(query, NULL, tableref, lateral, refs);
    1365             :         } else if (tableref->token == SQL_SELECT) {
    1366      102959 :                 return rel_subquery_optname(query, tableref, refs);
    1367             :         } else if (tableref->token == SQL_UNION || tableref->token == SQL_EXCEPT || tableref->token == SQL_INTERSECT) {
    1368             :                 /* subqueries will be called, ie no need to test for duplicate references */
    1369       43712 :                 sql_rel *tq = rel_setquery(query, tableref);
    1370             : 
    1371       43712 :                 if (!tq)
    1372             :                         return NULL;
    1373             :                 /* look for lateral joins */
    1374       43705 :                 symbol *optname = tableref->data.lval->t->type == type_symbol ? tableref->data.lval->t->data.sym : NULL;
    1375       43705 :                 return rel_table_optname(sql, tq, optname, refs);
    1376             :         } else {
    1377       61027 :                 return query_exp_optname(query, tableref, refs);
    1378             :         }
    1379             : }
    1380             : 
    1381             : static sql_exp *
    1382      123895 : rel_exp_variable_on_scope(mvc *sql, const char *sname, const char *vname)
    1383             : {
    1384      123895 :         sql_subtype *tpe = NULL;
    1385      123895 :         sql_var *var = NULL;
    1386      123895 :         sql_arg *a = NULL;
    1387      123895 :         int level = 1;
    1388             : 
    1389      123895 :         if (find_variable_on_scope(sql, sname, vname, &var, &a, &tpe, &level, "SELECT")) {
    1390      122944 :                 if (var) /* if variable is known from the stack or a global var */
    1391       37948 :                         return exp_param_or_declared(sql->sa, var->sname ? sa_strdup(sql->sa, var->sname) : NULL, sa_strdup(sql->sa, var->name), &(var->var.tpe), level);
    1392       90054 :                 if (a) /* if variable is a parameter */
    1393       90054 :                         return exp_param_or_declared(sql->sa, NULL, sa_strdup(sql->sa, vname), &(a->type), level);
    1394             :         }
    1395             :         return NULL;
    1396             : }
    1397             : 
    1398             : static sql_exp *
    1399        9259 : exps_get_exp(list *exps, int nth)
    1400             : {
    1401        9259 :         node *n = NULL;
    1402        9259 :         int i = 0;
    1403             : 
    1404        9259 :         if (exps)
    1405       27182 :                 for (n=exps->h, i=1; n && i<nth; n=n->next, i++)
    1406             :                         ;
    1407        9259 :         if (n && i == nth)
    1408        9256 :                 return n->data;
    1409             :         return NULL;
    1410             : }
    1411             : 
    1412             : static sql_rel *
    1413       31278 : rel_find_groupby(sql_rel *groupby)
    1414             : {
    1415       31278 :         if (groupby && !is_processed(groupby) && !is_base(groupby->op)) {
    1416       41048 :                 while(!is_processed(groupby) && !is_base(groupby->op)) {
    1417       33115 :                         if (is_groupby(groupby->op) || !groupby->l)
    1418             :                                 break;
    1419             :                         if (groupby->l)
    1420             :                                 groupby = groupby->l;
    1421             :                 }
    1422       31153 :                 if (groupby && is_groupby(groupby->op))
    1423       23220 :                         return groupby;
    1424             :         }
    1425             :         return NULL;
    1426             : }
    1427             : 
    1428             : static int
    1429         307 : is_groupby_col(sql_rel *gb, sql_exp *e)
    1430             : {
    1431         307 :         gb = rel_find_groupby(gb);
    1432             : 
    1433         307 :         if (gb) {
    1434         174 :                 if (exp_relname(e)) {
    1435         163 :                         if (exp_name(e) && exps_bind_column2(gb->r, exp_relname(e), exp_name(e), NULL))
    1436             :                                 return 1;
    1437             :                 } else {
    1438          11 :                         if (exp_name(e) && exps_bind_column(gb->r, exp_name(e), NULL, NULL, 1))
    1439             :                                 return 1;
    1440             :                 }
    1441             :         }
    1442             :         return 0;
    1443             : }
    1444             : 
    1445             : static void
    1446        4834 : set_dependent_( sql_rel *r)
    1447             : {
    1448        4834 :         if (is_select(r->op))
    1449        3658 :                 r = r->l;
    1450        4834 :         if (r && is_join(r->op))
    1451        1238 :                 set_dependent(r);
    1452        4834 : }
    1453             : 
    1454             : static
    1455          15 : sql_rel* find_union(visitor *v, sql_rel *rel) {
    1456          15 :         if (rel->op == op_union || rel->op == op_munion)
    1457           1 :                 v->data = rel;
    1458          15 :         return rel;
    1459             : }
    1460             : 
    1461             : static inline
    1462          54 : bool group_by_pk_project_uk_cond(mvc* sql, sql_rel* inner, sql_exp* exp,const char* sname, const char* tname) {
    1463          54 :         sql_table* t = find_table_or_view_on_scope(sql, NULL, sname, tname, "SELECT", false);
    1464          54 :         bool allow = false;
    1465          54 :         if (t) {
    1466          17 :                 sql_idx* pki = NULL;
    1467          17 :                 list *ukil = sa_list(sql->sa);
    1468             : 
    1469          23 :                 for (node * n = ol_first_node(t->idxs); n; n = n->next) {
    1470           6 :                         sql_idx *i = n->data;
    1471           6 :                         if (!i->key)
    1472           1 :                                 continue;
    1473           5 :                         switch (i->key->type) {
    1474           3 :                         case pkey:
    1475           3 :                                 pki = i;
    1476           3 :                                 continue;
    1477           2 :                         case ukey:
    1478             :                         case unndkey:
    1479           2 :                                 list_append(ukil, i);
    1480           2 :                                 continue;
    1481           0 :                         default:
    1482           0 :                                 continue;
    1483             :                         }
    1484             :                 }
    1485          17 :                 if (pki && pki->columns->cnt == 1 && inner->r && ((list*) inner->r)->cnt == 1) {
    1486             :                         /* for now only check simple case where primary key and group by expression is a single column*/
    1487           3 :                         sql_exp* gbe = ((list*) inner->r)->h->data;
    1488           3 :                         sql_column* pkc = ((sql_kc *)pki->columns->h->data)->c;
    1489           3 :                         if (gbe->type == e_column && strcmp(gbe->alias.name, pkc->base.name) == 0) {
    1490           2 :                                 node *n;
    1491           2 :                                 for (n = ukil->h; n; n = n->next){
    1492           2 :                                         sql_idx* uki = n->data;
    1493           2 :                                         if (uki->columns->cnt == 1) {
    1494             :                                                 /* for now only check simple case where unique key is a single column*/
    1495           2 :                                                 sql_column* ukc = ((sql_kc *)uki->columns->h->data)->c;
    1496           2 :                                                 if (strcmp(exp->alias.name, ukc->base.name) == 0) {
    1497             :                                                         allow = true;
    1498             :                                                         break;
    1499             :                                                 }
    1500             :                                         }
    1501             :                                 }
    1502             :                         }
    1503             :                 }
    1504             : 
    1505           2 :                 if (allow) {
    1506             :                         /* sufficiency condition: abort if relation contains union subrelation
    1507             :                         * because it may break functional dependency between pk and uk */
    1508           2 :                         visitor v = {.sql=sql};
    1509           2 :                         rel_visitor_topdown(&v, inner, &find_union);
    1510           2 :                         if (v.data)
    1511           1 :                                 allow = false;
    1512             :                 }
    1513             :         }
    1514             : 
    1515          54 :         return allow;
    1516             : 
    1517             : }
    1518             : 
    1519             : static sql_exp *
    1520     2414585 : rel_column_ref(sql_query *query, sql_rel **rel, symbol *column_r, int f)
    1521             : {
    1522     2414585 :         mvc *sql = query->sql;
    1523     2414585 :         sql_exp *exp = NULL;
    1524     2414585 :         dlist *l = NULL;
    1525     2414585 :         sql_rel *inner = rel?*rel:NULL, *outer = NULL;
    1526     2414585 :         int used_lower_after_processed = 0;
    1527             : 
    1528     2414585 :         assert((column_r->token == SQL_COLUMN || column_r->token == SQL_IDENT) && column_r->type == type_list);
    1529     2414585 :         l = column_r->data.lval;
    1530             : 
    1531     2414585 :         if (dlist_length(l) == 1) {
    1532     1113061 :                 const char *name = l->h->data.sval;
    1533     1113061 :                 if (!name)
    1534             :                         return NULL;
    1535             : 
    1536     1113061 :                 if (!exp && inner)
    1537     1016120 :                         if (!(exp = rel_bind_column(sql, inner, name, f, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    1538             :                                 return NULL;
    1539     1113047 :                 if (!exp && inner && ((is_sql_aggr(f) && (is_groupby(inner->op) || is_select(inner->op))) ||
    1540       24313 :                                                      (is_groupby(inner->op) && inner->flag))) {
    1541             :                         /* if inner is selection, ie having clause, get the left relation to reach group by */
    1542             :                         sql_rel *gp = inner;
    1543         258 :                         while (gp && is_select(gp->op))
    1544          27 :                                 gp = gp->l;
    1545         231 :                         if (gp && !is_basetable(gp->op) && gp->l && !(exp = rel_bind_column(sql, gp->l, name, f, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    1546             :                                 return NULL;
    1547             :                 }
    1548     1113047 :                 if (!exp && query && query_has_outer(query)) {
    1549         472 :                         int i;
    1550             : 
    1551         525 :                         for (i=query_has_outer(query)-1; i>= 0 && !exp && (outer = query_fetch_outer(query,i)); i--) {
    1552         496 :                                 if (!(exp = rel_bind_column(sql, outer, name, f, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    1553             :                                         return NULL;
    1554          84 :                                 if (!exp && is_groupby(outer->op)) {
    1555          31 :                                         if (!(exp = rel_bind_column(sql, outer->l, name, f, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    1556             :                                                 return NULL;
    1557             :                                         else
    1558          31 :                                                 used_lower_after_processed = is_processed(outer);
    1559             :                                 }
    1560         496 :                                 if (exp && is_simple_project(outer->op) && !rel_find_exp(outer, exp))
    1561           0 :                                         exp = rel_project_add_exp(sql, outer, exp);
    1562         443 :                                 if (exp)
    1563             :                                         break;
    1564             :                         }
    1565         472 :                         if (exp && exp->card != CARD_AGGR && is_groupby(outer->op) && !is_sql_aggr(f) && rel_find_exp(outer->l, exp))
    1566           5 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", name);
    1567         467 :                         if (exp && outer && outer->card <= CARD_AGGR && exp->card > CARD_AGGR && !is_sql_aggr(f))
    1568           0 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", name);
    1569         467 :                         if (exp && outer && !is_sql_aggr(f) && !is_sql_aggr(query_fetch_outer_state(query, i))) {
    1570         299 :                                 if (used_lower_after_processed || query_outer_used_exp( query, i, exp, f)) {
    1571           0 :                                         sql_exp *lu = used_lower_after_processed?exp:query_outer_last_used(query, i);
    1572           0 :                                         if (exp_name(lu) && exp_relname(lu) && !has_label(lu))
    1573           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column \"%s.%s\" from outer query", exp_relname(lu), exp_name(lu));
    1574           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query");
    1575             :                                 }
    1576             :                         }
    1577         467 :                         if (exp) {
    1578         438 :                                 int of = query_fetch_outer_state(query, i);
    1579         438 :                                 if (is_groupby(outer->op) && !is_sql_aggr(f)) {
    1580          37 :                                         exp = rel_groupby_add_aggr(sql, outer, exp);
    1581          37 :                                         exp->card = CARD_ATOM;
    1582         401 :                                 } else if (is_groupby(outer->op) && is_sql_aggr(f) && exps_any_match(outer->exps, exp))
    1583          57 :                                         exp = exp_ref(sql, exp);
    1584             :                                 else
    1585         344 :                                         exp->card = CARD_ATOM;
    1586         438 :                                 set_freevar(exp, i);
    1587         438 :                                 if (!is_sql_where(of) && !is_sql_aggr(of) && !is_sql_aggr(f) && !outer->grouped)
    1588         147 :                                         set_outer(outer);
    1589             :                         }
    1590         467 :                         if (exp && outer && (is_select(outer->op) || is_join(outer->op)))
    1591          43 :                                 set_dependent_(outer);
    1592             :                 }
    1593             : 
    1594             :                 /* some views are just in the stack, like before and after updates views */
    1595     1113042 :                 if (rel && sql->use_views) {
    1596          13 :                         sql_rel *v = NULL;
    1597          13 :                         int dup = stack_find_rel_view_projection_columns(sql, name, &v); /* trigger views are basetables relations, so those may conflict */
    1598             : 
    1599          13 :                         if (dup < 0 || (v && exp && *rel && is_base(v->op) && v != *rel)) /* comparing pointers, ugh */
    1600           2 :                                 return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "SELECT: identifier '%s' ambiguous", name);
    1601          11 :                         if (v && !exp) {
    1602           4 :                                 if (*rel)
    1603           1 :                                         *rel = rel_crossproduct(sql->sa, *rel, rel_dup(v), op_join);
    1604             :                                 else
    1605           3 :                                         *rel = rel_dup(v);
    1606           4 :                                 exp = rel_bind_column(sql, *rel, name, f, 0);
    1607             :                         }
    1608             :                 }
    1609     1113040 :                 if (!exp) /* If no column was found, try a variable or parameter */
    1610      120840 :                         exp = rel_exp_variable_on_scope(sql, NULL, name);
    1611             : 
    1612      120840 :                 if (!exp) {
    1613         951 :                         if (inner && !is_sql_aggr(f) && is_groupby(inner->op) && inner->l && (exp = rel_bind_column(sql, inner->l, name, f, 0)))
    1614          47 :                                 return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", name);
    1615             :                 }
    1616             : 
    1617     1112089 :                 if (!exp)
    1618         904 :                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: identifier '%s' unknown", name);
    1619     1112089 :                 if (exp && inner && inner->card <= CARD_AGGR && exp->card > CARD_AGGR && (is_sql_sel(f) || is_sql_having(f)) && (!is_sql_aggr(f) && !(inner->flag)))
    1620           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", name);
    1621     1112089 :                 if (exp && inner && is_groupby(inner->op) && !is_sql_aggr(f) && !is_freevar(exp) && !inner->flag)
    1622       26410 :                         exp = rel_groupby_add_aggr(sql, inner, exp);
    1623     1301524 :         } else if (dlist_length(l) == 2 || dlist_length(l) == 3) {
    1624     1301524 :                 const char *sname = NULL;
    1625     1301524 :                 const char *tname = l->h->data.sval;
    1626     1301524 :                 const char *cname = l->h->next->data.sval;
    1627     1301524 :                 if (dlist_length(l) == 3) {
    1628           1 :                         sname = l->h->data.sval;
    1629           1 :                         tname = l->h->next->data.sval;
    1630           1 :                         cname = l->h->next->next->data.sval;
    1631             :                 }
    1632     1301524 :                 if (!cname)
    1633             :                         return NULL;
    1634             : 
    1635     1286794 :                 if (!exp && rel && inner)
    1636     1286667 :                         if (!(exp = rel_bind_column3(sql, inner, sname, tname, cname, f)) && sql->session->status == -ERR_AMBIGUOUS)
    1637             :                                 return NULL;
    1638     1286793 :                 if (!exp && inner && is_sql_aggr(f) && (is_groupby(inner->op) || is_select(inner->op))) {
    1639             :                         /* if inner is selection, ie having clause, get the left relation to reach group by */
    1640             :                         sql_rel *gp = inner;
    1641          59 :                         while (gp && is_select(gp->op))
    1642           5 :                                 gp = gp->l;
    1643          54 :                         if (gp && !is_basetable(gp->op) && gp->l && !(exp = rel_bind_column3(sql, gp->l, sname, tname, cname, f)) && sql->session->status == -ERR_AMBIGUOUS)
    1644             :                                 return NULL;
    1645             :                 }
    1646     1286793 :                 if (!exp && query && query_has_outer(query)) {
    1647       12094 :                         int i;
    1648             : 
    1649       12304 :                         for (i=query_has_outer(query)-1; i>= 0 && !exp && (outer = query_fetch_outer(query,i)); i--) {
    1650       12303 :                                 if (!(exp = rel_bind_column3(sql, outer, sname, tname, cname, f | sql_outer)) && sql->session->status == -ERR_AMBIGUOUS)
    1651             :                                         return NULL;
    1652         293 :                                 if (!exp && is_groupby(outer->op)) {
    1653         102 :                                         if (!(exp = rel_bind_column3(sql, outer->l, sname, tname, cname, f)) && sql->session->status == -ERR_AMBIGUOUS)
    1654             :                                                 return NULL;
    1655             :                                         else
    1656         102 :                                                 used_lower_after_processed = is_processed(outer);
    1657             :                                 }
    1658       12303 :                                 if (exp && is_simple_project(outer->op) && !rel_find_exp(outer, exp))
    1659           0 :                                         exp = rel_project_add_exp(sql, outer, exp);
    1660       12093 :                                 if (exp)
    1661             :                                         break;
    1662             :                         }
    1663       12094 :                         if (exp && exp->card != CARD_AGGR && is_groupby(outer->op) && !is_sql_aggr(f) && rel_find_exp(outer->l, exp))
    1664           6 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", tname, cname);
    1665       12088 :                         if (exp && outer && outer->card <= CARD_AGGR && exp->card > CARD_AGGR && !is_sql_aggr(f))
    1666           0 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", tname, cname);
    1667       12088 :                         if (exp && outer && !is_sql_aggr(f)) {
    1668       11811 :                                 if (used_lower_after_processed || query_outer_used_exp( query, i, exp, f)) {
    1669           0 :                                         sql_exp *lu = used_lower_after_processed?exp:query_outer_last_used(query, i);
    1670           0 :                                         if (exp_name(lu) && exp_relname(lu) && !has_label(lu))
    1671           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column \"%s.%s\" from outer query", exp_relname(lu), exp_name(lu));
    1672           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query");
    1673             :                                 }
    1674             :                         }
    1675       12088 :                         if (exp) {
    1676       12087 :                                 int of = query_fetch_outer_state(query, i);
    1677       12087 :                                 if (is_groupby(outer->op) && !is_sql_aggr(f)) {
    1678          64 :                                         exp = rel_groupby_add_aggr(sql, outer, exp);
    1679          64 :                                         exp->card = CARD_ATOM;
    1680       12023 :                                 } else if (is_groupby(outer->op) && is_sql_aggr(f) && exps_any_match(outer->exps, exp))
    1681          52 :                                         exp = exp_ref(sql, exp);
    1682             :                                 else
    1683       11971 :                                         exp->card = CARD_ATOM;
    1684       12087 :                                 set_freevar(exp, i);
    1685       12087 :                                 if (!is_sql_where(of) && !is_sql_aggr(of) && !is_sql_aggr(f) && !outer->grouped)
    1686       10373 :                                         set_outer(outer);
    1687             :                         }
    1688       12088 :                         if (exp && outer && (is_select(outer->op) || is_join(outer->op)))
    1689        4791 :                                 set_dependent_(outer);
    1690             :                 }
    1691             : 
    1692             :                 /* some views are just in the stack, like before and after updates views */
    1693     1286787 :                 if (rel && sql->use_views) {
    1694          40 :                         sql_rel *v = stack_find_rel_view(sql, tname);
    1695             : 
    1696          40 :                         if (v && exp && *rel && is_base(v->op) && v != *rel) /* trigger views are basetables relations, so those may conflict */
    1697           0 :                                 return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "SELECT: identifier '%s.%s' ambiguous", tname, cname);
    1698          40 :                         if (v && !exp) {
    1699          20 :                                 if (*rel)
    1700           0 :                                         *rel = rel_crossproduct(sql->sa, *rel, rel_dup(v), op_join);
    1701             :                                 else
    1702          20 :                                         *rel = rel_dup(v);
    1703          20 :                                 if (!(exp = rel_bind_column3(sql, *rel, sname, tname, cname, f)) && sql->session->status == -ERR_AMBIGUOUS)
    1704             :                                         return NULL;
    1705             :                         }
    1706             :                 }
    1707     1286767 :                 if (!exp) { /* If no column was found, try a global variable */
    1708       67872 :                         sql_var *var = NULL;
    1709       67872 :                         sql_subtype *tpe = NULL;
    1710       67872 :                         int level = 0;
    1711       67872 :                         sql_arg *a = NULL;
    1712             : 
    1713       67872 :                         if (find_variable_on_scope(sql, tname, cname, &var, &a, &tpe, &level, "SELECT")) { /* search schema with table name, ugh */
    1714          10 :                                 assert(level == 0);
    1715          10 :                                 exp = exp_param_or_declared(sql->sa, sa_strdup(sql->sa, var->sname), sa_strdup(sql->sa, var->name), &(var->var.tpe), 0);
    1716             :                         }
    1717             :                 }
    1718       67872 :                 if (!exp) {
    1719       67862 :                         if (inner && !is_sql_aggr(f) && is_groupby(inner->op) && inner->l && (exp = rel_bind_column3(sql, inner->l, sname, tname, cname, f))) {
    1720          54 :                                 if (group_by_pk_project_uk_cond(sql, inner, exp, sname, tname)) {
    1721             :                                         /* SQL23 feature: very special case where primary key is used in GROUP BY expression and
    1722             :                                          * unique key is in the project list or ORDER BY clause */
    1723           1 :                                         sql->session->status = 0;
    1724           1 :                                         sql->errstr[0] = 0;
    1725           1 :                                         exp->card = CARD_AGGR;
    1726           1 :                                         list_append(inner->exps, exp);
    1727             :                                 }
    1728             :                                 else
    1729          53 :                                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", tname, cname);
    1730             :                         }
    1731             :                 }
    1732             : 
    1733           1 :                 if (!exp)
    1734       67808 :                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42S22) "SELECT: no such column '%s.%s'", tname, cname);
    1735     1218926 :                 if (exp && inner && inner->card <= CARD_AGGR && exp->card > CARD_AGGR && (is_sql_sel(f) || is_sql_having(f)) && !is_sql_aggr(f))
    1736           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", tname, cname);
    1737     1218926 :                 if (exp && inner && is_groupby(inner->op) && !is_sql_aggr(f) && !is_freevar(exp))
    1738       17732 :                         exp = rel_groupby_add_aggr(sql, inner, exp);
    1739           0 :         } else if (dlist_length(l) > 3) {
    1740           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "cross-database references are not implemented");
    1741             :         }
    1742     2331015 :         if (exp && !exp_is_atom(exp) && rel && !outer) {
    1743     2198597 :                 if (query->last_exp && query->last_rel == *rel && !is_sql_aggr(query->last_state) && is_sql_aggr(f)) {
    1744          55 :                         if (!is_groupby(query->last_rel->op) || list_empty(query->last_rel->r) || !exps_find_exp(query->last_rel->r, query->last_exp)) {
    1745          34 :                                 if (exp_relname(query->last_exp) && exp_name(query->last_exp) && !has_label(query->last_exp))
    1746          34 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", exp_relname(query->last_exp), exp_name(query->last_exp));
    1747           0 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    1748             :                         }
    1749             :                 }
    1750     2198563 :                 query->prev = query->last_exp;
    1751     2198563 :                 query->last_exp = exp;
    1752     2198563 :                 query->last_state = f;
    1753     2198563 :                 query->last_rel = *rel;
    1754             :         }
    1755             :         return exp;
    1756             : }
    1757             : 
    1758             : int
    1759     1035512 : rel_convert_types(mvc *sql, sql_rel *ll, sql_rel *rr, sql_exp **L, sql_exp **R, int scale_fixing, check_type tpe)
    1760             : {
    1761     1035512 :         sql_exp *ls = *L;
    1762     1035512 :         sql_exp *rs = *R;
    1763     1035512 :         sql_subtype *lt = exp_subtype(ls);
    1764     1035512 :         sql_subtype *rt = exp_subtype(rs);
    1765             : 
    1766     1035512 :         if (!rt && !lt) {
    1767           2 :                 sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) on both sides of an expression");
    1768           2 :                 return -1;
    1769             :         }
    1770     1035510 :         if (rt && (!lt || !lt->type))
    1771           7 :                  return rel_set_type_param(sql, rt, ll, ls, 0);
    1772     1035503 :         if (lt && (!rt || !rt->type))
    1773         648 :                  return rel_set_type_param(sql, lt, rr, rs, 0);
    1774             : 
    1775     1034855 :         if (rt && lt) {
    1776     1034855 :                 sql_subtype *i = lt;
    1777     1034855 :                 sql_subtype *r = rt;
    1778             : 
    1779     1034855 :                 if (subtype_cmp(lt, rt) != 0 || (tpe == type_equal_no_any && (lt->type->localtype==0 || rt->type->localtype==0))) {
    1780      280060 :                         sql_subtype super;
    1781             : 
    1782      280060 :                         cmp_supertype(&super, r, i);
    1783      280060 :                         if (scale_fixing) {
    1784             :                                 /* convert ls to super type */
    1785      280060 :                                 ls = exp_check_type(sql, &super, ll, ls, tpe);
    1786             :                                 /* convert rs to super type */
    1787      280060 :                                 rs = exp_check_type(sql, &super, rr, rs, tpe);
    1788             :                         } else {
    1789             :                                 /* convert ls to super type */
    1790           0 :                                 super.scale = lt->scale;
    1791           0 :                                 ls = exp_check_type(sql, &super, ll, ls, tpe);
    1792             :                                 /* convert rs to super type */
    1793           0 :                                 super.scale = rt->scale;
    1794           0 :                                 rs = exp_check_type(sql, &super, rr, rs, tpe);
    1795             :                         }
    1796             :                 }
    1797     1034855 :                 *L = ls;
    1798     1034855 :                 *R = rs;
    1799     1034855 :                 if (!ls || !rs)
    1800             :                         return -1;
    1801             :                 return 0;
    1802             :         }
    1803             :         return -1;
    1804             : }
    1805             : 
    1806             : static sql_rel *
    1807      103082 : push_select_exp(mvc *sql, sql_rel *rel, sql_exp *e, sql_exp *ls, int f)
    1808             : {
    1809      103082 :         if (is_outerjoin(rel->op)) {
    1810        1195 :                 if ((is_left(rel->op) || is_full(rel->op)) && rel_find_exp(rel->l, ls)) {
    1811          38 :                         rel_join_add_exp(sql->sa, rel, e);
    1812          38 :                         return rel;
    1813        1157 :                 } else if ((is_right(rel->op) || is_full(rel->op)) && rel_find_exp(rel->r, ls)) {
    1814           5 :                         rel_join_add_exp(sql->sa, rel, e);
    1815           5 :                         return rel;
    1816             :                 }
    1817        1152 :                 if (is_left(rel->op) && rel_find_exp(rel->r, ls)) {
    1818        1125 :                         rel->r = rel_push_select(sql, rel->r, ls, e, f);
    1819        1125 :                         return rel;
    1820          27 :                 } else if (is_right(rel->op) && rel_find_exp(rel->l, ls)) {
    1821          11 :                         rel->l = rel_push_select(sql, rel->l, ls, e, f);
    1822          11 :                         return rel;
    1823             :                 }
    1824             :         }
    1825             :         /* push select into the given relation */
    1826      101903 :         return rel_push_select(sql, rel, ls, e, f);
    1827             : }
    1828             : 
    1829             : static sql_rel *
    1830      211612 : push_join_exp(mvc *sql, sql_rel *rel, sql_exp *e, sql_exp *L, sql_exp *R, sql_exp *R2, int f)
    1831             : {
    1832      211612 :         sql_rel *r;
    1833      211612 :         if (/*is_semi(rel->op) ||*/ (is_outerjoin(rel->op) && !is_processed((rel)))) {
    1834       21446 :                 rel_join_add_exp(sql->sa, rel, e);
    1835       21446 :                 return rel;
    1836             :         }
    1837             :         /* push join into the given relation */
    1838      190166 :         if ((r = rel_push_join(sql, rel, L, R, R2, e, f)) != NULL)
    1839             :                 return r;
    1840           0 :         rel_join_add_exp(sql->sa, rel, e);
    1841           0 :         return rel;
    1842             : }
    1843             : 
    1844             : static sql_rel *
    1845        3759 : rel_select_push_filter_exp_down(mvc *sql, sql_rel *rel, sql_exp *e, list *l, list *r, int ff)
    1846             : {
    1847        3759 :         sql_exp *ll;
    1848        3759 :         if (exps_card(r) <= CARD_ATOM && (exps_are_atoms(r) || exps_have_freevar(sql, r) || exps_have_freevar(sql, l))) {
    1849        3660 :                 if (exps_card(l) == exps_card(r) || rel->processed)  /* bin compare op */
    1850          96 :                         return rel_select(sql->sa, rel, e);
    1851        3564 :                 if ((ll = exps_find_one_multi_exp(l)))
    1852        3564 :                         return push_select_exp(sql, rel, e, ll, ff);
    1853             :         }
    1854          99 :         if (is_outerjoin(rel->op))
    1855           7 :                 return rel_select(sql->sa, rel, e);
    1856          92 :         return rel_select_add_exp(sql->sa, rel, e);
    1857             : }
    1858             : 
    1859             : static sql_rel *
    1860        3640 : rel_filter(mvc *sql, sql_rel *rel, list *l, list *r, char *sname, char *filter_op, int anti, int ff)
    1861             : {
    1862        3640 :         node *n;
    1863        3640 :         sql_exp *e = NULL;
    1864        3640 :         sql_subfunc *f = NULL;
    1865        3640 :         list *tl = sa_list(sql->sa);
    1866        3640 :         bool found = false;
    1867             : 
    1868        7284 :         for (n = l->h; n; n = n->next){
    1869        3644 :                 sql_exp *e = n->data;
    1870             : 
    1871        3644 :                 list_append(tl, exp_subtype(e));
    1872             :         }
    1873       14482 :         for (n = r->h; n; n = n->next){
    1874       10842 :                 sql_exp *e = n->data;
    1875             : 
    1876       10842 :                 list_append(tl, exp_subtype(e));
    1877             :         }
    1878             :         /* find filter function */
    1879        3640 :         f = bind_func_(sql, sname, filter_op, tl, F_FILT, false, &found, false);
    1880        3640 :         if (f) {
    1881        3638 :                 node *n,*m = f->func->ops->h;
    1882        3638 :                 list *nexps = sa_list(sql->sa);
    1883             : 
    1884        7280 :                 for(n=l->h; m && n; m = m->next, n = n->next) {
    1885        3642 :                         sql_arg *a = m->data;
    1886        3642 :                         sql_exp *e = n->data;
    1887             : 
    1888        3642 :                         e = exp_check_type(sql, &a->type, rel, e, type_equal);
    1889        3642 :                         if (!e)
    1890             :                                 return NULL;
    1891        3642 :                         list_append(nexps, e);
    1892             :                 }
    1893        3638 :                 l = nexps;
    1894        3638 :                 nexps = sa_list(sql->sa);
    1895       14472 :                 for(n=r->h; m && n; m = m->next, n = n->next) {
    1896       10834 :                         sql_arg *a = m->data;
    1897       10834 :                         sql_exp *e = n->data;
    1898             : 
    1899       10834 :                         e = exp_check_type(sql, &a->type, rel, e, type_equal);
    1900       10834 :                         if (!e)
    1901             :                                 return NULL;
    1902       10834 :                         list_append(nexps, e);
    1903             :                 }
    1904             :                 r = nexps;
    1905             :         }
    1906        3640 :         if (!f)
    1907           4 :                 return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: %s FILTER function %s%s%s'%s'",
    1908           2 :                                                  found ? "insufficient privileges for" : "no such", sname ? "'":"", sname ? sname : "", sname ? "'.":"", filter_op);
    1909        3638 :         e = exp_filter(sql->sa, l, r, f, anti);
    1910             : 
    1911        3638 :         if (exps_one_is_rel(l) || exps_one_is_rel(r)) /* uncorrelated subquery case */
    1912           9 :                 return rel_select(sql->sa, rel, e);
    1913             :         /* atom or row => select */
    1914        7262 :         for (node *n=l->h; n; n = n->next) {
    1915        3633 :                 sql_exp *ls = n->data;
    1916             : 
    1917        3633 :                 if (ls->card > rel->card) {
    1918           0 :                         if (exp_name(ls) && !has_label(ls))
    1919           0 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(ls));
    1920           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    1921             :                 }
    1922             :         }
    1923       14436 :         for (node *n=r->h; n; n = n->next) {
    1924       10807 :                 sql_exp *rs = n->data;
    1925             : 
    1926       10807 :                 if (rs->card > rel->card) {
    1927           0 :                         if (exp_name(rs) && !has_label(rs))
    1928           0 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(rs));
    1929           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    1930             :                 }
    1931             :         }
    1932        3629 :         return rel_select_push_filter_exp_down(sql, rel, e, l, r, ff);
    1933             : }
    1934             : 
    1935             : static sql_rel *
    1936        3575 : rel_filter_exp_(mvc *sql, sql_rel *rel, sql_exp *ls, sql_exp *r1, sql_exp *r2, sql_exp *r3, char *filter_op, int anti, int f)
    1937             : {
    1938        3575 :         list *l = sa_list(sql->sa);
    1939        3575 :         list *r = sa_list(sql->sa);
    1940             : 
    1941        3575 :         list_append(l, ls);
    1942        3575 :         list_append(r, r1);
    1943        3575 :         if (r2)
    1944        3575 :                 list_append(r, r2);
    1945        3575 :         if (r3)
    1946        3575 :                 list_append(r, r3);
    1947        3575 :         return rel_filter(sql, rel, l, r, "sys", filter_op, anti, f);
    1948             : }
    1949             : 
    1950             : static sql_rel *
    1951      380128 : rel_select_push_compare_exp_down(mvc *sql, sql_rel *rel, sql_exp *e, sql_exp *ls, sql_exp *rs, sql_exp *rs2, int f)
    1952             : {
    1953      380128 :         if (!is_join(rel->op) && !is_select(rel->op))
    1954       85171 :                 return rel_select(sql->sa, rel, e);
    1955      378853 :         if ((rs->card <= CARD_ATOM || (rs2 && ls->card <= CARD_ATOM)) &&
    1956      167792 :                 (exp_is_atom(rs) || (rs2 && exp_is_atom(ls)) || exp_has_freevar(sql, rs) || exp_has_freevar(sql, ls)) &&
    1957         482 :                 (!rs2 || (rs2->card <= CARD_ATOM && (exp_is_atom(rs2) || exp_has_freevar(sql, rs2))))) {
    1958       83345 :                 if (ls->card == rs->card || (rs2 && (ls->card == rs2->card || rs->card == rs2->card)) || rel->processed) /* bin compare op */
    1959        1018 :                         return rel_select(sql->sa, rel, e);
    1960             : 
    1961       82327 :                 return push_select_exp(sql, rel, e, ls, f);
    1962             :         } else { /* join */
    1963      211612 :                 return push_join_exp(sql, rel, e, ls, rs, rs2, f);
    1964             :         }
    1965             :         return rel;
    1966             : }
    1967             : 
    1968             : static sql_rel *
    1969      353056 : rel_compare_exp_(sql_query *query, sql_rel *rel, sql_exp *ls, sql_exp *rs, sql_exp *rs2, int type, int anti, int quantifier, int f, int symmetric, int is_semantics)
    1970             : {
    1971      353056 :         mvc *sql = query->sql;
    1972      353056 :         sql_exp *e = NULL;
    1973             : 
    1974      353056 :         if (quantifier || exp_is_rel(ls) || exp_is_rel(rs) || (rs2 && exp_is_rel(rs2))) {
    1975        3350 :                 if (rs2) {
    1976           7 :                         e = exp_compare2(sql->sa, ls, rs, rs2, type, symmetric);
    1977           7 :                         if (anti)
    1978           0 :                                 set_anti(e);
    1979             :                 } else {
    1980        3343 :                         assert(!symmetric);
    1981        3343 :                         if (rel_convert_types(sql, rel, rel, &ls, &rs, 1, type_equal_no_any) < 0)
    1982             :                                 return NULL;
    1983        3343 :                         e = exp_compare_func(sql, ls, rs, compare_func((comp_type)type, anti), quantifier);
    1984             :                 }
    1985        3350 :                 return rel_select(sql->sa, rel, e);
    1986      349706 :         } else if (!rs2) {
    1987      346780 :                 assert(!symmetric);
    1988      346780 :                 if (ls->card < rs->card) {
    1989        8825 :                         sql_exp *swap = ls;
    1990        8825 :                         ls = rs;
    1991        8825 :                         rs = swap;
    1992        8825 :                         type = (int)swap_compare((comp_type)type);
    1993             :                 }
    1994      346780 :                 if (rel_convert_types(sql, rel, rel, &ls, &rs, 1, type_equal_no_any) < 0)
    1995             :                         return NULL;
    1996      346777 :                 e = exp_compare(sql->sa, ls, rs, type);
    1997      346777 :                 if (is_semantics) set_semantics(e);
    1998             :         } else {
    1999        2926 :                 assert(rs2);
    2000        2926 :                 if (rel_convert_types(sql, rel, rel, &ls, &rs, 1, type_equal_no_any) < 0)
    2001             :                         return NULL;
    2002        2926 :                 if (!(rs2 = exp_check_type(sql, exp_subtype(ls), rel, rs2, type_equal)))
    2003             :                         return NULL;
    2004        2926 :                 e = exp_compare2(sql->sa, ls, rs, rs2, type, symmetric);
    2005             :         }
    2006      349703 :         if (anti)
    2007         837 :                 set_anti(e);
    2008             : 
    2009      349703 :         if (!rel)
    2010           0 :                 return rel_select(sql->sa, rel_project_exp(sql, exp_atom_bool(sql->sa, 1)), e);
    2011             : 
    2012             :         /* atom or row => select */
    2013      349703 :         if (ls->card > rel->card || rs->card > rel->card || (rs2 && rs2->card > rel->card)) {
    2014           0 :                 sql_exp *e = ls->card > rel->card ? ls : rs->card > rel->card ? rs : rs2;
    2015           0 :                 if (exp_name(e) && !has_label(e))
    2016           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(e));
    2017           0 :                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    2018             :         }
    2019      349703 :         return rel_select_push_compare_exp_down(sql, rel, e, ls, rs, rs2, f);
    2020             : }
    2021             : 
    2022             : static sql_rel *
    2023      350123 : rel_compare_exp(sql_query *query, sql_rel *rel, sql_exp *ls, sql_exp *rs, char *compare_op, int reduce, int quantifier, int need_not, int f, int is_semantics)
    2024             : {
    2025      350123 :         mvc *sql = query->sql;
    2026      350123 :         comp_type type = cmp_equal;
    2027             : 
    2028      350123 :         if (!ls || !rs)
    2029             :                 return NULL;
    2030             : 
    2031      350123 :         if (!quantifier && ((!rel && !query_has_outer(query)) || !reduce)) {
    2032             :                 /* TODO to handle filters here */
    2033           0 :                 sql_exp *e;
    2034             : 
    2035           0 :                 if (rel_convert_types(sql, rel, rel, &ls, &rs, 1, type_equal_no_any) < 0)
    2036             :                         return NULL;
    2037           0 :                 e = rel_binop_(sql, rel, ls, rs, "sys", compare_op, card_value, true);
    2038             : 
    2039           0 :                 if (!e)
    2040             :                         return NULL;
    2041           0 :                 if (!reduce) {
    2042           0 :                         if (rel->op == op_project) {
    2043           0 :                                 append(rel->exps, e);
    2044             :                         } else {
    2045           0 :                                 list *exps = new_exp_list(sql->sa);
    2046             : 
    2047           0 :                                 append(exps, e);
    2048           0 :                                 return rel_project(sql->sa, rel, exps);
    2049             :                         }
    2050             :                 } else {
    2051           0 :                         return rel_select(sql->sa, rel, e);
    2052             :                 }
    2053             :         }
    2054      350123 :         type = compare_str2type(compare_op);
    2055      350123 :         assert(type != cmp_filter);
    2056      350123 :         return rel_compare_exp_(query, rel, ls, rs, NULL, type, need_not, quantifier, f, 0, is_semantics);
    2057             : }
    2058             : 
    2059             : static sql_rel *
    2060      350094 : rel_compare(sql_query *query, sql_rel *rel, symbol *sc, symbol *lo, symbol *ro, char *compare_op, int f, exp_kind k, int quantifier, int is_semantics)
    2061             : {
    2062      350094 :         mvc *sql = query->sql;
    2063      350094 :         sql_exp *rs = NULL, *ls;
    2064      350094 :         comp_type cmp_type = compare_str2type(compare_op);
    2065      350094 :         exp_kind ek = {type_value, card_column, FALSE};
    2066      350094 :         int need_not = 0;
    2067             : 
    2068      350094 :         if ((quantifier == 1 && cmp_type == cmp_equal) ||
    2069      350059 :             (quantifier == 2 && cmp_type == cmp_notequal)) {
    2070          61 :                 dnode *n = sc->data.lval->h;
    2071          61 :                 dlist *dl = dlist_create(sql->sa);
    2072             :                 /* map into IN/NOT IN */
    2073          61 :                 sc->token = cmp_type==cmp_equal?SQL_IN:SQL_NOT_IN;
    2074          61 :                 n->next->type = type_list;
    2075          61 :                 n->next->data.lval = dl;
    2076          61 :                 n->next->next->next = NULL; /* remove quantifier */
    2077          61 :                 dl->h = n->next->next;
    2078          61 :                 n->next->next = NULL; /* (remove comparison) moved righthand side */
    2079          61 :                 return rel_logical_exp(query, rel, sc, f);
    2080             :         }
    2081             :         /* <> ANY -> NOT (= ALL) */
    2082      350033 :         if (quantifier == 1 && cmp_type == cmp_notequal) {
    2083           5 :                 need_not = 1;
    2084           5 :                 quantifier = 2;
    2085           5 :                 cmp_type = cmp_equal;
    2086           5 :                 compare_op = "=";
    2087             :         }
    2088             : 
    2089      350033 :         if ((lo->token == SQL_SELECT || lo->token == SQL_UNION || lo->token == SQL_EXCEPT || lo->token == SQL_INTERSECT || lo->token == SQL_VALUES) &&
    2090          41 :                 (ro->token != SQL_SELECT && ro->token != SQL_UNION && ro->token != SQL_EXCEPT && ro->token != SQL_INTERSECT && ro->token != SQL_VALUES)) {
    2091          28 :                 symbol *tmp = lo; /* swap subquery to the right hand side */
    2092             : 
    2093          28 :                 lo = ro;
    2094          28 :                 ro = tmp;
    2095             : 
    2096          28 :                 if (compare_op[0] == '>')
    2097          16 :                         compare_op[0] = '<';
    2098          12 :                 else if (compare_op[0] == '<' && compare_op[1] != '>')
    2099           1 :                         compare_op[0] = '>';
    2100          28 :                 cmp_type = swap_compare(cmp_type);
    2101             :         }
    2102             : 
    2103      350033 :         ls = rel_value_exp(query, &rel, lo, f|sql_farg, ek);
    2104      350033 :         if (!ls)
    2105             :                 return NULL;
    2106      349985 :         if (ls && rel && exp_has_freevar(sql, ls) && is_sql_sel(f))
    2107           0 :                 ls = rel_project_add_exp(sql, rel, ls);
    2108      349985 :         if (quantifier)
    2109          57 :                 ek.card = card_set;
    2110             : 
    2111      349985 :         rs = rel_value_exp(query, &rel, ro, f|sql_farg, ek);
    2112      349985 :         if (!rs)
    2113             :                 return NULL;
    2114      349955 :         if (ls->card > rs->card && rs->card == CARD_AGGR && is_sql_having(f)) {
    2115           0 :                 if (exp_name(ls) && exp_relname(ls) && !has_label(ls))
    2116           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", exp_relname(ls), exp_name(ls));
    2117           0 :                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    2118             :         }
    2119      349955 :         if (rs->card > ls->card && ls->card == CARD_AGGR && is_sql_having(f)) {
    2120           0 :                 if (exp_name(rs) && exp_relname(rs) && !has_label(rs))
    2121           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", exp_relname(rs), exp_name(rs));
    2122           0 :                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    2123             :         }
    2124      349955 :         return rel_compare_exp(query, rel, ls, rs, compare_op, k.reduce, quantifier, need_not, f, is_semantics);
    2125             : }
    2126             : 
    2127             : static sql_exp*
    2128      560623 : _rel_nop(mvc *sql, char *sname, char *fname, list *tl, sql_rel *rel, list *exps, exp_kind ek)
    2129             : {
    2130      560623 :         bool found = false;
    2131      560623 :         int table_func = (ek.card == card_relation);
    2132      560623 :         sql_ftype type = (ek.card == card_loader)?F_LOADER:((ek.card == card_none)?F_PROC:
    2133             :                    ((ek.card == card_relation)?F_UNION:F_FUNC));
    2134             : 
    2135      560623 :         sql_subfunc *f = bind_func_(sql, sname, fname, tl, type, false, &found, false);
    2136      560623 :         if (f && !(exps = check_arguments_and_find_largest_any_type(sql, rel, exps, f, table_func, false)))
    2137             :                 f = NULL;
    2138             : 
    2139      560355 :         if (f)
    2140      560355 :                 return exp_op(sql->sa, exps, f);
    2141         267 :         char *arg_list = nary_function_arg_types_2str(sql, tl, list_length(tl));
    2142         551 :         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: %s operator %s%s%s'%s'(%s)",
    2143         267 :                                          found ? "insufficient privileges for" : "no such", sname ? "'":"", sname ? sname : "", sname ? "'.":"", fname, arg_list ? arg_list : "");
    2144             : }
    2145             : 
    2146             : static sql_exp *
    2147        2110 : exp_exist(sql_query *query, sql_rel *rel, sql_exp *le, int exists)
    2148             : {
    2149        2110 :         mvc *sql = query->sql;
    2150        2110 :         sql_subfunc *exists_func = NULL;
    2151        2110 :         sql_subtype *t;
    2152        2110 :         sql_exp *res;
    2153             : 
    2154        2110 :         if (!exp_name(le))
    2155         753 :                 exp_label(sql->sa, le, ++sql->label);
    2156        2110 :         if (exp_is_rel(le)) { /* for the subquery case, propagate to the inner query */
    2157        2110 :                 sql_rel *r = exp_rel_get_rel(sql->sa, le);
    2158        2110 :                 if (is_project(r->op) && !list_empty(r->exps)) {
    2159        6151 :                         for (node *n = r->exps->h; n; n = n->next)
    2160        4041 :                                 if (!exp_subtype(n->data) && rel_set_type_param(sql, sql_bind_localtype("bit"), r, n->data, 0) < 0) /* workaround */
    2161             :                                         return NULL;
    2162        2110 :                         le->tpe = *exp_subtype(r->exps->h->data); /* just take the first expression type */
    2163             :                 }
    2164           0 :         } else if (!exp_subtype(le) && rel_set_type_param(sql, sql_bind_localtype("bit"), rel, le, 0) < 0) /* workaround */
    2165             :                 return NULL;
    2166        2110 :         t = exp_subtype(le);
    2167             : 
    2168        2171 :         if (!(exists_func = sql_bind_func(sql, "sys", exists ? "sql_exists" : "sql_not_exists", t, NULL, F_FUNC, true, true)))
    2169           0 :                 return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "exist operator on type %s missing", t ? t->type->base.name : "unknown");
    2170        2110 :         res = exp_unop(sql->sa, le, exists_func);
    2171        2110 :         set_has_no_nil(res);
    2172        2110 :         return res;
    2173             : }
    2174             : 
    2175             : static sql_exp *
    2176        1105 : rel_exists_value_exp(sql_query *query, sql_rel **rel, symbol *sc, int f)
    2177             : {
    2178        1105 :         exp_kind ek = {type_value, card_exists, FALSE};
    2179        1105 :         sql_exp *le, *e;
    2180             : 
    2181        1105 :         le = rel_value_exp(query, rel, sc->data.sym, f|sql_farg, ek);
    2182        1105 :         if (!le)
    2183             :                 return NULL;
    2184        1104 :         if (!(e = exp_exist(query, rel ? *rel : NULL, le, sc->token == SQL_EXISTS)))
    2185             :                 return NULL;
    2186             :         /* only freevar should have CARD_AGGR */
    2187        1104 :         e->card = CARD_ATOM;
    2188        1104 :         return e;
    2189             : }
    2190             : 
    2191             : static sql_rel *
    2192        1014 : rel_exists_exp(sql_query *query, sql_rel *rel, symbol *sc, int f)
    2193             : {
    2194        1014 :         exp_kind ek = {type_value, card_exists, TRUE};
    2195        1014 :         mvc *sql = query->sql;
    2196        1014 :         sql_rel *sq = NULL;
    2197             : 
    2198        1014 :         if (is_psm_call(f) || is_sql_merge(f))
    2199           3 :                 return sql_error(sql, 02, SQLSTATE(42000) "%s: subqueries not supported inside %s", is_psm_call(f) ? "CALL" : "MERGE", is_psm_call(f) ? "CALL statements" : "MERGE conditions");
    2200        1013 :         if (rel) {
    2201        1013 :                 query_processed(query);
    2202        1013 :                 query_push_outer(query, rel, f);
    2203             :         }
    2204        1013 :         sq = rel_subquery(query, sc->data.sym, ek);
    2205        1013 :         if (rel)
    2206        1013 :                 rel = query_pop_outer(query);
    2207        1013 :         assert(!is_sql_sel(f));
    2208        1013 :         if (sq) {
    2209        1006 :                 sql_exp *e = exp_rel(sql, sq);
    2210        1006 :                 if (!(e = exp_exist(query, rel, e, sc->token == SQL_EXISTS)))
    2211             :                         return NULL;
    2212             :                 /* only freevar should have CARD_AGGR */
    2213        1006 :                 e->card = CARD_ATOM;
    2214        1006 :                 rel = rel_select_add_exp(sql->sa, rel, e);
    2215        1006 :                 return rel;
    2216             :         }
    2217             :         return NULL;
    2218             : }
    2219             : 
    2220             : static int
    2221        8662 : is_project_true(sql_rel *r)
    2222             : {
    2223        8662 :         if (r && !r->l && list_length(r->exps) == 1) {
    2224          82 :                 sql_exp *e = r->exps->h->data;
    2225          82 :                 if (exp_is_atom(e) && exp_is_true(e))
    2226             :                         return 1;
    2227             :         }
    2228             :         return 0;
    2229             : }
    2230             : 
    2231             : static sql_exp *
    2232       65202 : rel_in_value_exp(sql_query *query, sql_rel **rel, symbol *sc, int f)
    2233             : {
    2234       65202 :         mvc *sql = query->sql;
    2235       65202 :         exp_kind ek = {type_value, card_column, TRUE};
    2236       65202 :         dlist *dl = sc->data.lval;
    2237       65202 :         symbol *lo = NULL;
    2238       65202 :         dnode *n = dl->h->next, *dn = NULL;
    2239       65202 :         sql_exp *le = NULL, *re, *e = NULL;
    2240       65202 :         list *ll = sa_list(sql->sa);
    2241       65202 :         int is_tuple = 0, add_select = 0;
    2242             : 
    2243             :         /* complex case */
    2244             :         //if (dl->h->type == type_list) { /* (a,b..) in (.. ) */
    2245       65202 :         if (dl->h->type == type_symbol && dl->h->data.sym->token == SQL_VALUES) {
    2246        4227 :                 lo = dl->h->data.sym;
    2247        4227 :                 dn = lo->data.lval->h->data.lval->h;
    2248        4227 :                 lo = dn->data.sym;
    2249        4227 :                 dn = dn->next;
    2250             :         } else {
    2251       60975 :                 lo = dl->h->data.sym;
    2252             :         }
    2253      206220 :         for( ; lo; lo = dn?dn->data.sym:NULL, dn = dn?dn->next:NULL ) {
    2254       70513 :                 le = rel_value_exp(query, rel, lo, f|sql_farg, ek);
    2255       70513 :                 if (!le)
    2256             :                         return NULL;
    2257       70509 :                 append(ll, le);
    2258             :         }
    2259       65198 :         if (list_length(ll) == 1) {
    2260       60973 :                 le = ll->h->data;
    2261       60973 :                 ek.card = card_set;
    2262             :         } else {
    2263        4225 :                 le = exp_values(sql->sa, ll);
    2264        4225 :                 exp_label(sql->sa, le, ++sql->label);
    2265        4225 :                 ek.type = list_length(ll);
    2266        4225 :                 is_tuple = list_length(ll);
    2267             :         }
    2268             :         /* list of values or subqueries */
    2269             :         /* 3 right hand side cases,
    2270             :          *      1 - value/tuple list of atoms
    2271             :          *      2 - list of scalar sub queries (needs check on 1 row/column per query)
    2272             :          *      3 - single tabular ie. multi column, multi row query
    2273             :          */
    2274       65198 :         if (n->type == type_list) {
    2275       65198 :                 sql_exp *values;
    2276       65198 :                 list *vals = sa_list(sql->sa);
    2277             : 
    2278       65198 :                 n = dl->h->next;
    2279       65198 :                 n = n->data.lval->h;
    2280             : 
    2281      319782 :                 for (; n; n = n->next) {
    2282      254623 :                         if (n->type == type_list) {
    2283           0 :                                 dnode *m = n->data.lval->h;
    2284           0 :                                 list *rl = sa_list(sql->sa);
    2285           0 :                                 for (; m; m = m->next) {
    2286           0 :                                         assert(m->type == type_symbol);
    2287           0 :                                         sql_exp *re = rel_value_exp(query, rel, m->data.sym, f|sql_farg, ek);
    2288           0 :                                         if (!re)
    2289             :                                                 return NULL;
    2290           0 :                                         append(rl, re);
    2291             :                                 }
    2292           0 :                                 re = exp_values(sql->sa, rl);
    2293             :                         } else {
    2294      254623 :                                 assert(n->type == type_symbol);
    2295      254623 :                                 re = rel_value_exp(query, rel, n->data.sym, f|sql_farg, ek);
    2296             :                         }
    2297      254623 :                         if (!re)
    2298             :                                 return NULL;
    2299      254599 :                         if (is_tuple && (!exp_is_rel(re) && !is_values(re)))
    2300           4 :                                 return sql_error(sql, 02, SQLSTATE(42000) "Cannot match a tuple to a single value");
    2301        4223 :                         if (is_tuple && exp_is_rel(re)) {
    2302        4223 :                                 sql_rel *r = exp_rel_get_rel(sql->sa, re);
    2303             : 
    2304        4223 :                                 if (!r)
    2305           0 :                                         return sql_error(sql, 02, SQLSTATE(42000) "Subquery missing");
    2306        4223 :                                 if (r->nrcols != ek.type)
    2307          12 :                                         return sql_error(sql, 02, SQLSTATE(42000) "Subquery has too %s columns", (r->nrcols < ek.type) ? "few" : "many");
    2308        4214 :                                 re = exp_rel_label(sql, re);
    2309        4214 :                                 if (r && r->l && n->next) /* not a value list */
    2310           2 :                                         return sql_error(sql, 02, SQLSTATE(42000) "subquery must return only one column");
    2311      250372 :                         } else if (exp_is_rel(re)) {
    2312        8663 :                                 sql_rel *r = exp_rel_get_rel(sql->sa, re);
    2313        8663 :                                 add_select = 1;
    2314        8663 :                                 if (rel && *rel && is_join((*rel)->op))
    2315        4506 :                                         set_dependent((*rel));
    2316        8663 :                                 if (is_project(r->op) && is_project_true(r->l) && list_length(r->exps) == 1)
    2317          80 :                                         re = r->exps->h->data;
    2318      241709 :                         } else if (is_values(re) && is_tuple != list_length(exp_get_values(re))) {
    2319           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "Tuple sizes do not match");
    2320             :                         }
    2321      254584 :                         append(vals, re);
    2322             :                 }
    2323             : 
    2324       65159 :                 if (list_empty(vals))
    2325           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "The list of values for IN operator cannot be empty");
    2326             : 
    2327       65159 :                 values = exp_values(sql->sa, vals);
    2328       65159 :                 exp_label(sql->sa, values, ++sql->label);
    2329       65159 :                 if (is_tuple) {
    2330        4210 :                         if (!(values = exp_tuples_set_supertype(sql, exp_get_values(le), values)))
    2331             :                                 return NULL;
    2332        4210 :                         if (!(le->f = tuples_check_types(sql, exp_get_values(le), values)))
    2333             :                                 return NULL;
    2334             :                 } else { /* if it's not a tuple, enforce coercion on the type for every element on the list */
    2335       60949 :                         sql_subtype super, *le_tpe = exp_subtype(le), *values_tpe = NULL;
    2336             : 
    2337      311320 :                         for (node *m = vals->h; m; m = m->next) { /* first get values supertype */
    2338      250371 :                                 sql_exp *e = m->data;
    2339      250371 :                                 sql_subtype *tpe = exp_subtype(e);
    2340             : 
    2341      250371 :                                 if (values_tpe && tpe) {
    2342      189419 :                                         cmp_supertype(&super, values_tpe, tpe);
    2343      189419 :                                         *values_tpe = super;
    2344       60952 :                                 } else if (!values_tpe && tpe) {
    2345       60945 :                                         super = *tpe;
    2346       60945 :                                         values_tpe = &super;
    2347             :                                 }
    2348             :                         }
    2349       60949 :                         if (!le_tpe)
    2350           2 :                                 le_tpe = values_tpe;
    2351       60949 :                         if (!values_tpe)
    2352           4 :                                 values_tpe = le_tpe;
    2353       60949 :                         if (!le_tpe || !values_tpe)
    2354           1 :                                 return sql_error(sql, 01, SQLSTATE(42000) "For the IN operator, both sides must have a type defined");
    2355       60948 :                         cmp_supertype(&super, values_tpe, le_tpe); /* compute supertype */
    2356             : 
    2357             :                         /* on selection/join cases we can generate cmp expressions instead of anyequal for trivial cases */
    2358       60948 :                         if ((is_sql_where(f) || is_sql_having(f)) && !is_sql_farg(f) && !exp_has_rel(le) && exps_are_atoms(vals)) {
    2359       22665 :                                 if (list_length(vals) == 1) { /* use cmp_equal instead of cmp_in for 1 expression */
    2360         112 :                                         sql_exp *first = vals->h->data;
    2361         112 :                                         if (rel_convert_types(sql, rel ? *rel : NULL, rel ? *rel : NULL, &le, &first, 1, type_equal_no_any) < 0)
    2362           0 :                                                 return NULL;
    2363         134 :                                         e = exp_compare(sql->sa, le, first, (sc->token == SQL_IN) ? cmp_equal : cmp_notequal);
    2364             :                                 } else { /* use cmp_in instead of anyequal for n simple expressions */
    2365      108215 :                                         for (node *n = vals->h ; n ; n = n->next)
    2366       85662 :                                                 if ((n->data = exp_check_type(sql, &super, rel ? *rel : NULL, n->data, type_equal)) == NULL)
    2367             :                                                         return NULL;
    2368       22553 :                                         if ((le = exp_check_type(sql, &super, rel ? *rel : NULL, le, type_equal)) == NULL)
    2369             :                                                 return NULL;
    2370       29865 :                                         e = exp_in(sql->sa, le, vals, (sc->token == SQL_IN) ? cmp_in : cmp_notin);
    2371             :                                 }
    2372             :                         }
    2373       22665 :                         if (!e) { /* after computing supertype, check types for each IN value */
    2374      202879 :                                 for (node *n = vals->h ; n ; n = n->next)
    2375      164596 :                                         if ((n->data = exp_check_type(sql, &super, rel ? *rel : NULL, n->data, type_equal)) == NULL)
    2376             :                                                 return NULL;
    2377       38283 :                                 values->tpe = *exp_subtype(vals->h->data);
    2378       38283 :                                 if (!(le = exp_check_type(sql, &super, rel ? *rel : NULL, le, type_equal)))
    2379             :                                         return NULL;
    2380             :                         }
    2381             :                 }
    2382       60948 :                 if (!e) {
    2383       42493 :                         if (add_select && rel && *rel && !is_project((*rel)->op) && !is_select((*rel)->op) && !is_base((*rel)->op))
    2384        4505 :                                 *rel = rel_select(sql->sa, *rel, NULL);
    2385       42493 :                         if ((rel && *rel) || exp_has_rel(le) || exp_has_rel(values))
    2386       36835 :                                 e = exp_in_func(sql, le, values, (sc->token == SQL_IN), is_tuple);
    2387             :                         else
    2388        5658 :                                 e = exp_in_aggr(sql, le, values, (sc->token == SQL_IN), is_tuple);
    2389             :                 }
    2390             :         }
    2391             :         return e;
    2392             : }
    2393             : 
    2394             : static sql_rel *
    2395       32948 : rel_in_exp(sql_query *query, sql_rel *rel, symbol *sc, int f)
    2396             : {
    2397       32948 :         mvc *sql = query->sql;
    2398       32948 :         sql_exp *e = rel_in_value_exp(query, &rel, sc, f);
    2399             : 
    2400       32948 :         assert(!is_sql_sel(f));
    2401       32948 :         if (!e || !rel)
    2402             :                 return NULL;
    2403             : 
    2404       32921 :         if (e->type == e_cmp) { /* it's a exp_in or cmp_equal of simple expressions, push down early on if possible */
    2405       22663 :                 sql_exp *ls = e->l;
    2406       22663 :                 bool rlist = (e->flag == cmp_in || e->flag == cmp_notin);
    2407       22663 :                 unsigned int rcard = rlist ? exps_card(e->r) : exp_card(e->r);
    2408       22663 :                 int r_is_atoms = rlist ? exps_are_atoms(e->r) : exp_is_atom(e->r);
    2409       22663 :                 int r_has_freevar = rlist ? exps_have_freevar(sql, e->r) : exp_has_freevar(sql, e->r);
    2410             : 
    2411       22663 :                 if (rcard <= CARD_ATOM && (r_is_atoms || r_has_freevar || exp_has_freevar(sql, ls))) {
    2412       22663 :                         if ((exp_card(ls) == rcard) || rel->processed) /* bin compare op */
    2413        5472 :                                 return rel_select(sql->sa, rel, e);
    2414             : 
    2415       17191 :                         return push_select_exp(sql, rel, e, ls, f);
    2416             :                 } else { /* join */
    2417           0 :                         sql_exp *rs = rlist ? exps_find_one_multi_exp(e->r) : e->r;
    2418           0 :                         if (rs)
    2419           0 :                                 return push_join_exp(sql, rel, e, ls, rs, NULL, f);
    2420             :                 }
    2421             :         }
    2422       10258 :         if (is_outerjoin(rel->op))
    2423          13 :                 return rel_select(sql->sa, rel, e);
    2424       10245 :         return rel_select_add_exp(sql->sa, rel, e);
    2425             : }
    2426             : 
    2427             : static int
    2428        3111 : exp_between_check_types(sql_subtype *res, sql_subtype *t1, sql_subtype *t2, sql_subtype *t3)
    2429             : {
    2430        3111 :         bool type_found = false;
    2431        3111 :         sql_subtype super;
    2432             : 
    2433        3111 :         if (t1 && t2) {
    2434        3105 :                 cmp_supertype(&super, t2, t1);
    2435        3105 :                 type_found = true;
    2436           6 :         } else if (t1) {
    2437           3 :                 super = *t1;
    2438           3 :                 type_found = true;
    2439           3 :         } else if (t2) {
    2440           1 :                 super = *t2;
    2441           1 :                 type_found = true;
    2442             :         }
    2443        3111 :         if (t3) {
    2444        3108 :                 if (type_found)
    2445        3107 :                         cmp_supertype(&super, t3, &super);
    2446             :                 else
    2447           1 :                         super = *t3;
    2448             :                 type_found = true;
    2449             :         }
    2450           3 :         if (!type_found)
    2451             :                 return -1;
    2452        3110 :         *res = super;
    2453        3110 :         return 0;
    2454             : }
    2455             : 
    2456             : static bool
    2457      128051 : exp_is_null_no_value_opt(sql_exp *e)
    2458             : {
    2459      128051 :         if (!e)
    2460             :                 return false;
    2461      131994 :         while (is_convert(e->type))
    2462        3943 :                 e = e->l;
    2463      256075 :         return e->type == e_atom && e->l && atom_null(e->l);
    2464             : }
    2465             : 
    2466             : sql_exp *
    2467     2737107 : rel_logical_value_exp(sql_query *query, sql_rel **rel, symbol *sc, int f, exp_kind ek)
    2468             : {
    2469     2737107 :         mvc *sql = query->sql;
    2470             : 
    2471     2737107 :         if (!sc)
    2472             :                 return NULL;
    2473             : 
    2474     2737107 :         if (mvc_highwater(sql))
    2475           2 :                 return sql_error(sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
    2476             : 
    2477     2737106 :         switch (sc->token) {
    2478       59870 :         case SQL_OR:
    2479             :         case SQL_AND:
    2480             :         {
    2481       59870 :                 symbol *lo = sc->data.lval->h->data.sym;
    2482       59870 :                 symbol *ro = sc->data.lval->h->next->data.sym;
    2483       59870 :                 sql_exp *ls, *rs;
    2484             : 
    2485       59870 :                 if (!(ls = rel_value_exp(query, rel, lo, f|sql_farg, ek)))
    2486             :                         return NULL;
    2487       59869 :                 if (!(rs = rel_value_exp(query, rel, ro, f|sql_farg, ek)))
    2488             :                         return NULL;
    2489      103701 :                 return rel_binop_(sql, rel ? *rel : NULL, ls, rs, "sys", sc->token == SQL_OR ? "or": "and", card_value, false);
    2490             :         }
    2491           5 :         case SQL_FILTER:
    2492             :                 /* [ x,..] filter [ y,..] */
    2493             :                 /* todo add anti, [ x,..] not filter [ y,...] */
    2494             :                 /* no correlation */
    2495             :         {
    2496           5 :                 dnode *ln = sc->data.lval->h->data.lval->h;
    2497           5 :                 dnode *rn = sc->data.lval->h->next->next->data.lval->h;
    2498           5 :                 dlist *filter_op = sc->data.lval->h->next->data.lval;
    2499           5 :                 char *fname = qname_schema_object(filter_op);
    2500           5 :                 char *sname = qname_schema(filter_op);
    2501           5 :                 list *exps, *tl;
    2502           5 :                 sql_subtype *obj_type = NULL;
    2503             : 
    2504           5 :                 exps = sa_list(sql->sa);
    2505           5 :                 tl = sa_list(sql->sa);
    2506          15 :                 for (; ln; ln = ln->next) {
    2507           5 :                         symbol *sym = ln->data.sym;
    2508             : 
    2509           5 :                         sql_exp *e = rel_value_exp(query, rel, sym, f|sql_farg, ek);
    2510           5 :                         if (!e)
    2511             :                                 return NULL;
    2512           5 :                         if (!obj_type)
    2513           5 :                                 obj_type = exp_subtype(e);
    2514           5 :                         list_append(exps, e);
    2515           5 :                         append(tl, exp_subtype(e));
    2516             :                 }
    2517          13 :                 for (; rn; rn = rn->next) {
    2518          10 :                         symbol *sym = rn->data.sym;
    2519             : 
    2520          10 :                         sql_exp *e = rel_value_exp(query, rel, sym, f|sql_farg, ek);
    2521          10 :                         if (!e)
    2522             :                                 return NULL;
    2523           8 :                         list_append(exps, e);
    2524           8 :                         append(tl, exp_subtype(e));
    2525             :                 }
    2526             :                 /* find the predicate filter function */
    2527           3 :                 return _rel_nop(sql, sname, fname, tl, rel ? *rel : NULL, exps, ek);
    2528             :         }
    2529      127770 :         case SQL_COMPARE:
    2530             :         {
    2531      127770 :                 dnode *n = sc->data.lval->h;
    2532      127770 :                 symbol *lo = n->data.sym;
    2533      127770 :                 symbol *ro = n->next->next->data.sym;
    2534      127770 :                 char *compare_op = n->next->data.sval;
    2535      127770 :                 int quantifier = 0, need_not = 0;
    2536      127770 :                 sql_exp *rs = NULL, *ls;
    2537      127770 :                 comp_type cmp_type = compare_str2type(compare_op);
    2538      127770 :                 bool is_not_distinct_from = false;
    2539      127770 :                 bool is_distinct_from = false;
    2540             : 
    2541             :                 /*
    2542             :                  * = ANY -> IN, <> ALL -> NOT( = ANY) -> NOT IN
    2543             :                  * = ALL -> all(groupby(all, nil)), <> ANY -> NOT ( = ALL )
    2544             :                  */
    2545      127770 :                 if (n->next->next->next)
    2546         261 :                         quantifier = n->next->next->next->data.i_val + 1;
    2547         261 :                 assert(quantifier == 0 || quantifier == 1 || quantifier == 2 || quantifier == 3 || quantifier == 4);
    2548             : 
    2549             :                 /* [NOT] DISTINCT FROM */
    2550      127770 :                 if (quantifier == 3) {
    2551             :                                 is_not_distinct_from = true;
    2552             :                                 quantifier = 0;
    2553             :                 }
    2554      127755 :                 else if (quantifier == 4) {
    2555          13 :                                 is_distinct_from = true;
    2556          13 :                                 quantifier = 0;
    2557             :                 }
    2558             : 
    2559      127770 :                 if ((quantifier == 1 && cmp_type == cmp_equal) ||
    2560      127703 :                     (quantifier == 2 && cmp_type == cmp_notequal)) {
    2561          95 :                         dlist *dl = dlist_create(sql->sa);
    2562             :                         /* map into IN/NOT IN */
    2563          95 :                         sc->token = cmp_type==cmp_equal?SQL_IN:SQL_NOT_IN;
    2564          95 :                         n->next->type = type_list;
    2565          95 :                         n->next->data.lval = dl;
    2566          95 :                         n->next->next->next = NULL; /* remove quantifier */
    2567          95 :                         dl->h = n->next->next;
    2568          95 :                         n->next->next = NULL; /* (remove comparison) moved righthand side */
    2569          95 :                         return rel_logical_value_exp(query, rel, sc, f, ek);
    2570             :                 }
    2571             :                 /* <> ANY -> NOT (= ALL) */
    2572      127675 :                 if (quantifier == 1 && cmp_type == cmp_notequal) {
    2573          18 :                         need_not = 1;
    2574          18 :                         quantifier = 2;
    2575          18 :                         cmp_type = cmp_equal;
    2576          18 :                         compare_op = "=";
    2577             :                 }
    2578             : 
    2579      127675 :                 ls = rel_value_exp(query, rel, lo, f|sql_farg, ek);
    2580      127675 :                 if (!ls)
    2581             :                         return NULL;
    2582      127663 :                 if (quantifier)
    2583         138 :                         ek.card = card_set;
    2584             : 
    2585      127663 :                 rs = rel_value_exp(query, rel, ro, f|sql_farg, ek);
    2586      127663 :                 if (!rs)
    2587             :                         return NULL;
    2588             : 
    2589      127653 :                 if (is_distinct_from || is_not_distinct_from) {
    2590          28 :                         if (rel_convert_types(sql, rel ? *rel : NULL, rel ? *rel : NULL, &ls, &rs, 1, type_equal_no_any) < 0)
    2591             :                                 return NULL;
    2592          41 :                         sql_exp* e = exp_compare(sql->sa, ls, rs, is_not_distinct_from?cmp_equal:cmp_notequal);
    2593          28 :                         set_semantics(e);
    2594          28 :                         return e;
    2595             :                 }
    2596             : 
    2597      127625 :                 if (rs->type == e_atom)
    2598      112243 :                         quantifier = 0;
    2599             : 
    2600      127625 :                 if (!exp_is_rel(ls) && !exp_is_rel(rs) && ls->card < rs->card) {
    2601        3836 :                         sql_exp *swap = ls; /* has to swap parameters like in the rel_logical_exp case */
    2602        3836 :                         ls = rs;
    2603        3836 :                         rs = swap;
    2604        3836 :                         cmp_type = swap_compare(cmp_type);
    2605             :                 }
    2606             : 
    2607      127625 :                 if (rel_convert_types(sql, rel ? *rel : NULL, rel ? *rel : NULL, &ls, &rs, 1, type_equal_no_any) < 0)
    2608             :                         return NULL;
    2609      127622 :                 if (exp_is_null_no_value_opt(ls) && exp_is_null_no_value_opt(rs))
    2610           7 :                         return exp_atom(sql->sa, atom_general(sql->sa, sql_bind_localtype("bit"), NULL, 0));
    2611             : 
    2612      127615 :                 return exp_compare_func(sql, ls, rs, compare_func(cmp_type, need_not), quantifier);
    2613             :         }
    2614             :         /* Set Member ship */
    2615       32254 :         case SQL_IN:
    2616             :         case SQL_NOT_IN:
    2617       32254 :                 return rel_in_value_exp(query, rel, sc, f);
    2618        1105 :         case SQL_EXISTS:
    2619             :         case SQL_NOT_EXISTS:
    2620        1105 :                 return rel_exists_value_exp(query, rel, sc, f);
    2621         355 :         case SQL_LIKE:
    2622             :         case SQL_NOT_LIKE:
    2623             :         {
    2624         355 :                 symbol *lo = sc->data.lval->h->data.sym;
    2625         355 :                 symbol *ro = sc->data.lval->h->next->data.sym;
    2626         355 :                 int insensitive = sc->data.lval->h->next->next->data.i_val;
    2627         355 :                 int anti = (sc->token == SQL_NOT_LIKE) != (sc->data.lval->h->next->next->next->data.i_val != 0);
    2628         355 :                 sql_subtype *st = sql_bind_localtype("str");
    2629         355 :                 sql_exp *le = rel_value_exp(query, rel, lo, f|sql_farg, ek), *re, *ee = NULL, *ie = exp_atom_bool(sql->sa, insensitive);
    2630             : 
    2631         355 :                 if (!le)
    2632             :                         return NULL;
    2633             : 
    2634         355 :                 if (!exp_subtype(le))
    2635           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: parameter not allowed on "
    2636             :                                         "left hand side of LIKE operator");
    2637             : 
    2638         355 :                 lo = ro->data.lval->h->data.sym;
    2639             :                 /* like uses a single string pattern */
    2640         355 :                 ek.card = card_value;
    2641         355 :                 re = rel_value_exp(query, rel, lo, f|sql_farg, ek);
    2642         355 :                 if (!re)
    2643             :                         return NULL;
    2644         351 :                 if ((re = exp_check_type(sql, st, rel ? *rel : NULL, re, type_equal)) == NULL)
    2645           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "LIKE: wrong type, should be string");
    2646         351 :                 if ((le = exp_check_type(sql, st, rel ? *rel : NULL, le, type_equal)) == NULL)
    2647           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "LIKE: wrong type, should be string");
    2648             :                 /* Do we need to escape ? */
    2649         351 :                 if (dlist_length(ro->data.lval) == 2) {
    2650          63 :                         char *escape = ro->data.lval->h->next->data.sval;
    2651          63 :                         ee = exp_atom(sql->sa, atom_string(sql->sa, st, sa_strdup(sql->sa, escape)));
    2652             :                 } else {
    2653         288 :                         ee = exp_atom(sql->sa, atom_string(sql->sa, st, sa_strdup(sql->sa, "")));
    2654             :                 }
    2655         585 :                 return rel_nop_(sql, rel ? *rel : NULL, le, re, ee, ie, "sys", anti ? "not_like" : "like", card_value);
    2656             :         }
    2657         186 :         case SQL_BETWEEN:
    2658             :         case SQL_NOT_BETWEEN:
    2659             :         {
    2660         186 :                 symbol *lo = sc->data.lval->h->data.sym;
    2661         186 :                 int symmetric = sc->data.lval->h->next->data.i_val;
    2662         186 :                 symbol *ro1 = sc->data.lval->h->next->next->data.sym;
    2663         186 :                 symbol *ro2 = sc->data.lval->h->next->next->next->data.sym;
    2664         186 :                 sql_exp *le, *re1, *re2;
    2665         186 :                 sql_subtype super;
    2666             : 
    2667         186 :                 assert(sc->data.lval->h->next->type == type_int);
    2668             : 
    2669         186 :                 if (!(le = rel_value_exp(query, rel, lo, f|sql_farg, ek)))
    2670             :                         return NULL;
    2671         185 :                 if (!(re1 = rel_value_exp(query, rel, ro1, f|sql_farg, ek)))
    2672             :                         return NULL;
    2673         182 :                 if (!(re2 = rel_value_exp(query, rel, ro2, f|sql_farg, ek)))
    2674             :                         return NULL;
    2675             : 
    2676         177 :                 if (exp_between_check_types(&super, exp_subtype(le), exp_subtype(re1), exp_subtype(re2)) < 0)
    2677           0 :                         return sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) on both sides of an expression");
    2678             : 
    2679         177 :                 if ((le = exp_check_type(sql, &super, rel ? *rel:NULL, le, type_equal)) == NULL ||
    2680         177 :                     (re1 = exp_check_type(sql, &super, rel ? *rel:NULL, re1, type_equal)) == NULL ||
    2681         177 :                     (re2 = exp_check_type(sql, &super, rel ? *rel:NULL, re2, type_equal)) == NULL)
    2682           0 :                         return NULL;
    2683             : 
    2684         177 :                 le = exp_compare2(sql->sa, le, re1, re2, 3, symmetric);
    2685         177 :                 if (sc->token == SQL_NOT_BETWEEN)
    2686          60 :                         set_anti(le);
    2687             :                 return le;
    2688             :         }
    2689       15437 :         case SQL_IS_NULL:
    2690             :         case SQL_IS_NOT_NULL:
    2691             :         /* is (NOT) NULL */
    2692             :         {
    2693       15437 :                 sql_exp *le = rel_value_exp(query, rel, sc->data.sym, f|sql_farg, ek);
    2694             : 
    2695       15437 :                 if (!le)
    2696             :                         return NULL;
    2697       19788 :                 le = rel_unop_(sql, rel ? *rel : NULL, le, "sys", sc->token == SQL_IS_NULL ? "isnull" : "isnotnull", card_value);
    2698       15434 :                 if (!le)
    2699             :                         return NULL;
    2700       15434 :                 set_has_no_nil(le);
    2701       15434 :                 return le;
    2702             :         }
    2703       19488 :         case SQL_NOT: {
    2704       19488 :                 sql_exp *le = rel_value_exp(query, rel, sc->data.sym, f|sql_farg, ek);
    2705             : 
    2706       19488 :                 if (!le)
    2707             :                         return NULL;
    2708       19486 :                 return rel_unop_(sql, rel ? *rel : NULL, le, "sys", "not", card_value);
    2709             :         }
    2710             :         case SQL_ATOM: {
    2711     2459124 :                 AtomNode *an = (AtomNode *) sc;
    2712     2459124 :                 assert(an && an->a);
    2713     2459124 :                 return exp_atom(sql->sa, an->a);
    2714             :         }
    2715       20991 :         case SQL_IDENT:
    2716             :         case SQL_COLUMN:
    2717       20991 :                 return rel_column_ref(query, rel, sc, f);
    2718         202 :         case SQL_UNION:
    2719             :         case SQL_EXCEPT:
    2720             :         case SQL_INTERSECT: {
    2721         202 :                 sql_rel *sq;
    2722             : 
    2723         202 :                 if (is_psm_call(f) || is_sql_merge(f))
    2724           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: subqueries not supported inside %s", is_psm_call(f) ? "CALL" : "MERGE", is_psm_call(f) ? "CALL statements" : "MERGE conditions");
    2725         200 :                 if (rel && *rel)
    2726         142 :                         query_push_outer(query, *rel, f);
    2727         200 :                 sq = rel_setquery(query, sc);
    2728         200 :                 if (rel && *rel) {
    2729         142 :                         *rel = query_pop_outer(query);
    2730         142 :                         if (is_sql_join(f) && is_groupby((*rel)->op)) {
    2731           0 :                                 return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in JOIN conditions");
    2732         142 :                         } else if (is_sql_where(f) && is_groupby((*rel)->op)) {
    2733           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions not allowed in WHERE clause");
    2734         142 :                         } else if ((is_sql_update_set(f) || is_sql_psm(f)) && is_groupby((*rel)->op)) {
    2735           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions not allowed in SET, WHILE, IF, ELSE, CASE, WHEN, RETURN, ANALYZE clauses");
    2736             :                         }
    2737             :                 }
    2738         200 :                 if (!sq)
    2739             :                         return NULL;
    2740         198 :                 if (ek.type == type_value && ek.card <= card_set && is_project(sq->op) && list_length(sq->exps) > 1)
    2741           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery must return only one column");
    2742         197 :                 if (ek.type == type_relation && is_project(sq->op) && list_length(sq->exps) != ek.type)
    2743           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery has too %s columns", list_length(sq->exps) < ek.type ? "few" : "many");
    2744         303 :                 if (ek.type == type_value && ek.card < card_set && sq->card >= CARD_AGGR && (is_sql_sel(f) | is_sql_having(f) | is_sql_farg(f) |
    2745         106 :                         ( is_sql_where(f) && rel && (!*rel || is_basetable((*rel)->op) || is_simple_project((*rel)->op) || is_joinop((*rel)->op)))))
    2746         106 :                         sq = rel_zero_or_one(sql, sq, ek);
    2747         197 :                 return exp_rel(sql, sq);
    2748             :         }
    2749           1 :         case SQL_DEFAULT:
    2750           1 :                 return sql_error(sql, 02, SQLSTATE(42000) "DEFAULT keyword not allowed outside insert and update statements");
    2751         318 :         default: {
    2752         318 :                 sql_exp *le = rel_value_exp(query, rel, sc, f|sql_farg, ek);
    2753         318 :                 sql_subtype bt;
    2754             : 
    2755         318 :                 if (!le)
    2756             :                         return NULL;
    2757         316 :                 sql_find_subtype(&bt, "boolean", 0, 0);
    2758         316 :                 if ((le = exp_check_type(sql, &bt, rel ? *rel : NULL, le, type_equal)) == NULL)
    2759             :                         return NULL;
    2760         316 :                 return rel_binop_(sql, rel ? *rel : NULL, le, exp_atom_bool(sql->sa, 1), "sys", "=", 0, true);
    2761             :         }
    2762             :         }
    2763             :         /* never reached, as all switch cases have a `return` */
    2764             : }
    2765             : 
    2766             : sql_rel *
    2767      629725 : rel_logical_exp(sql_query *query, sql_rel *rel, symbol *sc, int f)
    2768             : {
    2769      629725 :         mvc *sql = query->sql;
    2770      629725 :         exp_kind ek = {type_value, card_column, TRUE};
    2771             : 
    2772      629725 :         if (!sc)
    2773             :                 return NULL;
    2774             : 
    2775      629725 :         if (mvc_highwater(sql))
    2776           0 :                 return sql_error(sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
    2777             : 
    2778      629725 :         switch (sc->token) {
    2779       14559 :         case SQL_OR:
    2780             :         {
    2781       14559 :                 list *exps = NULL, *lexps = NULL, *rexps = NULL;
    2782       14559 :                 symbol *lo = sc->data.lval->h->data.sym;
    2783       14559 :                 symbol *ro = sc->data.lval->h->next->data.sym;
    2784       14559 :                 sql_rel *lr, *rr;
    2785             : 
    2786       14559 :                 if (!rel)
    2787             :                         return NULL;
    2788             : 
    2789       14559 :                 lr = rel;
    2790       14559 :                 rr = rel_dup(lr);
    2791             : 
    2792       14559 :                 if (is_outerjoin(rel->op) && !is_processed(rel)) {
    2793          44 :                         exps = rel->exps;
    2794             : 
    2795          44 :                         lr = rel_select_copy(sql->sa, lr, sa_list(sql->sa));
    2796          44 :                         lr = rel_logical_exp(query, lr, lo, f | sql_or);
    2797          44 :                         if (!lr)
    2798             :                                 return NULL;
    2799          43 :                         query_processed(query);
    2800          43 :                         rr = rel_select_copy(sql->sa, rr, sa_list(sql->sa));
    2801          43 :                         rr = rel_logical_exp(query, rr, ro, f | sql_or);
    2802          43 :                         if (!rr)
    2803             :                                 return NULL;
    2804          42 :                         if (lr->l == rr->l) {
    2805          42 :                                 lexps = lr->exps;
    2806          42 :                                 lr = lr->l;
    2807          42 :                                 rexps = rr->exps;
    2808          42 :                                 rr = rr->l;
    2809             :                         }
    2810          42 :                         rel = NULL;
    2811             :                 } else {
    2812       14515 :                         lr = rel_logical_exp(query, lr, lo, f | sql_or);
    2813       14515 :                         if (!lr)
    2814             :                                 return NULL;
    2815       14513 :                         rr = rel_logical_exp(query, rr, ro, f | sql_or);
    2816             :                 }
    2817             : 
    2818       14555 :                 if (!lr || !rr)
    2819             :                         return NULL;
    2820       14555 :                 return rel_or(sql, rel, lr, rr, exps, lexps, rexps);
    2821             :         }
    2822      193936 :         case SQL_AND:
    2823             :         {
    2824      193936 :                 symbol *lo = sc->data.lval->h->data.sym;
    2825      193936 :                 symbol *ro = sc->data.lval->h->next->data.sym;
    2826      193936 :                 rel = rel_logical_exp(query, rel, lo, f);
    2827      193936 :                 if (!rel)
    2828             :                         return NULL;
    2829      193929 :                 return rel_logical_exp(query, rel, ro, f);
    2830             :         }
    2831          65 :         case SQL_FILTER:
    2832             :                 /* [ x,..] filter [ y,..] */
    2833             :                 /* todo add anti, [ x,..] NOT filter [ y,...] */
    2834             :                 /* no correlation */
    2835             :         {
    2836          65 :                 dnode *ln = sc->data.lval->h->data.lval->h;
    2837          65 :                 dnode *rn = sc->data.lval->h->next->next->data.lval->h;
    2838          65 :                 dlist *filter_op = sc->data.lval->h->next->data.lval;
    2839          65 :                 char *fname = qname_schema_object(filter_op);
    2840          65 :                 char *sname = qname_schema(filter_op);
    2841          65 :                 list *l, *r;
    2842             : 
    2843          65 :                 l = sa_list(sql->sa);
    2844          65 :                 r = sa_list(sql->sa);
    2845         199 :                 for (; ln; ln = ln->next) {
    2846          69 :                         symbol *sym = ln->data.sym;
    2847             : 
    2848          69 :                         sql_exp *e = rel_value_exp(query, &rel, sym, f|sql_farg, ek);
    2849          69 :                         if (!e)
    2850             :                                 return NULL;
    2851          69 :                         list_append(l, e);
    2852             :                 }
    2853         182 :                 for (; rn; rn = rn->next) {
    2854         117 :                         symbol *sym = rn->data.sym;
    2855             : 
    2856         117 :                         sql_exp *e = rel_value_exp(query, &rel, sym, f|sql_farg, ek);
    2857         117 :                         if (!e)
    2858             :                                 return NULL;
    2859         117 :                         list_append(r, e);
    2860             :                 }
    2861          65 :                 return rel_filter(sql, rel, l, r, sname, fname, 0, f);
    2862             :         }
    2863      350094 :         case SQL_COMPARE:
    2864             :         {
    2865      350094 :                 dnode *n = sc->data.lval->h;
    2866      350094 :                 symbol *lo = n->data.sym;
    2867      350094 :                 symbol *ro = n->next->next->data.sym;
    2868      350094 :                 char *compare_op = n->next->data.sval;
    2869      350094 :                 int quantifier = 0;
    2870      350094 :                 int is_semantics = 0;
    2871             : 
    2872      350094 :                 if (n->next->next->next)
    2873         128 :                         quantifier = n->next->next->next->data.i_val + 1;
    2874         128 :                 assert(quantifier == 0 || quantifier == 1 || quantifier == 2 || quantifier == 3 || quantifier == 4);
    2875             : 
    2876      350094 :                 if (quantifier >= 3) {
    2877          10 :                         if (quantifier == 4)
    2878           6 :                                 compare_op = "<>";
    2879             :                         quantifier = 0;
    2880             :                         is_semantics = 1;
    2881             :                 }
    2882      350094 :                 return rel_compare(query, rel, sc, lo, ro, compare_op, f, ek, quantifier, is_semantics);
    2883             :         }
    2884             :         /* Set Member ship */
    2885       32948 :         case SQL_IN:
    2886             :         case SQL_NOT_IN:
    2887       32948 :                 return rel_in_exp(query, rel, sc, f);
    2888        1014 :         case SQL_EXISTS:
    2889             :         case SQL_NOT_EXISTS:
    2890        1014 :                 return rel_exists_exp(query, rel , sc, f);
    2891        3578 :         case SQL_LIKE:
    2892             :         case SQL_NOT_LIKE:
    2893             :         {
    2894        3578 :                 symbol *lo = sc->data.lval->h->data.sym;
    2895        3578 :                 symbol *ro = sc->data.lval->h->next->data.sym;
    2896        3578 :                 int insensitive = sc->data.lval->h->next->next->data.i_val;
    2897        3578 :                 int anti = (sc->token == SQL_NOT_LIKE) != (sc->data.lval->h->next->next->next->data.i_val != 0);
    2898        3578 :                 sql_subtype *st = sql_bind_localtype("str");
    2899        3578 :                 sql_exp *le = rel_value_exp(query, &rel, lo, f|sql_farg, ek), *re, *ee = NULL, *ie = exp_atom_bool(sql->sa, insensitive);
    2900             : 
    2901        3578 :                 if (!le)
    2902             :                         return NULL;
    2903             : 
    2904        3576 :                 if (!exp_subtype(le))
    2905           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: parameter not allowed on "
    2906             :                                         "left hand side of LIKE operator");
    2907             : 
    2908             :                 /* Do we need to escape ? */
    2909        3576 :                 if (dlist_length(ro->data.lval) == 2) {
    2910          82 :                         char *escape = ro->data.lval->h->next->data.sval;
    2911          82 :                         ee = exp_atom(sql->sa, atom_string(sql->sa, st, sa_strdup(sql->sa, escape)));
    2912             :                 } else {
    2913        3494 :                         ee = exp_atom(sql->sa, atom_string(sql->sa, st, sa_strdup(sql->sa, "")));
    2914             :                 }
    2915        3576 :                 ro = ro->data.lval->h->data.sym;
    2916        3576 :                 re = rel_value_exp(query, &rel, ro, f|sql_farg, ek);
    2917        3576 :                 if (!re)
    2918             :                         return NULL;
    2919        3575 :                 if ((re = exp_check_type(sql, st, rel, re, type_equal)) == NULL)
    2920           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "LIKE: wrong type, should be string");
    2921        3575 :                 if ((le = exp_check_type(sql, st, rel, le, type_equal)) == NULL)
    2922           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "LIKE: wrong type, should be string");
    2923        3575 :                 return rel_filter_exp_(sql, rel, le, re, ee, ie, "like", anti, f);
    2924             :         }
    2925        2938 :         case SQL_BETWEEN:
    2926             :         case SQL_NOT_BETWEEN:
    2927             :         {
    2928        2938 :                 symbol *lo = sc->data.lval->h->data.sym;
    2929        2938 :                 int symmetric = sc->data.lval->h->next->data.i_val;
    2930        2938 :                 symbol *ro1 = sc->data.lval->h->next->next->data.sym;
    2931        2938 :                 symbol *ro2 = sc->data.lval->h->next->next->next->data.sym;
    2932        2938 :                 sql_exp *le, *re1, *re2;
    2933        2938 :                 sql_subtype super;
    2934             : 
    2935        2938 :                 assert(sc->data.lval->h->next->type == type_int);
    2936             : 
    2937        2938 :                 if (!(le = rel_value_exp(query, &rel, lo, f|sql_farg, ek)))
    2938             :                         return NULL;
    2939        2936 :                 if (!(re1 = rel_value_exp(query, &rel, ro1, f|sql_farg, ek)))
    2940             :                         return NULL;
    2941        2934 :                 if (!(re2 = rel_value_exp(query, &rel, ro2, f|sql_farg, ek)))
    2942             :                         return NULL;
    2943             : 
    2944        2934 :                 if (exp_between_check_types(&super, exp_subtype(le), exp_subtype(re1), exp_subtype(re2)) < 0)
    2945           1 :                         return sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) on both sides of an expression");
    2946             : 
    2947        2933 :                 if ((le = exp_check_type(sql, &super, rel, le, type_equal)) == NULL ||
    2948        2933 :                     (re1 = exp_check_type(sql, &super, rel, re1, type_equal)) == NULL ||
    2949        2933 :                     (re2 = exp_check_type(sql, &super, rel, re2, type_equal)) == NULL)
    2950           0 :                         return NULL;
    2951             : 
    2952        2933 :                 return rel_compare_exp_(query, rel, le, re1, re2, 3, sc->token == SQL_NOT_BETWEEN ? 1 : 0, 0, f, symmetric, 0);
    2953             :         }
    2954        7948 :         case SQL_IS_NULL:
    2955             :         case SQL_IS_NOT_NULL:
    2956             :         /* is (NOT) NULL */
    2957             :         {
    2958        7948 :                 sql_exp *le = rel_value_exp(query, &rel, sc->data.sym, f|sql_farg, ek);
    2959        7948 :                 sql_subtype *t;
    2960             : 
    2961        7948 :                 if (!le)
    2962             :                         return NULL;
    2963        7944 :                 if (!(t = exp_subtype(le)))
    2964           1 :                         return sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) for IS%s NULL operator", sc->token == SQL_IS_NOT_NULL ? " NOT" : "");
    2965        7943 :                 le = exp_compare(sql->sa, le, exp_atom(sql->sa, atom_general(sql->sa, t, NULL, 0)), cmp_equal);
    2966        7943 :                 if (sc->token == SQL_IS_NOT_NULL)
    2967        2605 :                         set_anti(le);
    2968        7943 :                 set_has_no_nil(le);
    2969        7943 :                 set_semantics(le);
    2970        7943 :                 return rel_select_push_compare_exp_down(sql, rel, le, le->l, le->r, NULL, f);
    2971             :         }
    2972       20851 :         case SQL_NOT: {
    2973       20851 :                 sql_exp *le = rel_value_exp(query, &rel, sc->data.sym, f|sql_farg, ek);
    2974       20851 :                 sql_subtype bt;
    2975             : 
    2976       20851 :                 sql_find_subtype(&bt, "boolean", 0, 0);
    2977       20851 :                 if (!le || !(le = exp_check_type(sql, &bt, rel, le, type_equal)))
    2978           1 :                         return NULL;
    2979       20850 :                 le = exp_compare(sql->sa, le, exp_atom_bool(sql->sa, 0), cmp_equal);
    2980       20850 :                 return rel_select_push_compare_exp_down(sql, rel, le, le->l, le->r, NULL, f);
    2981             :         }
    2982         625 :         case SQL_ATOM: {
    2983             :                 /* TRUE or FALSE */
    2984         625 :                 sql_rel *or = rel;
    2985         625 :                 AtomNode *an = (AtomNode *) sc;
    2986         625 :                 sql_exp *e = exp_atom(sql->sa, an->a);
    2987             : 
    2988         625 :                 if (e) {
    2989         625 :                         sql_subtype bt;
    2990             : 
    2991         625 :                         sql_find_subtype(&bt, "boolean", 0, 0);
    2992         625 :                         e = exp_check_type(sql, &bt, rel, e, type_equal);
    2993             :                 }
    2994         625 :                 if (!e || or != rel)
    2995           2 :                         return NULL;
    2996         623 :                 e = exp_compare(sql->sa, e, exp_atom_bool(sql->sa, 1), cmp_equal);
    2997         623 :                 return rel_select_push_compare_exp_down(sql, rel, e, e->l, e->r, NULL, f);
    2998             :         }
    2999         716 :         case SQL_IDENT:
    3000             :         case SQL_COLUMN: {
    3001         716 :                 sql_rel *or = rel;
    3002         716 :                 sql_exp *e = rel_column_ref(query, &rel, sc, f);
    3003             : 
    3004         716 :                 if (e) {
    3005         713 :                         sql_subtype bt;
    3006             : 
    3007         713 :                         sql_find_subtype(&bt, "boolean", 0, 0);
    3008         713 :                         e = exp_check_type(sql, &bt, rel, e, type_equal);
    3009             :                 }
    3010         713 :                 if (!e || or != rel)
    3011           3 :                         return NULL;
    3012         713 :                 e = exp_compare(sql->sa, e, exp_atom_bool(sql->sa, 1), cmp_equal);
    3013         713 :                 return rel_select_push_compare_exp_down(sql, rel, e, e->l, e->r, NULL, f);
    3014             :         }
    3015          12 :         case SQL_UNION:
    3016             :         case SQL_EXCEPT:
    3017             :         case SQL_INTERSECT: {
    3018          12 :                 sql_rel *sq;
    3019             : 
    3020          12 :                 if (is_psm_call(f) || is_sql_merge(f))
    3021           3 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: set operations not supported inside %s", is_psm_call(f) ? "CALL" : "MERGE", is_psm_call(f) ? "CALL statements" : "MERGE conditions");
    3022          11 :                 if (rel)
    3023          11 :                         query_push_outer(query, rel, f);
    3024          11 :                 sq = rel_setquery(query, sc);
    3025          11 :                 if (rel) {
    3026          11 :                         rel = query_pop_outer(query);
    3027          11 :                         if (is_sql_join(f) && is_groupby(rel->op)) {
    3028           0 :                                 return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in JOIN conditions");
    3029          11 :                         } else if (is_sql_where(f) && is_groupby(rel->op)) {
    3030           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions not allowed in WHERE clause");
    3031          11 :                         } else if ((is_sql_update_set(f) || is_sql_psm(f)) && is_groupby(rel->op)) {
    3032           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions not allowed in SET, WHILE, IF, ELSE, CASE, WHEN, RETURN, ANALYZE clauses");
    3033             :                         }
    3034             :                 }
    3035          11 :                 if (!sq)
    3036             :                         return NULL;
    3037          11 :                 assert(ek.type == type_value); /* I don't expect IN tuple matching calls to land here */
    3038          11 :                 if (is_sql_where(f) && is_groupby(rel->op))
    3039           0 :                         assert(0);
    3040          11 :                 if (ek.card <= card_set && is_project(sq->op) && list_length(sq->exps) > 1)
    3041           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery must return only one column");
    3042          10 :                 if (!rel)
    3043             :                         return sq;
    3044          10 :                 sq = rel_zero_or_one(sql, sq, ek);
    3045          10 :                 if (is_sql_where(f) || is_sql_having(f)) {
    3046          10 :                         sql_exp *le = exp_rel(sql, sq);
    3047          10 :                         sql_subtype bt;
    3048             : 
    3049          10 :                         sql_find_subtype(&bt, "boolean", 0, 0);
    3050          10 :                         le = exp_check_type(sql, &bt, rel, le, type_equal);
    3051          10 :                         if (!le)
    3052             :                                 return NULL;
    3053          10 :                         le = exp_compare(sql->sa, le, exp_atom_bool(sql->sa, 1), cmp_equal);
    3054          10 :                         return rel_select_push_compare_exp_down(sql, rel, le, le->l, le->r, NULL, f);
    3055             :                 } else {
    3056           0 :                         sq = rel_crossproduct(sql->sa, rel, sq, (f==sql_sel || is_single(sq))?op_left:op_join);
    3057           0 :                         set_processed(sq);
    3058             :                 }
    3059           0 :                 return sq;
    3060             :         }
    3061           0 :         case SQL_DEFAULT:
    3062           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "DEFAULT keyword not allowed outside insert and update statements");
    3063         441 :         default: {
    3064         441 :                 sql_exp *le = rel_value_exp(query, &rel, sc, f|sql_farg, ek);
    3065         441 :                 sql_subtype bt;
    3066             : 
    3067         441 :                 if (!le)
    3068             :                         return NULL;
    3069         416 :                 if (le && (!is_compare(le->type) || le->flag > cmp_filter)) {
    3070         285 :                         sql_find_subtype(&bt, "boolean", 0, 0);
    3071         285 :                         if (!(le = exp_check_type(sql, &bt, rel, le, type_equal)))
    3072             :                                 return NULL;
    3073         285 :                         le = exp_compare(sql->sa, le, exp_atom_bool(sql->sa, 1), cmp_equal);
    3074             :                 }
    3075         416 :                 if (le->flag == cmp_filter)
    3076         130 :                         return rel_select_push_filter_exp_down(sql, rel, le, le->l, le->r, f);
    3077             :                 else
    3078         286 :                         return rel_select_push_compare_exp_down(sql, rel, le, le->l, le->r, le->f, f);
    3079             :         }
    3080             :         }
    3081             :         /* never reached, as all switch cases have a `return` */
    3082             : }
    3083             : 
    3084             : static sql_exp * _rel_aggr(sql_query *query, sql_rel **rel, int distinct, char *sname, char *aname, dnode *arguments, symbol *orderby, int f);
    3085             : static sql_exp *rel_aggr(sql_query *query, sql_rel **rel, symbol *se, int f);
    3086             : 
    3087             : sql_exp *
    3088       35724 : rel_unop_(mvc *sql, sql_rel *rel, sql_exp *e, char *sname, char *fname, int card)
    3089             : {
    3090       35724 :         bool found = false;
    3091       35724 :         sql_subtype *t = exp_subtype(e);
    3092       35724 :         sql_ftype type = (card == card_loader)?F_LOADER:((card == card_none)?F_PROC:
    3093             :                    ((card == card_relation)?F_UNION:F_FUNC));
    3094             : 
    3095       35724 :         sql_subfunc *f = bind_func(sql, sname, fname, t, NULL, 1, type, false, &found, false);
    3096       35724 :         if (f) {
    3097       35723 :                 sql_arg *a = f->func->ops->h->data;
    3098       35723 :                 t = &a->type;
    3099             :         }
    3100       35724 :         if (f && t && type_has_tz(t) && f->func->fix_scale == SCALE_FIX) {
    3101             :                 /* set timezone (using msec (.3)) */
    3102           0 :                 sql_subtype *intsec = sql_bind_subtype(sql->sa, "sec_interval", 10 /*hour to second */, 3);
    3103           0 :                 atom *a = atom_int(sql->sa, intsec, sql->timezone);
    3104           0 :                 sql_exp *tz = exp_atom(sql->sa, a);
    3105             : 
    3106           0 :                 e = rel_binop_(sql, rel, e, tz, "sys", "sql_add", card, true);
    3107           0 :                 if (!e)
    3108             :                         return NULL;
    3109             :         }
    3110       35724 :         if (f) {
    3111       35723 :                 if (check_card(card, f)) {
    3112       35723 :                         list *args = list_append(sa_list(sql->sa), e);
    3113       71446 :                         args = check_arguments_and_find_largest_any_type(sql, rel, args, f, card == card_relation && e->card > CARD_ATOM, false);
    3114       35723 :                         if (args)
    3115       35723 :                                 return exp_op(sql->sa, args, f);
    3116             :                 }
    3117           0 :                 found = false; /* reset found */
    3118           0 :                 f = NULL;
    3119             :         }
    3120             :         /* reset error */
    3121           1 :         sql->session->status = 0;
    3122           1 :         sql->errstr[0] = '\0';
    3123           3 :         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: %s unary operator %s%s%s'%s'(%s)",
    3124           2 :                                          found ? "insufficient privileges for" : "no such", sname ? "'":"", sname ? sname : "", sname ? "'.":"", fname, t ? t->type->base.name : "?");
    3125             : }
    3126             : 
    3127             : sql_exp *
    3128      184505 : rel_binop_(mvc *sql, sql_rel *rel, sql_exp *l, sql_exp *r, char *sname, char *fname, int card, bool exact)
    3129             : {
    3130      184505 :         sql_subtype *t1 = exp_subtype(l), *t2 = exp_subtype(r);
    3131      184505 :         sql_ftype type = (card == card_loader)?F_LOADER:((card == card_none)?F_PROC:((card == card_relation)?F_UNION:F_FUNC));
    3132      184505 :         bool found = false;
    3133             : 
    3134      184505 :         if (card == card_loader)
    3135           0 :                 card = card_none;
    3136             : 
    3137      184505 :         if (is_commutative(sname, fname) && l->card < r->card) { /* move constants to the right if possible */
    3138             :                 sql_subtype *tmp = t1;
    3139      184505 :                 t1 = t2;
    3140      184505 :                 t2 = tmp;
    3141             :                 sql_exp *res = l;
    3142      184505 :                 l = r;
    3143      184505 :                 r = res;
    3144             :         }
    3145             :         /* swap complex types (intervals) to left side of +, * */
    3146      184505 :         if (t1 && t2 && is_commutative(sname, fname)) {
    3147          68 :                 if ((EC_INTERVAL(t1->type->eclass) && EC_TEMP(t2->type->eclass)) ||
    3148          68 :            ((!EC_TEMP(t1->type->eclass) && !EC_INTERVAL(t1->type->eclass)) && EC_INTERVAL(t2->type->eclass))) {
    3149             :                         sql_subtype *tmp = t1;
    3150      184505 :                         t1 = t2;
    3151      184505 :                         t2 = tmp;
    3152             :                         sql_exp *res = l;
    3153      184505 :                         l = r;
    3154      184505 :                         r = res;
    3155             :                 }
    3156             :         }
    3157             : 
    3158      184505 :         sql_subfunc *f = bind_func(sql, sname, fname, t1, t2, 2, type, false, &found, exact);
    3159      184505 :         if (f && check_card(card,f)) {
    3160      184505 :                 t1 = exp_subtype(l);
    3161      184505 :                 t2 = exp_subtype(r);
    3162      184505 :                 list *args = list_append(list_append(sa_list(sql->sa), l), r);
    3163             : 
    3164      184505 :                 args = check_arguments_and_find_largest_any_type(sql, rel, args, f, 0, false);
    3165      184505 :                 if (args)
    3166      184505 :                                 return exp_op(sql->sa, args, f);
    3167             :                 return NULL;
    3168             :         }
    3169             : 
    3170           0 :         if (!exp_subtype(l) || !exp_subtype(r))
    3171           0 :                 return sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) on both sides of an expression");
    3172           0 :         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: %s binary operator %s%s%s'%s'(%s,%s)",
    3173           0 :                                          found ? "insufficient privileges for" : "no such", sname ? "'":"", sname ? sname : "", sname ? "'.":"", fname,
    3174           0 :                                          exp_subtype(l)->type->base.name, exp_subtype(r)->type->base.name);
    3175             : }
    3176             : 
    3177             : sql_exp *
    3178         547 : rel_nop_(mvc *sql, sql_rel *rel, sql_exp *a1, sql_exp *a2, sql_exp *a3, sql_exp *a4, char *sname, char *fname, int card)
    3179             : {
    3180         547 :         list *tl = sa_list(sql->sa);
    3181         547 :         sql_subfunc *f = NULL;
    3182         547 :         sql_ftype type = (card == card_none)?F_PROC:((card == card_relation)?F_UNION:F_FUNC);
    3183             : 
    3184             :         /* rel_nop_ should only be called for functions available to everyone, ie defined at sql_types! */
    3185         547 :         (void) rel;
    3186         547 :         append(tl, exp_subtype(a1));
    3187         547 :         append(tl, exp_subtype(a2));
    3188         547 :         append(tl, exp_subtype(a3));
    3189         547 :         if (a4)
    3190         379 :                 append(tl, exp_subtype(a4));
    3191             : 
    3192         547 :         if (!(f = bind_func_(sql, sname, fname, tl, type, false, NULL, true)))
    3193             :                 return NULL;
    3194         547 :         if (!a4)
    3195         168 :                 return exp_op3(sql->sa, a1,a2,a3,f);
    3196         379 :         return exp_op4(sql->sa, a1,a2,a3,a4,f);
    3197             : }
    3198             : 
    3199             : static sql_func *
    3200           3 : inplace_func(mvc *sql)
    3201             : {
    3202           3 :         sql_func *f = SA_NEW(sql->sa, sql_func);
    3203             : 
    3204           3 :     *f = (sql_func) {
    3205             :         .mod = "",
    3206             :         .imp = "",
    3207             :         .type = F_PROC,
    3208             :         .lang = FUNC_LANG_INT,
    3209             :         .query = NULL,
    3210           3 :         .ops = sql->params,
    3211             :         .res = NULL,
    3212             :     };
    3213           3 :     base_init(sql->sa, &f->base, 0, true, NULL);
    3214           3 :     f->base.new = 1;
    3215           3 :     f->base.id = -1;
    3216           3 :     f->base.name = "-1";
    3217           3 :     f->instantiated = TRUE;
    3218           3 :         return f;
    3219             : }
    3220             : 
    3221             : static list *
    3222           3 : reorder_args(mvc *sql, list *exps, list *names, list *params)
    3223             : {
    3224           3 :         list *nexps = sa_list(sql->sa);
    3225           8 :         for(node *n = params->h; n; n = n->next) {
    3226           5 :                 sql_arg *a = n->data;
    3227           5 :                 int found =0;
    3228           7 :                 for(node *m = names->h, *o = exps->h; m && o; m = m->next, o = o->next) {
    3229           7 :                         if (strcmp(m->data, a->name) == 0) {
    3230           5 :                                 append(nexps, o->data);
    3231           5 :                                 found = 1;
    3232           5 :                                 break;
    3233             :                         }
    3234             :                 }
    3235           5 :                 if (!found)
    3236             :                         return NULL;
    3237             :         }
    3238             :         return nexps;
    3239             : }
    3240             : 
    3241             : static sql_exp *
    3242      593209 : rel_nop(sql_query *query, sql_rel **rel, symbol *se, int fs, exp_kind ek)
    3243             : {
    3244      593209 :         mvc *sql = query->sql;
    3245      593209 :         dnode *l = se->data.lval->h;
    3246      593209 :         dnode *ops = l->next->next->data.lval?l->next->next->data.lval->h:NULL;
    3247      593209 :         list *names = NULL;
    3248      593209 :         char *fname = NULL, *sname = NULL;;
    3249             : 
    3250      593209 :         if (l->type != type_int) {
    3251      587300 :                 int nargs = 0;
    3252      587300 :                 fname = qname_schema_object(l->data.lval);
    3253      587299 :                 sname = qname_schema(l->data.lval);
    3254             : 
    3255     1740135 :                 for (dnode *n = ops; n; n = n->next)
    3256     1152836 :                         nargs++;
    3257             : 
    3258             :                 /* first try aggregate */
    3259      587300 :                 if (find_func(sql, sname, fname, nargs, F_AGGR, false, NULL, NULL)) {
    3260       26594 :                         dnode *dn = l->next->next;
    3261       26594 :                         symbol *orderby = dn->next?dn->next->data.sym:NULL;
    3262       26594 :                         return _rel_aggr(query, rel, l->next->data.i_val, sname, fname, dn->data.lval->h, orderby, fs);
    3263             :                 }
    3264             :         }
    3265             : 
    3266      566615 :         int nr_args = 0;
    3267      566615 :         int split = (l->type == type_int && l->data.i_val == -1);
    3268      566615 :         list *exps = sa_list(sql->sa), *tl = sa_list(sql->sa);
    3269      566615 :         exp_kind iek = {type_value, card_column, FALSE};
    3270             : 
    3271      566615 :         if (split)
    3272           3 :                 names = sa_list(sql->sa);
    3273     1700038 :         for (; ops; ops = ops->next, nr_args++) {
    3274     1133510 :                 sql_exp *e = rel_value_exp(query, rel, ops->data.sym, fs|sql_farg, iek);
    3275     1133510 :                 if (!e)
    3276             :                         return NULL;
    3277     1133424 :                 if (split) {
    3278           5 :                         ops = ops->next;
    3279           5 :                         append(names, ops->data.sval);
    3280             :                 }
    3281     1133424 :                 append(exps, e);
    3282     1133424 :                 append(tl, exp_subtype(e));
    3283             :         }
    3284      566528 :         if (l->type == type_int) {
    3285             :                 /* exec nr (ops)*/
    3286        5908 :                 int nr = l->data.i_val;
    3287        5908 :                 cq *q = NULL;
    3288             : 
    3289        5908 :                 if (nr == -1 || (q = qc_find(sql->qc, nr))) {
    3290        5906 :                         list *nexps = new_exp_list(sql->sa);
    3291        5906 :                         sql_func *f = q?q->f:inplace_func(sql);
    3292        5906 :                         list *ops = q?f->ops:sql->params;
    3293             : 
    3294        5906 :                         tl = sa_list(sql->sa);
    3295        5906 :                         if (list_length(ops) != list_length(exps))
    3296           3 :                                 return sql_error(sql, 02, SQLSTATE(42000) "EXEC called with wrong number of arguments: expected %d, got %d", list_length(ops), list_length(exps));
    3297        5903 :                         if (split) {
    3298           3 :                                 exps = reorder_args(sql, exps, names, ops);
    3299           3 :                                 if (!exps)
    3300           0 :                                         return sql_error(sql, 02, SQLSTATE(42000) "EXEC called with wrong arguments");
    3301             :                         }
    3302        5903 :                         if (exps->h && ops) {
    3303       21928 :                                 for (node *n = exps->h, *m = ops->h; n && m; n = n->next, m = m->next) {
    3304       16116 :                                         sql_arg *a = m->data;
    3305       16116 :                                         sql_exp *e = n->data;
    3306       16116 :                                         sql_subtype *ntp = &a->type;
    3307             : 
    3308       16116 :                                         if (ntp && ntp->type)
    3309       16116 :                                                 e = exp_check_type(sql, ntp, NULL, e, type_equal);
    3310             :                                         else
    3311           0 :                                                 a->type = *exp_subtype(e);
    3312       16116 :                                         if (!e)
    3313             :                                                 return NULL;
    3314       16115 :                                         append(nexps, e);
    3315       16115 :                                         append(tl, exp_subtype(e));
    3316             :                                 }
    3317             :                         }
    3318             : 
    3319        5902 :                         if (q)
    3320        5899 :                                 sql->type = q->type;
    3321        5992 :                         return exp_op(sql->sa, list_empty(nexps) ? NULL : nexps, sql_dup_subfunc(sql->sa, f, tl, NULL));
    3322             :                 } else {
    3323           2 :                         return sql_error(sql, 02, SQLSTATE(42000) "EXEC: PREPARED Statement missing '%d'", nr);
    3324             :                 }
    3325             :         }
    3326             : 
    3327      560620 :         if (nr_args == 2 && is_commutative(sname, fname)) {
    3328      124173 :                 sql_subtype *t1 = tl->h->data;
    3329      124173 :                 sql_subtype *t2 = tl->t->data;
    3330             : 
    3331      124173 :                 if (t1 && t2 && ((EC_INTERVAL(t1->type->eclass) && EC_TEMP(t2->type->eclass)) ||
    3332      124164 :            ((!EC_TEMP(t1->type->eclass) && !EC_INTERVAL(t1->type->eclass)) && EC_INTERVAL(t2->type->eclass)))) {
    3333           8 :                         list_revert(exps);
    3334           8 :                         list_revert(tl);
    3335             :                 }
    3336             : 
    3337             :         }
    3338             : 
    3339      560620 :         if (!sname && strcmp(fname, "field") == 0) { /* map into join */
    3340           1 :                 if (list_length(exps) < 2)
    3341           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "Field function called with not enough arguments");
    3342           0 :                 sql_exp *le = exps->h->data;
    3343           0 :                 set_freevar(le, 1);
    3344           0 :                 list_remove_data(exps, NULL, le);
    3345           0 :                 sql_exp *re = exp_values(sql->sa, exps);
    3346           0 :                 exp_label(sql->sa, re, ++sql->label);
    3347           0 :                 sql_rel *r = rel_project(sql->sa, NULL, append(sa_list(sql->sa), re));
    3348           0 :                 sql_exp *id = NULL;
    3349           0 :                 rel_add_identity(sql, r, &id);
    3350           0 :                 re = exp_ref(sql, re);
    3351           0 :                 id = exp_ref(sql, id);
    3352           0 :                 if (r) {
    3353           0 :                         r->nrcols = list_length(exps);
    3354           0 :                         sql_exp *e = exp_compare(sql->sa, le, re, cmp_equal);
    3355           0 :                         r = rel_select(sql->sa, r, e);
    3356           0 :                         r = rel_project(sql->sa, r, append(sa_list(sql->sa), exp_convert(sql, id, exp_subtype(id), sql_bind_localtype("int"))));
    3357           0 :                         re = exp_rel(sql, r);
    3358           0 :                         return re;
    3359             :                 }
    3360             :         }
    3361      560619 :         return _rel_nop(sql, sname, fname, tl, rel ? *rel : NULL, exps, ek);
    3362             : }
    3363             : 
    3364             : typedef struct aggr_input {
    3365             :         sql_query *query;
    3366             :         int groupby;
    3367             :         char *err;
    3368             : } aggr_input;
    3369             : 
    3370             : static sql_exp *
    3371         373 : exp_valid(visitor *v, sql_rel *rel, sql_exp *e, int depth)
    3372             : {
    3373         373 :         aggr_input *ai = v->data;
    3374         373 :         (void)rel; (void)depth;
    3375             : 
    3376         373 :         int vf = is_freevar(e);
    3377         373 :         if (!v->changes && vf && vf < ai->groupby) { /* check need with outer query */
    3378          14 :                 sql_rel *sq = query_fetch_outer(ai->query, vf-1);
    3379             : 
    3380             :                 /* problem freevar have cardinality CARD_ATOM */
    3381          14 :                 if (sq->card <= CARD_AGGR && exp_card(e) != CARD_AGGR && is_alias(e->type)) {
    3382           4 :                         if (!exps_bind_column(sq->exps, e->l, e->r, NULL, 0)) {
    3383           1 :                                 v->changes = 1;
    3384           1 :                                 ai->err = SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query";
    3385             :                         }
    3386             :                 }
    3387         359 :         } else if (!v->changes && vf && vf == ai->groupby) { /* check if input is already aggregated */
    3388         280 :                 sql_rel *sq = query_fetch_outer(ai->query, vf-1);
    3389         280 :                 sql_exp *a = NULL;
    3390             : 
    3391         280 :                 if (sq->card <= CARD_AGGR && is_alias(e->type)) {
    3392         187 :                         if ((a = exps_bind_column(sq->exps, e->l, e->r, NULL, 0)) && is_aggr(a->type)) { /* aggregate */
    3393           9 :                                 v->changes = 1;
    3394           9 :                                 ai->err = SQLSTATE(42000) "SELECT: aggregate function calls cannot be nested";
    3395             :                         }
    3396             :                 }
    3397             :         }
    3398         373 :         return e;
    3399             : }
    3400             : 
    3401             : static char *
    3402         273 : exps_valid(sql_query *query, list *exps, int groupby)
    3403             : {
    3404         273 :         aggr_input ai = { .query = query, .groupby = groupby };
    3405         273 :         visitor v = { .sql = query->sql, .data = &ai };
    3406             : 
    3407         273 :         exps_exp_visitor_topdown(&v, NULL, exps, 0, &exp_valid, true);
    3408         273 :         if (v.changes)
    3409          10 :                 return ai.err;
    3410             :         return NULL;
    3411             : }
    3412             : 
    3413             : static list * rel_order_by(sql_query *query, sql_rel **R, symbol *orderby, int needs_distinct, int f);
    3414             : 
    3415             : static sql_exp *
    3416       46201 : _rel_aggr(sql_query *query, sql_rel **rel, int distinct, char *sname, char *aname, dnode *args, symbol *orderby, int f)
    3417             : {
    3418       46201 :         mvc *sql = query->sql;
    3419       46201 :         exp_kind ek = {type_value, card_column, FALSE};
    3420       46201 :         sql_subfunc *a = NULL;
    3421       46201 :         int no_nil = 0, group = 0, all_aggr = query_has_outer(query), i;
    3422       46201 :         unsigned int all_freevar = 0;
    3423       46201 :         sql_rel *groupby = rel ? *rel : NULL, *sel = NULL, *gr, *og = NULL, *res = groupby;
    3424       46201 :         sql_rel *subquery = NULL;
    3425       46201 :         list *exps = NULL, *ungrouped_cols = NULL;
    3426       46201 :         bool is_grouping = !strcmp(aname, "grouping"), has_args = false, found = false, used_rel = false;
    3427             : 
    3428       46201 :         if (!all_aggr) {
    3429       31126 :                 if (!groupby) {
    3430           4 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3431           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: missing group by", toUpperCopy(uaname, aname));
    3432       31122 :                 } else if (is_sql_groupby(f)) {
    3433           8 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3434           8 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate function '%s' not allowed in GROUP BY clause", toUpperCopy(uaname, aname), aname);
    3435       31114 :                 } else if (is_sql_values(f)) {
    3436           1 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3437           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed on an unique value", toUpperCopy(uaname, aname));
    3438       31113 :                 } else if (is_sql_join(f)) { /* the is_sql_join test must come before is_sql_where, because the join conditions are handled with sql_where */
    3439           3 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3440           3 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in JOIN conditions", toUpperCopy(uaname, aname));
    3441       31110 :                 } else if (is_sql_where(f)) {
    3442           4 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3443           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in WHERE clause", toUpperCopy(uaname, aname));
    3444       31106 :                 } else if (is_sql_update_set(f) || is_sql_psm(f)) {
    3445           1 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3446           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in SET, WHILE, IF, ELSE, CASE, WHEN, RETURN, ANALYZE clauses (use subquery)", toUpperCopy(uaname, aname));
    3447       31105 :                 } else if (is_sql_aggr(f)) {
    3448           4 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3449           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions cannot be nested", toUpperCopy(uaname, aname));
    3450       31101 :                 } else if (is_psm_call(f)) {
    3451           0 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3452           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed inside CALL", toUpperCopy(uaname, aname));
    3453       31101 :                 } else if (is_sql_from(f)) {
    3454           0 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3455           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in functions in FROM", toUpperCopy(uaname, aname));
    3456             :                 }
    3457             :         }
    3458             : 
    3459       46176 :         exps = sa_list(sql->sa);
    3460       46176 :         if (args && args->data.sym) {
    3461       26682 :                 bool arguments_correlated = true, all_const = true;
    3462             : 
    3463       26682 :                 all_freevar = all_aggr?1:0;
    3464       62185 :                 for (i = 0; args && args->data.sym; args = args->next, i++) {
    3465       35563 :                         int base = (!groupby || !is_project(groupby->op) || is_base(groupby->op) || is_processed(groupby));
    3466       27526 :                         sql_rel *gl = base?groupby:groupby->l, *ogl = gl; /* handle case of subqueries without correlation */
    3467       35563 :                         sql_exp *e = rel_value_exp(query, &gl, args->data.sym, (f | sql_aggr)& ~sql_farg, ek);
    3468       35563 :                         bool found_one_freevar = false;
    3469             : 
    3470       35563 :                         if (!e)
    3471          60 :                                 return NULL;
    3472       35509 :                         used_rel |= (rel_has_exp(gl, e, true) == 0);
    3473       35509 :                         has_args = true;
    3474       35509 :                         if (gl && gl != ogl) {
    3475           5 :                                 if (gl->grouped) {
    3476           5 :                                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3477           5 :                                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions cannot be nested", toUpperCopy(uaname, aname));
    3478             :                                 }
    3479           0 :                                 if (!base)
    3480           0 :                                         groupby->l = subquery = gl;
    3481             :                                 else
    3482             :                                         groupby = subquery = gl;
    3483             :                         }
    3484       35504 :                         sql_subtype *t = exp_subtype(e);
    3485       35504 :                         if (!t) { /* we also do not expect parameters here */
    3486           1 :                                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3487           1 :                                 return sql_error(sql, 02, SQLSTATE(42000) "%s: parameters not allowed as arguments to aggregate functions", toUpperCopy(uaname, aname));
    3488             :                         }
    3489       35503 :                         if (!t->type->localtype) {
    3490          52 :                                 if (e->type == e_atom && !e->f) {
    3491          49 :                                         t = sql_bind_localtype("bte");
    3492          49 :                                         e->tpe = *t;
    3493          49 :                                         if (e->l)
    3494          49 :                                                 e->l = atom_set_type(sql->sa, e->l, t);
    3495             :                                 }
    3496             :                         }
    3497             : 
    3498       35503 :                         all_aggr &= (exp_card(e) <= CARD_AGGR && !exp_is_atom(e) && is_aggr(e->type) && !is_func(e->type) && (!groupby || !is_groupby(groupby->op) || !groupby->r || !exps_find_exp(groupby->r, e)));
    3499       35503 :                         exp_only_freevar(query, e, &arguments_correlated, &found_one_freevar, &ungrouped_cols);
    3500       35503 :                         all_freevar &= (arguments_correlated && found_one_freevar) || (is_atom(e->type)?all_freevar:0); /* no uncorrelated variables must be found, plus at least one correlated variable to push this aggregate to an outer query */
    3501       35503 :                         all_const &= is_atom(e->type);
    3502       35503 :                         list_append(exps, e);
    3503             :                 }
    3504       26622 :                 if (all_const)
    3505         229 :                         all_freevar = 0;
    3506             :         }
    3507       46116 :         if (!all_freevar) {
    3508       45823 :                 if (is_sql_groupby(f)) {
    3509           1 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3510           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate function '%s' not allowed in GROUP BY clause", toUpperCopy(uaname, aname), aname);
    3511       45822 :                 } else if (is_sql_from(f)) {
    3512           2 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3513           2 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in functions in FROM", toUpperCopy(uaname, aname));
    3514       45820 :                 } else if (is_sql_aggr(f) && groupby->grouped) {
    3515           2 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3516           2 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions cannot be nested", toUpperCopy(uaname, aname));
    3517       45818 :                 } else if (is_sql_values(f)) {
    3518           1 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3519           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed on an unique value", toUpperCopy(uaname, aname));
    3520       45817 :                 } else if (is_sql_join(f)) { /* the is_sql_join test must come before is_sql_where, because the join conditions are handled with sql_where */
    3521           1 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3522           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in JOIN conditions", toUpperCopy(uaname, aname));
    3523       45816 :                 } else if (is_sql_where(f)) {
    3524           5 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3525           5 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in WHERE clause", toUpperCopy(uaname, aname));
    3526       45811 :                 } else if (!all_aggr && !list_empty(ungrouped_cols)) {
    3527         123 :                         for (node *n = ungrouped_cols->h ; n ; n = n->next) {
    3528          66 :                                 sql_rel *outer;
    3529          66 :                                 sql_exp *e = (sql_exp*) n->data;
    3530             : 
    3531          66 :                                 if ((outer = query_fetch_outer(query, is_freevar(e)-1))) {
    3532          66 :                                         int of = query_fetch_outer_state(query, is_freevar(e)-1);
    3533          66 :                                         if (outer->grouped) {
    3534          24 :                                                 bool err = false, was_processed = false;
    3535             : 
    3536          24 :                                                 if (is_processed(outer)) {
    3537           5 :                                                         was_processed = true;
    3538           5 :                                                         reset_processed(outer);
    3539             :                                                 }
    3540          24 :                                                 if (!is_groupby_col(outer, e))
    3541           3 :                                                         err = true;
    3542          24 :                                                 if (was_processed)
    3543           5 :                                                         set_processed(outer);
    3544          24 :                                                 if (err) {
    3545           3 :                                                         if (exp_name(e) && exp_relname(e) && !has_label(e))
    3546           3 :                                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column \"%s.%s\" from outer query", exp_relname(e), exp_name(e));
    3547           0 :                                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query");
    3548             :                                                 }
    3549          42 :                                         } else if (!used_rel && is_sql_where(of)) {
    3550           2 :                                                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3551           2 :                                                 return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions not allowed in WHERE clause", toUpperCopy(uaname, aname));
    3552          40 :                                         } else if (!is_sql_aggr(of)) {
    3553          36 :                                                 set_outer(outer);
    3554             :                                         }
    3555             :                                 }
    3556             :                         }
    3557             :                 }
    3558             :         }
    3559             : 
    3560       46099 :         if (all_freevar) { /* case 2, ie use outer */
    3561         293 :                 int card;
    3562         293 :                 sql_exp *exp = NULL;
    3563             :                 /* find proper groupby relation */
    3564         593 :                 for (node *n = exps->h; n; n = n->next) {
    3565         300 :                         sql_exp *e = n->data;
    3566             : 
    3567         300 :                         int vf = exp_freevar_offset(sql, e);
    3568         300 :                         if (vf > (int)all_freevar)
    3569          23 :                                 all_freevar = vf;
    3570         300 :                         exp = e;
    3571             :                 }
    3572         293 :                 if (query_has_outer(query) >= all_freevar) {
    3573         293 :                         int sql_state = query_fetch_outer_state(query,all_freevar-1);
    3574         293 :                         res = groupby = query_fetch_outer(query, all_freevar-1);
    3575         293 :                         card = query_outer_used_card(query, all_freevar-1);
    3576             :                         /* given groupby validate all input expressions */
    3577         293 :                         char *err;
    3578         293 :                         if (groupby && !is_groupby(groupby->op)) {
    3579         112 :                                 sql_exp *p = query_outer_last_used(query, all_freevar-1);
    3580         112 :                                 if (p && !is_aggr(p->type) && !is_groupby_col(groupby, p)) {
    3581          20 :                                         if (p->type == e_column)
    3582          20 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", (char*)p->l, (char*)p->r);
    3583           0 :                                         if (exp_name(p) && exp_relname(p) && !has_label(p))
    3584           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", exp_relname(p), exp_name(p));
    3585           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    3586             :                                 }
    3587             :                         }
    3588         273 :                         if ((err = exps_valid(query, exps, all_freevar)) != NULL) {
    3589          10 :                                 strcpy(sql->errstr, err);
    3590          10 :                                 sql->session->status = -ERR_GROUPBY;
    3591          10 :                                 return NULL;
    3592             :                         }
    3593         263 :                         if (exp && !is_groupby_col(res, exp)) {
    3594         192 :                                 if (is_sql_groupby(sql_state))
    3595           2 :                                         return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate function '%s' not allowed in GROUP BY clause", aname);
    3596         190 :                                 if (is_sql_aggr(sql_state) && groupby->grouped) {
    3597           3 :                                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3598           3 :                                         return sql_error(sql, 02, SQLSTATE(42000) "%s: aggregate functions cannot be nested", toUpperCopy(uaname, aname));
    3599             :                                 }
    3600         187 :                                 if (is_sql_values(sql_state))
    3601           0 :                                         return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed on an unique value");
    3602         187 :                                 if (is_sql_update_set(sql_state) || is_sql_psm(f))
    3603           2 :                                         return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in SET, WHILE, IF, ELSE, CASE, WHEN, RETURN, ANALYZE clauses");
    3604         185 :                                 if (is_sql_join(sql_state))
    3605           4 :                                         return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in JOIN conditions");
    3606         181 :                                 if (is_sql_where(sql_state))
    3607           3 :                                         return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in WHERE clause");
    3608         178 :                                 if (is_psm_call(sql_state))
    3609           0 :                                         return sql_error(sql, 05, SQLSTATE(42000) "CALL: aggregate functions not allowed inside CALL");
    3610         178 :                                 if (is_sql_from(sql_state))
    3611           0 :                                         return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in functions in FROM");
    3612         178 :                                 if (card > CARD_AGGR) { /* used an expression before on the non grouped relation */
    3613           0 :                                         sql_exp *lu = query_outer_last_used(query, all_freevar-1);
    3614           0 :                                         if (lu->type == e_column)
    3615           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column \"%s.%s\" from outer query", (char*)lu->l, (char*)lu->r);
    3616           0 :                                         if (exp_name(lu) && exp_relname(lu) && !has_label(lu))
    3617           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column \"%s.%s\" from outer query", exp_relname(lu), exp_name(lu));
    3618           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query");
    3619             :                                 }
    3620         178 :                                 if (is_outer(groupby))
    3621           2 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query");
    3622             :                         }
    3623             :                 } else {
    3624             :                         all_freevar = 0;
    3625             :                 }
    3626       45806 :         } else if (!subquery && groupby && is_outer(groupby) && !is_groupby(groupby->op))
    3627           1 :                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: subquery uses ungrouped column from outer query");
    3628             : 
    3629             :         /* find having select */
    3630       46052 :         if (!subquery && groupby && !is_processed(groupby) && is_sql_having(f)) {
    3631         108 :                 og = groupby;
    3632         108 :                 while(!is_processed(groupby) && !is_base(groupby->op)) {
    3633          63 :                         if (is_select(groupby->op) || !groupby->l)
    3634             :                                 break;
    3635             :                         if (groupby->l)
    3636             :                                 groupby = groupby->l;
    3637             :                 }
    3638          67 :                 if (groupby && is_select(groupby->op) && !is_processed(groupby)) {
    3639          22 :                         group = 1;
    3640          22 :                         sel = groupby;
    3641             :                         /* At the end we switch back to the old projection relation og.
    3642             :                          * During the partitioning and ordering we add the expressions to the intermediate relations. */
    3643             :                 }
    3644          22 :                 if (!sel)
    3645             :                         groupby = og;
    3646          22 :                 if (sel && sel->l)
    3647          22 :                         groupby = sel->l;
    3648             :         }
    3649             : 
    3650             :         /* find groupby */
    3651       46052 :         if (!subquery && groupby && !is_processed(groupby) && !is_base(groupby->op)) {
    3652       30971 :                 og = groupby;
    3653       30971 :                 groupby = rel_find_groupby(groupby);
    3654       30971 :                 if (groupby)
    3655             :                         group = 1;
    3656             :                 else
    3657             :                         groupby = og;
    3658             :         }
    3659             : 
    3660        7925 :         if (!groupby && exps_card(exps) > CARD_ATOM) {
    3661           0 :                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3662           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "%s: missing group by", toUpperCopy(uaname, aname));
    3663             :         }
    3664             : 
    3665       46052 :         if (!subquery && groupby && groupby->op != op_groupby) {             /* implicit groupby */
    3666       21267 :                 if (!all_freevar && query->last_exp && !is_sql_aggr(query->last_state)) {
    3667           9 :                         if (exp_relname(query->last_exp) && exp_name(query->last_exp) && !has_label(query->last_exp))
    3668           9 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s.%s' in query results without an aggregate function", exp_relname(query->last_exp), exp_name(query->last_exp));
    3669           0 :                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    3670             :                 }
    3671       21258 :                 res = groupby = rel_groupby(sql, groupby, NULL);
    3672             :         }
    3673       46043 :         if (subquery) {
    3674           0 :                 assert(!all_freevar);
    3675           0 :                 res = groupby;
    3676           0 :                 if (is_sql_sel(f) && is_left(subquery->op) && !is_groupby(groupby->op)) {
    3677           0 :                         res = groupby = rel_groupby(sql, groupby, NULL);
    3678           0 :                         exps_set_freevar(sql, exps, groupby); /* mark free variables */
    3679           0 :                 } else if (!is_groupby(groupby->op)) {
    3680           0 :                         res = groupby = rel_groupby(sql, groupby, NULL);
    3681             :                 }
    3682           0 :                 assert(!is_base(groupby->op));
    3683             :         }
    3684       46043 :         if ((!exps || exps_card(exps) > CARD_ATOM) && (!res || !groupby))
    3685             :                 return NULL;
    3686             : 
    3687       46043 :         list *obe = NULL;
    3688       46043 :         bool handled_order = true;
    3689       46043 :         if (args)
    3690             :                         return NULL;
    3691       46043 :         if (orderby) { /* handle order by */
    3692        4464 :                 int base = (!groupby || !is_project(groupby->op) || is_base(groupby->op) || is_processed(groupby));
    3693        4464 :                 sql_rel *gl = base?groupby:groupby->l;//, *ogl = gl; /* handle case of subqueries without correlation */
    3694        4464 :                 obe = rel_order_by(query, &gl, orderby, 0, f);
    3695        4464 :                 if (!obe)
    3696           0 :                         return NULL;
    3697        4464 :                 handled_order = false;
    3698             :         }
    3699             : 
    3700       46043 :         if (all_freevar) {
    3701         247 :                 query_update_outer(query, res, all_freevar-1);
    3702       45796 :         } else if (rel) {
    3703       45796 :                 *rel = res;
    3704       45796 :                 if (query->last_rel != res) {
    3705       41834 :                         query->last_rel = res;
    3706       41834 :                         query->last_state |= sql_aggr;
    3707             :                 }
    3708             :         }
    3709             : 
    3710       46043 :         if (!has_args) {        /* count(*) case */
    3711       19490 :                 obe = NULL; /* no errors, although the order by is useless */
    3712       19490 :                 sql_exp *e;
    3713             : 
    3714       19490 :                 if (strcmp(aname, "count") != 0) {
    3715           2 :                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3716           2 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: unable to perform '%s(*)'", toUpperCopy(uaname, aname), aname);
    3717             :                 }
    3718       19488 :                 a = sql_bind_func(sql, "sys", aname, sql_bind_localtype("void"), NULL, F_AGGR, true, true);
    3719       19488 :                 e = exp_aggr(sql->sa, NULL, a, distinct, 0, groupby?groupby->card:CARD_ATOM, 0);
    3720             : 
    3721       19488 :                 if (!groupby)
    3722             :                         return e;
    3723       19488 :                 if (all_freevar)
    3724           0 :                         query_outer_used_exp(query, all_freevar-1, e, sql_aggr);
    3725       19488 :                 e = rel_groupby_add_aggr(sql, groupby, e);
    3726       19488 :                 if (!group && !all_freevar)
    3727             :                         return e;
    3728        3231 :                 if (all_freevar) {
    3729           0 :                         assert(!is_simple_project(res->op));
    3730           0 :                         e->card = CARD_ATOM;
    3731           0 :                         set_freevar(e, all_freevar-1);
    3732           0 :                         return e;
    3733             :                 }
    3734             :                 return e;
    3735             :         }
    3736             : 
    3737             :         /* use cnt as nils shouldn't be counted */
    3738       26553 :         no_nil = 1;
    3739             : 
    3740       26553 :         gr = groupby;
    3741       26553 :         if (gr && gr->op == op_project && gr->l)
    3742             :                 gr = gr->l;
    3743             : 
    3744       26553 :         if (is_grouping) {
    3745         109 :                 sql_subtype *tpe;
    3746         109 :                 list *l = (list*) groupby->r;
    3747             : 
    3748         109 :                 if (list_length(l) <= 7)
    3749         107 :                         tpe = sql_bind_localtype("bte");
    3750           2 :                 else if (list_length(l) <= 15)
    3751           2 :                         tpe = sql_bind_localtype("sht");
    3752           0 :                 else if (list_length(l) <= 31)
    3753           0 :                         tpe = sql_bind_localtype("int");
    3754           0 :                 else if (list_length(l) <= 63)
    3755           0 :                         tpe = sql_bind_localtype("lng");
    3756             : #ifdef HAVE_HGE
    3757           0 :                 else if (list_length(l) <= 127)
    3758           0 :                         tpe = sql_bind_localtype("hge");
    3759             : #endif
    3760             :                 else
    3761           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: GROUPING the number of grouping columns is larger"
    3762             :                                                                 " than the maximum number of representable bits from this server (%d > %d)", list_length(l),
    3763             : #ifdef HAVE_HGE
    3764             :                                                         127
    3765             : #else
    3766             :                                                          63
    3767             : #endif
    3768             :                                                         );
    3769         109 :                 a = sql_bind_func_result(sql, sname, aname, F_AGGR, true, tpe, 1, exp_subtype(exps->h->data));
    3770             :         } else {
    3771       26444 :                 a = sql_bind_func_(sql, sname, aname, exp_types(sql->sa, exps), F_AGGR, false, false);
    3772       26444 :                 if (!a && obe && list_length(obe) == 1) { /* try to find aggregation function with requires order by column */
    3773           0 :                         list *nexps = append(sa_list(sql->sa), obe->h->data);
    3774           0 :                         nexps = list_merge(nexps, exps, (fdup) NULL);
    3775           0 :                         a = sql_bind_func_(sql, sname, aname, exp_types(sql->sa, nexps), F_AGGR, false, false);
    3776           0 :                         if (a && a->func->order_required) {
    3777             :                                 /* reset error */
    3778           0 :                                 handled_order = true;
    3779           0 :                                 sql->session->status = 0;
    3780           0 :                                 sql->errstr[0] = '\0';
    3781           0 :                                 exps = nexps;
    3782           0 :                                 obe = NULL;
    3783             :                         }
    3784             :                 }
    3785             :         }
    3786             : 
    3787       26553 :         if (a) {
    3788       26548 :                 found = true;
    3789       26548 :                 if (!execute_priv(sql, a->func))
    3790           0 :                         a = NULL;
    3791       26548 :                 if (!is_grouping && a && !(exps = check_arguments_and_find_largest_any_type(sql, rel ? *rel : NULL, exps, a, 0, false)))
    3792           0 :                         a = NULL;
    3793             :         }
    3794             : 
    3795       26553 :         if (a) {
    3796       26548 :                 bool hasnil = have_nil(exps) || (strcmp(aname, "count") != 0 && (!groupby || list_empty(groupby->r))); /* for global case, the aggregate may return NULL */
    3797       26548 :                 sql_exp *e = exp_aggr(sql->sa, exps, a, distinct, no_nil, groupby?groupby->card:CARD_ATOM, hasnil);
    3798             : 
    3799       26548 :                 if (!obe && a->func->order_required && !handled_order) {
    3800             :                         /* TODO proper error on missing order by */
    3801             :                         return NULL;
    3802             :                 }
    3803        4464 :                 if (obe && !a->func->order_required && !a->func->opt_order)
    3804             :                         obe = NULL;
    3805        4464 :                 if (obe) /* add order by expressions */
    3806        4464 :                         e->r = append(sa_list(sql->sa), obe);
    3807       26548 :                 if (!groupby)
    3808             :                         return e;
    3809       26548 :                 if (all_freevar)
    3810         247 :                         query_outer_aggregated(query, all_freevar-1, e);
    3811       26548 :                 e = rel_groupby_add_aggr(sql, groupby, e);
    3812       26548 :                 if (!group && !all_freevar)
    3813             :                         return e;
    3814       19933 :                 if (all_freevar) {
    3815         247 :                         rel_bind_vars(sql, groupby->l, exps);
    3816         247 :                         assert(!is_simple_project(res->op));
    3817         247 :                         e->card = CARD_ATOM;
    3818         247 :                         set_freevar(e, all_freevar-1);
    3819         247 :                         return e;
    3820             :                 }
    3821             :                 return e;
    3822             :         }
    3823           5 :         const char *type = "unknown";
    3824           5 :         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    3825             : 
    3826           5 :         if (!list_empty(exps)) {
    3827           5 :                 sql_exp *e = exps->h->data;
    3828           5 :                 type = exp_subtype(e)->type->base.name;
    3829             :         }
    3830          15 :         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "%s: %s aggregate %s%s%s'%s'(%s)", toUpperCopy(uaname, aname), found ? "insufficient privileges for" : "no such",
    3831             :                                          sname ? "'":"", sname ? sname : "", sname ? "'.":"", aname, type);
    3832             : }
    3833             : 
    3834             : static sql_exp *
    3835       19607 : rel_aggr(sql_query *query, sql_rel **rel, symbol *se, int f)
    3836             : {
    3837       19607 :         dlist *l = se->data.lval;
    3838       19607 :         dnode *d = l->h->next->next;
    3839       19607 :         symbol *orderby = d->next?d->next->data.sym:NULL;
    3840       19607 :         int distinct = l->h->next->data.i_val;
    3841       19607 :         char *aname = qname_schema_object(l->h->data.lval);
    3842       19607 :         char *sname = qname_schema(l->h->data.lval);
    3843             : 
    3844       19607 :         return _rel_aggr(query, rel, distinct, sname, aname, d->data.lval?d->data.lval->h:NULL, orderby, f);
    3845             : }
    3846             : 
    3847             : static sql_exp *
    3848      102377 : rel_case(sql_query *query, sql_rel **rel, symbol *opt_cond, dlist *when_search_list, symbol *opt_else, int f)
    3849             : {
    3850      102377 :         mvc *sql = query->sql;
    3851      102377 :         sql_subtype *tpe = NULL;
    3852      102377 :         list *conds = new_exp_list(sql->sa), *results = new_exp_list(sql->sa);
    3853      102377 :         sql_subtype *restype = NULL, *condtype = NULL, ctype, rtype, bt;
    3854      102377 :         sql_exp *res = NULL, *opt_cond_exp = NULL;
    3855      102377 :         exp_kind ek = {type_value, card_column, FALSE};
    3856             : 
    3857      102377 :         if (opt_cond) {
    3858        9630 :                 if (!(opt_cond_exp = rel_value_exp(query, rel, opt_cond, f, ek)))
    3859             :                         return NULL;
    3860        9629 :                 condtype = exp_subtype(opt_cond_exp);
    3861             :         }
    3862             : 
    3863      252813 :         for (dnode *dn = when_search_list->h; dn; dn = dn->next) {
    3864      150446 :                 sql_exp *cond = NULL, *result = NULL;
    3865      150446 :                 dlist *when = dn->data.sym->data.lval;
    3866             : 
    3867      150446 :                 if (opt_cond)
    3868       46416 :                         cond = rel_value_exp(query, rel, when->h->data.sym, f, ek);
    3869             :                 else
    3870      104030 :                         cond = rel_logical_value_exp(query, rel, when->h->data.sym, f, ek);
    3871      150446 :                 if (!cond)
    3872             :                         return NULL;
    3873      150440 :                 append(conds, cond);
    3874      150440 :                 tpe = exp_subtype(cond);
    3875      150440 :                 if (tpe && condtype) {
    3876       57693 :                         result_datatype(&ctype, condtype, tpe);
    3877       57693 :                         condtype = &ctype;
    3878       92747 :                 } else if (tpe) {
    3879       92745 :                         condtype = tpe;
    3880             :                 }
    3881             : 
    3882      150440 :                 if (!(result = rel_value_exp(query, rel, when->h->next->data.sym, f, ek)))
    3883             :                         return NULL;
    3884      150437 :                 append(results, result);
    3885      150437 :                 tpe = exp_subtype(result);
    3886      150437 :                 if (tpe && restype) {
    3887       48067 :                         result_datatype(&rtype, restype, tpe);
    3888       48067 :                         restype = &rtype;
    3889      102370 :                 } else if (tpe) {
    3890      102364 :                         restype = tpe;
    3891             :                 }
    3892             :         }
    3893      102367 :         if (opt_else) {
    3894       99749 :                 if (!(res = rel_value_exp(query, rel, opt_else, f, ek)))
    3895             :                         return NULL;
    3896             : 
    3897       99749 :                 tpe = exp_subtype(res);
    3898       99749 :                 if (tpe && restype) {
    3899       99744 :                         result_datatype(&rtype, restype, tpe);
    3900       99744 :                         restype = &rtype;
    3901           5 :                 } else if (tpe) {
    3902             :                         restype = tpe;
    3903             :                 }
    3904             : 
    3905       99747 :                 if (!restype)
    3906           2 :                         return sql_error(sql, 02, SQLSTATE(42000) "Result type missing");
    3907       99747 :                 if (restype->type->localtype == TYPE_void) /* NULL */
    3908           0 :                         restype = sql_bind_localtype("str");
    3909             : 
    3910       99747 :                 if (!(res = exp_check_type(sql, restype, rel ? *rel : NULL, res, type_equal)))
    3911             :                         return NULL;
    3912             :         } else {
    3913        2618 :                 if (!restype)
    3914           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "Result type missing");
    3915        2618 :                 if (restype->type->localtype == TYPE_void) /* NULL */
    3916           7 :                         restype = sql_bind_localtype("str");
    3917        2618 :                 res = exp_null(sql->sa, restype);
    3918             :         }
    3919             : 
    3920      102365 :         if (!condtype)
    3921           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "Condition type missing");
    3922      102365 :         if (condtype->type->localtype == TYPE_void) /* NULL */
    3923           0 :                 condtype = sql_bind_localtype("str");
    3924      102365 :         if (opt_cond_exp && !(opt_cond_exp = exp_check_type(sql, condtype, rel ? *rel : NULL, opt_cond_exp, type_equal)))
    3925             :                 return NULL;
    3926      102365 :         sql_find_subtype(&bt, "boolean", 0, 0);
    3927      102365 :         list *args = sa_list(sql->sa);
    3928      102365 :         if (opt_cond_exp)
    3929        9623 :                 append(args, opt_cond_exp);
    3930      252795 :         for (node *n = conds->h, *m = results->h; n && m; n = n->next, m = m->next) {
    3931      150431 :                 sql_exp *cond = n->data;
    3932      150431 :                 sql_exp *result = m->data;
    3933             : 
    3934      150431 :                 if (!(result = exp_check_type(sql, restype, rel ? *rel : NULL, result, type_equal)))
    3935             :                         return NULL;
    3936             : 
    3937      150431 :                 if (!(cond = exp_check_type(sql, condtype, rel ? *rel : NULL, cond, type_equal)))
    3938             :                         return NULL;
    3939      150430 :                 if (!opt_cond_exp && !(cond = exp_check_type(sql, &bt, rel ? *rel : NULL, cond, type_equal)))
    3940             :                         return NULL;
    3941      150430 :                 append(args, cond);
    3942      150430 :                 append(args, result);
    3943             :         }
    3944      102364 :         assert(res);
    3945      102364 :         list_append(args, res);
    3946      102364 :         list *types = sa_list(sql->sa);
    3947      102364 :         types = append(append(types, restype), restype);
    3948      195106 :         sql_subfunc *case_func = find_func(sql, NULL, opt_cond_exp?"casewhen":"case", list_length(types), F_FUNC, true, NULL, NULL);
    3949      102364 :         res = exp_op(sql->sa, args, case_func);
    3950      102364 :         ((sql_subfunc*)res->f)->res->h->data = sql_create_subtype(sql->sa, restype->type, restype->digits, restype->scale);
    3951      102364 :         return res;
    3952             : }
    3953             : 
    3954             : static sql_exp *
    3955        4066 : rel_complex_case(sql_query *query, sql_rel **rel, dlist *case_args, int f, str func)
    3956             : {
    3957        4066 :         exp_kind ek = {type_value, card_column, FALSE};
    3958        4066 :         list *args = sa_list(query->sql->sa);
    3959        4066 :         sql_subtype *restype = NULL, rtype;
    3960        4066 :         sql_exp *res;
    3961             : 
    3962             :         /* generate nested func calls */
    3963       12654 :         for(dnode *dn = case_args->h; dn; dn = dn->next) {
    3964        8601 :                 sql_exp *a = rel_value_exp(query, rel, dn->data.sym, f, ek);
    3965        8601 :                 if (!a)
    3966             :                         return NULL;
    3967        8588 :                 append(args, a);
    3968             :                 /* all arguments should have the same type */
    3969        8588 :                 sql_subtype *tpe = exp_subtype(a);
    3970        8588 :                 if (tpe && restype) {
    3971        4524 :                         result_datatype(&rtype, restype, tpe);
    3972        4524 :                         restype = &rtype;
    3973        4064 :                 } else if (tpe) {
    3974        4064 :                         restype = tpe;
    3975             :                 }
    3976             :         }
    3977        4053 :         if (!restype)
    3978           0 :                 return sql_error(query->sql, 02, SQLSTATE(42000) "Result type missing");
    3979        4053 :         if (restype->type->localtype == TYPE_void) /* NULL */
    3980           0 :                 restype = sql_bind_localtype("str");
    3981        4053 :         list *nargs = sa_list(query->sql->sa);
    3982       12626 :         for (node *m = args->h; m; m = m->next) {
    3983        8573 :                 sql_exp *result = m->data;
    3984             : 
    3985        8573 :                 if (!(result = exp_check_type(query->sql, restype, rel ? *rel : NULL, result, type_equal)))
    3986             :                         return NULL;
    3987        8573 :                 append(nargs, result);
    3988             :         }
    3989        4053 :         list *types = append(append(sa_list(query->sql->sa), restype), restype);
    3990        4053 :         sql_subfunc *fnc = find_func(query->sql, NULL, func, list_length(types), F_FUNC, true, NULL, NULL);
    3991        4053 :         res = exp_op(query->sql->sa, nargs, fnc);
    3992        4053 :         ((sql_subfunc*)res->f)->res->h->data = sql_create_subtype(query->sql->sa, restype->type, restype->digits, restype->scale);
    3993        4053 :         return res;
    3994             : }
    3995             : 
    3996             : static sql_exp *
    3997      106443 : rel_case_exp(sql_query *query, sql_rel **rel, symbol *se, int f)
    3998             : {
    3999      106443 :         dlist *l = se->data.lval;
    4000             : 
    4001      106443 :         if (se->token == SQL_COALESCE) {
    4002        3915 :                 return rel_complex_case(query, rel, l, f | sql_farg, "coalesce");
    4003      102528 :         } else if (se->token == SQL_NULLIF) {
    4004         151 :                 return rel_complex_case(query, rel, l, f | sql_farg, "nullif");
    4005      102377 :         } else if (l->h->type == type_list) {
    4006       92747 :                 dlist *when_search_list = l->h->data.lval;
    4007       92747 :                 symbol *opt_else = l->h->next->data.sym;
    4008             : 
    4009       92747 :                 return rel_case(query, rel, NULL, when_search_list, opt_else, f | sql_farg);
    4010             :         } else {
    4011        9630 :                 symbol *scalar_exp = l->h->data.sym;
    4012        9630 :                 dlist *when_value_list = l->h->next->data.lval;
    4013        9630 :                 symbol *opt_else = l->h->next->next->data.sym;
    4014             : 
    4015        9630 :                 return rel_case(query, rel, scalar_exp, when_value_list, opt_else, f | sql_farg);
    4016             :         }
    4017             : }
    4018             : 
    4019             : #define E_ATOM_STRING(e) ((atom*)(e)->l)->data.val.sval
    4020             : 
    4021             : static sql_exp *
    4022      176254 : rel_cast(sql_query *query, sql_rel **rel, symbol *se, int f)
    4023             : {
    4024      176254 :         mvc *sql = query->sql;
    4025      176254 :         dlist *dl = se->data.lval;
    4026      176254 :         symbol *s = dl->h->data.sym;
    4027      176254 :         sql_subtype *tpe = &dl->h->next->data.typeval;
    4028      176254 :         exp_kind ek = {type_value, card_column, FALSE};
    4029      176254 :         sql_exp *e = rel_value_exp(query, rel, s, f|sql_farg, ek);
    4030             : 
    4031      176254 :         if (!e)
    4032             :                 return NULL;
    4033             : 
    4034             :         /* strings may need to be truncated */
    4035      176239 :         if (EC_VARCHAR(tpe->type->eclass) && tpe->digits > 0) {
    4036       85770 :                 sql_subtype *et = exp_subtype(e);
    4037             :                 /* truncate only if the number of digits are smaller or from clob */
    4038       85770 :                 if (et && EC_VARCHAR(et->type->eclass) && (tpe->digits < et->digits || et->digits == 0)) {
    4039        2920 :                         sql_subfunc *c = sql_bind_func(sql, "sys", "truncate", et, sql_bind_localtype("int"), F_FUNC, true, true);
    4040        2920 :                         if (c)
    4041        2920 :                                 e = exp_binop(sql->sa, e, exp_atom_int(sql->sa, tpe->digits), c);
    4042             :                 }
    4043             :         }
    4044             : 
    4045      176239 :         if (tpe->type->eclass == EC_DEC) {
    4046         469 :                 sql_subtype *et = exp_subtype(e);
    4047         469 :                 if (e->type == e_atom && !tpe->digits) {
    4048           9 :                         if (et->type->eclass == EC_NUM || et->type->eclass == EC_DEC) {
    4049           6 :                                 tpe->digits = atom_num_digits(e->l);
    4050           6 :                                 tpe = sql_bind_subtype(sql->sa, "decimal", tpe->digits, et->scale);
    4051           3 :                         } else if (EC_VARCHAR(et->type->eclass)) {
    4052           2 :                                 char *s = E_ATOM_STRING(e);
    4053           2 :                                 unsigned int min_precision = 0, min_scale = 0;
    4054           2 :                                 bool dot_seen = false;
    4055          16 :                                 for (size_t i = 0; i < strlen(s); i++) {
    4056          14 :                                         if (isdigit(s[i])) {
    4057          12 :                                                 min_precision++;
    4058          12 :                                                 if (dot_seen)
    4059          10 :                                                         min_scale++;
    4060           2 :                                         } else if (s[i] == '.') {
    4061           2 :                                                 dot_seen = true;
    4062             :                                         }
    4063             :                                 }
    4064           2 :                                 tpe = sql_bind_subtype(sql->sa, "decimal", min_precision, min_scale);
    4065             :                         } else { /* fallback */
    4066           1 :                                 tpe = sql_bind_subtype(sql->sa, "decimal", 18, 3);
    4067             :                         }
    4068         460 :                 } else if (!tpe->digits && !tpe->scale) {
    4069         105 :                         if (et->type->eclass == EC_NUM)
    4070          36 :                                 tpe = sql_bind_subtype(sql->sa, "decimal", et->digits, 0);
    4071             :                         else /* fallback */
    4072          69 :                                 tpe = sql_bind_subtype(sql->sa, "decimal", 18, 3);
    4073             :                 }
    4074             :         }
    4075             : 
    4076      176239 :         if (e)
    4077      176239 :                 e = exp_check_type(sql, tpe, rel ? *rel : NULL, e, type_cast);
    4078             : 
    4079      176239 :         if (e && e->type == e_convert)
    4080       49885 :                 exp_label(sql->sa, e, ++sql->label);
    4081             : 
    4082             :         return e;
    4083             : }
    4084             : 
    4085             : static sql_exp *
    4086        1590 : rel_next_value_for( mvc *sql, symbol *se )
    4087             : {
    4088        1590 :         char *sname = qname_schema(se->data.lval);
    4089        1590 :         char *seqname = qname_schema_object(se->data.lval);
    4090        1590 :         sql_sequence *seq = NULL;
    4091        1590 :         sql_subtype t;
    4092        1590 :         sql_subfunc *f;
    4093             : 
    4094        1590 :         if (!sname)
    4095         225 :                 sname = "sys";
    4096        1590 :         if (!stack_find_rel_view(sql, seqname)) {
    4097        1389 :                 if (!(seq = find_sequence_on_scope(sql, sname, seqname, "NEXT VALUE FOR")))
    4098             :                         return NULL;
    4099        1387 :                 if (!mvc_schema_privs(sql, seq->s))
    4100           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "NEXT VALUE FOR: access denied for %s to schema '%s'", get_string_global_var(sql, "current_user"), seq->s->base.name);
    4101             :         }
    4102        1588 :         sql_find_subtype(&t, "varchar", 0, 0);
    4103        1588 :         f = sql_bind_func(sql, "sys", "next_value_for", &t, &t, F_FUNC, true, true);
    4104        1588 :         assert(f);
    4105             :         /* sequence found in the stack, ie just created. use given schema? */
    4106        1588 :         return exp_binop(sql->sa, exp_atom_str(sql->sa, seq && seq->s ? seq->s->base.name : sname, &t), exp_atom_str(sql->sa, seqname, &t), f);
    4107             : }
    4108             : 
    4109             : /* some users like to use aliases already in the groupby */
    4110             : static sql_exp *
    4111         165 : rel_selection_ref(sql_query *query, sql_rel **rel, char *name, dlist *selection)
    4112             : {
    4113         165 :         allocator *sa = query->sql->sa;
    4114         165 :         dlist *nl;
    4115         165 :         exp_kind ek = {type_value, card_column, FALSE};
    4116         165 :         sql_exp *res = NULL;
    4117         165 :         symbol *nsym;
    4118             : 
    4119         165 :         if (!selection)
    4120             :                 return NULL;
    4121             : 
    4122         679 :         for (dnode *n = selection->h; n; n = n->next) {
    4123             :                 /* we only look for columns */
    4124         517 :                 tokens to = n->data.sym->token;
    4125         517 :                 if (to == SQL_COLUMN || to == SQL_IDENT) {
    4126         515 :                         dlist *l = n->data.sym->data.lval;
    4127             :                         /* AS name */
    4128         515 :                         if (l->h->next->data.sval && strcmp(l->h->next->data.sval, name) == 0) {
    4129         146 :                                 sql_exp *ve = rel_value_exp(query, rel, l->h->data.sym, sql_sel|sql_groupby, ek);
    4130         146 :                                 if (ve) {
    4131         145 :                                         if (res)
    4132           1 :                                                 return sql_error(query->sql, ERR_AMBIGUOUS, SQLSTATE(42000) "SELECT: identifier '%s' ambiguous", name);
    4133         144 :                                         res = ve;
    4134             : 
    4135         144 :                                         nl = dlist_create(sa);
    4136         144 :                                         exp_setname(query->sql, ve, NULL, name);
    4137             :                                         /* now we should rewrite the selection such that it uses the new group by column */
    4138         144 :                                         dlist_append_string(sa, nl, sa_strdup(sa, name));
    4139         144 :                                         nsym = symbol_create_list(sa, to, nl);
    4140         144 :                                         nl = dlist_create(sa);
    4141         144 :                                         dlist_append_symbol(sa, nl, nsym);
    4142             :                                         /* no alias */
    4143         144 :                                         dlist_append_symbol(sa, nl, NULL);
    4144         144 :                                         n->data.sym = symbol_create_list(sa, to, nl);
    4145             :                                 }
    4146             :                         }
    4147             :                 }
    4148             :         }
    4149             :         return res;
    4150             : }
    4151             : 
    4152             : static char*
    4153         197 : symbol_get_identifier(symbol *sym)
    4154             : {
    4155         197 :         dlist *syml;
    4156             : 
    4157         197 :         if (sym->token != SQL_COLUMN && sym->token != SQL_IDENT)
    4158             :                 return NULL;
    4159         182 :         syml = sym->data.lval;
    4160         182 :         if (dlist_length(syml) > 1)
    4161             :                 return NULL;
    4162             : 
    4163         165 :         return syml->h->data.sval;
    4164             : }
    4165             : 
    4166             : static sql_exp*
    4167       36823 : rel_group_column(sql_query *query, sql_rel **rel, symbol *grp, dlist *selection, list *exps, int f)
    4168             : {
    4169       36823 :         sql_query *lquery = query_create(query->sql);
    4170       36823 :         mvc *sql = query->sql;
    4171       36823 :         exp_kind ek = {type_value, card_value, TRUE};
    4172       36823 :         sql_exp *e = rel_value_exp2(lquery, rel, grp, f, ek);
    4173             : 
    4174       36823 :         if (e && exp_is_atom(e)) {
    4175          35 :                 sql_subtype *tpe = exp_subtype(e);
    4176          35 :                 if (!is_atom(e->type) ||!tpe || tpe->type->eclass != EC_NUM) {
    4177           3 :                         if (!tpe)
    4178           2 :                                 return sql_error(sql, 02, SQLSTATE(42000) "Cannot have a parameter (?) for group by column");
    4179           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: non-integer constant in GROUP BY");
    4180             :                 }
    4181             :         }
    4182             : 
    4183         192 :         if (!e) {
    4184         192 :                 char buf[ERRSIZE], *name;
    4185         192 :                 int status = sql->session->status;
    4186         192 :                 strcpy(buf, sql->errstr);
    4187             :                 /* reset error */
    4188         192 :                 sql->session->status = 0;
    4189         192 :                 sql->errstr[0] = '\0';
    4190             : 
    4191         192 :                 if ((name = symbol_get_identifier(grp))) {
    4192         163 :                         e = rel_selection_ref(query, rel, name, selection);
    4193         163 :                         if (!e) { /* attempt to find in the existing list of group by expressions */
    4194          29 :                                 for (node *n = exps->h ; n && !e ; n = n->next) {
    4195           9 :                                         sql_exp *ge = (sql_exp *) n->data;
    4196           9 :                                         const char *gen = exp_name(ge);
    4197             : 
    4198           9 :                                         if (gen && strcmp(name, gen) == 0)
    4199           5 :                                                 e = exp_ref(sql, ge);
    4200             :                                 }
    4201             :                         }
    4202             :                 }
    4203          49 :                 if (!e && query_has_outer(query)) {
    4204             :                         /* reset error */
    4205          32 :                         sql->session->status = 0;
    4206          32 :                         sql->errstr[0] = '\0';
    4207          32 :                         e = rel_value_exp2(query, rel, grp, f, ek);
    4208             :                 }
    4209          44 :                 if (!e) {
    4210          13 :                         if (sql->errstr[0] == 0) {
    4211          10 :                                 sql->session->status = status;
    4212          10 :                                 strcpy(sql->errstr, buf);
    4213             :                         }
    4214          13 :                         return NULL;
    4215             :                 }
    4216             :         }
    4217       36804 :         if (!exp_subtype(e))
    4218           0 :                 return sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) for group by column");
    4219             :         return e;
    4220             : }
    4221             : 
    4222             : static list*
    4223          44 : list_power_set(allocator *sa, list* input) /* cube */
    4224             : {
    4225          44 :         list *res = sa_list(sa);
    4226             :         /* N stores total number of subsets */
    4227          44 :         int N = (int) pow(2, input->cnt);
    4228             : 
    4229             :         /* generate each subset one by one */
    4230         266 :         for (int i = 0; i < N; i++) {
    4231         222 :                 list *ll = sa_list(sa);
    4232         222 :                 int j = 0; /* check every bit of i */
    4233         844 :                 for (node *n = input->h ; n ; n = n->next) {
    4234             :                         /* if j'th bit of i is set, then append */
    4235         622 :                         if (i & (1 << j))
    4236         311 :                                 list_prepend(ll, n->data);
    4237         622 :                         j++;
    4238             :                 }
    4239         222 :                 list_prepend(res, ll);
    4240             :         }
    4241          44 :         return res;
    4242             : }
    4243             : 
    4244             : static list*
    4245          61 : list_rollup(allocator *sa, list* input)
    4246             : {
    4247          61 :         list *res = sa_list(sa);
    4248             : 
    4249         188 :         for (int counter = input->cnt; counter > 0; counter--) {
    4250         127 :                 list *ll = sa_list(sa);
    4251         127 :                 int j = 0;
    4252         376 :                 for (node *n = input->h; n && j < counter; j++, n = n->next)
    4253         249 :                         list_append(ll, n->data);
    4254         127 :                 list_append(res, ll);
    4255             :         }
    4256          61 :         list_append(res, sa_list(sa)); /* global aggregate case */
    4257          61 :         return res;
    4258             : }
    4259             : 
    4260             : static int
    4261         320 : list_equal(list* list1, list* list2)
    4262             : {
    4263         375 :         for (node *n = list1->h; n ; n = n->next) {
    4264         328 :                 sql_exp *e = (sql_exp*) n->data;
    4265         328 :                 if (!exps_find_exp(list2, e))
    4266             :                         return 1;
    4267             :         }
    4268          78 :         for (node *n = list2->h; n ; n = n->next) {
    4269          47 :                 sql_exp *e = (sql_exp*) n->data;
    4270          47 :                 if (!exps_find_exp(list1, e))
    4271             :                         return 1;
    4272             :         }
    4273             :         return 0;
    4274             : }
    4275             : 
    4276             : static list*
    4277          26 : lists_cartesian_product_and_distinct(allocator *sa, list *l1, list *l2)
    4278             : {
    4279          26 :         list *res = sa_list(sa);
    4280             : 
    4281             :         /* for each list of l2, merge into each list of l1 while removing duplicates */
    4282         116 :         for (node *n = l1->h ; n ; n = n->next) {
    4283          90 :                 list *sub_list = (list*) n->data;
    4284             : 
    4285         279 :                 for (node *m = l2->h ; m ; m = m->next) {
    4286         189 :                         list *other = (list*) m->data;
    4287         189 :                         list_append(res, list_distinct(list_merge(list_dup(sub_list, (fdup) NULL), other, (fdup) NULL), (fcmp) list_equal, (fdup) NULL));
    4288             :                 }
    4289             :         }
    4290          26 :         return res;
    4291             : }
    4292             : 
    4293             : static list*
    4294       13584 : rel_groupings(sql_query *query, sql_rel **rel, symbol *groupby, dlist *selection, int f, bool grouping_sets, list **sets)
    4295             : {
    4296       13584 :         mvc *sql = query->sql;
    4297       13584 :         list *exps = new_exp_list(sql->sa);
    4298             : 
    4299       13584 :         if (mvc_highwater(sql))
    4300           0 :                 return sql_error(sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
    4301             : 
    4302       50314 :         for (dnode *o = groupby->data.lval->h; o; o = o->next) {
    4303       36750 :                 symbol *grouping = o->data.sym;
    4304       36750 :                 list *next_set = NULL;
    4305             : 
    4306       36750 :                 if (grouping->token == SQL_GROUPING_SETS) { /* call recursively, and merge the generated sets */
    4307          39 :                         list *other = rel_groupings(query, rel, grouping, selection, f, true, &next_set);
    4308          39 :                         if (!other)
    4309          20 :                                 return NULL;
    4310          39 :                         exps = list_distinct(list_merge(exps, other, (fdup) NULL), (fcmp) exp_equal, (fdup) NULL);
    4311             :                 } else {
    4312       36711 :                         dlist *dl = grouping->data.lval;
    4313       36711 :                         if (dl) {
    4314       36685 :                                 list *set_cols = new_exp_list(sql->sa); /* columns and combination of columns to be used for the next set */
    4315             : 
    4316       73459 :                                 for (dnode *oo = dl->h; oo; oo = oo->next) {
    4317       36793 :                                         symbol *grp = oo->data.sym;
    4318       36793 :                                         list *next_tuple = new_exp_list(sql->sa); /* next tuple of columns */
    4319             : 
    4320       36793 :                                         if (grp->token == SQL_COLUMN_GROUP) { /* set of columns */
    4321          71 :                                                 assert(is_sql_group_totals(f));
    4322         172 :                                                 for (dnode *ooo = grp->data.lval->h; ooo; ooo = ooo->next) {
    4323         101 :                                                         symbol *elm = ooo->data.sym;
    4324         101 :                                                         sql_exp *e = rel_group_column(query, rel, elm, selection, exps, f);
    4325         101 :                                                         if (!e)
    4326             :                                                                 return NULL;
    4327         101 :                                                         assert(e->type == e_column);
    4328         101 :                                                         list_append(next_tuple, e);
    4329         101 :                                                         list_append(exps, e);
    4330             :                                                 }
    4331             :                                         } else { /* single column or expression */
    4332       36722 :                                                 sql_exp *e = rel_group_column(query, rel, grp, selection, exps, f);
    4333       36722 :                                                 if (!e)
    4334             :                                                         return NULL;
    4335       36703 :                                                 if (e->type != e_column && !exp_is_atom(e)) { /* store group by expressions in the stack */
    4336         142 :                                                         if (is_sql_group_totals(f))
    4337           0 :                                                                 return sql_error(sql, 02, SQLSTATE(42000) "GROUP BY: grouping expressions not possible with ROLLUP, CUBE and GROUPING SETS");
    4338         142 :                                                         if (!exp_has_rel(e) && !frame_push_groupby_expression(sql, grp, e))
    4339             :                                                                 return NULL;
    4340             :                                                 }
    4341       36703 :                                                 list_append(next_tuple, e);
    4342       36703 :                                                 list_append(exps, e);
    4343             :                                         }
    4344       36774 :                                         list_append(set_cols, next_tuple);
    4345             :                                 }
    4346       36666 :                                 if (is_sql_group_totals(f)) {
    4347         178 :                                         if (grouping->token == SQL_ROLLUP)
    4348          61 :                                                 next_set = list_rollup(sql->sa, set_cols);
    4349         117 :                                         else if (grouping->token == SQL_CUBE)
    4350          44 :                                                 next_set = list_power_set(sql->sa, set_cols);
    4351             :                                         else /* the list of sets is not used in the "GROUP BY a, b, ..." case */
    4352          73 :                                                 next_set = list_append(new_exp_list(sql->sa), set_cols);
    4353             :                                 }
    4354          26 :                         } else if (is_sql_group_totals(f) && grouping_sets) /* The GROUP BY () case is the global aggregate which is always added by ROLLUP and CUBE */
    4355          17 :                                 next_set = list_append(new_exp_list(sql->sa), new_exp_list(sql->sa));
    4356             :                 }
    4357       36731 :                 if (is_sql_group_totals(f)) { /* if there are no sets, set the found one, otherwise calculate cartesian product and merge the distinct ones */
    4358         235 :                         if (!next_set)
    4359           1 :                                 return sql_error(sql, 02, SQLSTATE(42000) "GROUP BY: GROUPING SETS is empty");
    4360         234 :                         if (!*sets)
    4361         157 :                                 *sets = next_set;
    4362             :                         else
    4363          77 :                                 *sets = grouping_sets ? list_merge(*sets, next_set, (fdup) NULL) : lists_cartesian_product_and_distinct(sql->sa, *sets, next_set);
    4364             :                 }
    4365             :         }
    4366             :         return exps;
    4367             : }
    4368             : 
    4369             : static list*
    4370        2339 : rel_partition_groupings(sql_query *query, sql_rel **rel, symbol *partitionby, dlist *selection, int f)
    4371             : {
    4372        2339 :         mvc *sql = query->sql;
    4373        2339 :         dnode *o = partitionby->data.lval->h;
    4374        2339 :         list *exps = new_exp_list(sql->sa);
    4375             : 
    4376        7048 :         for (; o; o = o->next) {
    4377        2378 :                 symbol *grp = o->data.sym;
    4378        2378 :                 exp_kind ek = {type_value, card_value, TRUE};
    4379        2378 :                 sql_exp *e = rel_value_exp2(query, rel, grp, f, ek);
    4380             : 
    4381        2378 :                 if (!e) {
    4382           5 :                         int status = sql->session->status;
    4383           5 :                         char buf[ERRSIZE], *name;
    4384             : 
    4385             :                         /* reset error */
    4386           5 :                         sql->session->status = 0;
    4387           5 :                         strcpy(buf, sql->errstr);
    4388           5 :                         sql->errstr[0] = '\0';
    4389             : 
    4390           5 :                         if ((name = symbol_get_identifier(grp))) {
    4391           2 :                                 e = rel_selection_ref(query, rel, name, selection);
    4392           2 :                                 if (!e) { /* attempt to find in the existing list of partition by expressions */
    4393           3 :                                         for (node *n = exps->h ; n ; n = n->next) {
    4394           1 :                                                 sql_exp *ge = (sql_exp *) n->data;
    4395           1 :                                                 const char *gen = exp_name(ge);
    4396             : 
    4397           1 :                                                 if (gen && strcmp(name, gen) == 0) {
    4398           0 :                                                         e = exp_ref(sql, ge);
    4399           0 :                                                         break;
    4400             :                                                 }
    4401             :                                         }
    4402             :                                 }
    4403             :                         }
    4404           5 :                         if (!e) {
    4405           5 :                                 if (sql->errstr[0] == 0) {
    4406           5 :                                         sql->session->status = status;
    4407           5 :                                         strcpy(sql->errstr, buf);
    4408             :                                 }
    4409           5 :                                 return NULL;
    4410             :                         }
    4411             :                 }
    4412             : 
    4413        2373 :                 if (exp_has_rel(e))
    4414           3 :                         return sql_error(sql, 02, SQLSTATE(42000) "PARTITION BY: subqueries not allowed in PARTITION BY clause");
    4415             : 
    4416        2370 :                 if (e->type != e_column) { /* store group by expressions in the stack */
    4417          42 :                         if (!frame_push_groupby_expression(sql, grp, e))
    4418             :                                 return NULL;
    4419             :                 }
    4420             : 
    4421        2370 :                 if (e->card > CARD_AGGR)
    4422        2248 :                         e->card = CARD_AGGR;
    4423        2370 :                 append(exps, e);
    4424             :         }
    4425             :         return exps;
    4426             : }
    4427             : 
    4428             : /* find selection expressions matching the order by column expression */
    4429             : /* complex columns only */
    4430             : static sql_exp *
    4431       68605 : rel_order_by_column_exp(sql_query *query, sql_rel **R, symbol *column_r, int needs_distinct, int f)
    4432             : {
    4433       68605 :         mvc *sql = query->sql;
    4434       68605 :         sql_rel *r = *R, *p = NULL;
    4435       68605 :         sql_exp *e = NULL, *found = NULL;
    4436       68605 :         exp_kind ek = {type_value, card_column, FALSE};
    4437             : 
    4438       68605 :         if (!r)
    4439             :                 return e;
    4440             : 
    4441       68605 :         if (is_simple_project(r->op) && r->l && is_processed(r)) {
    4442       68543 :                 p = r;
    4443       68543 :                 r = r->l;
    4444             :         }
    4445             : 
    4446       68605 :         e = rel_value_exp(query, &r, column_r, f, ek);
    4447             : 
    4448       68605 :         if (r && !p)
    4449          62 :                 *R = r;
    4450       68543 :         else if (r)
    4451       68543 :                 p->l = r;
    4452       68605 :         if (e && p) {
    4453       68516 :                 if (is_project(p->op) && (found = exps_any_match(p->exps, e))) { /* if one of the projections matches, return a reference to it */
    4454       60785 :                         e = exp_ref(sql, found);
    4455             :                 } else {
    4456        7731 :                         if (needs_distinct)
    4457           2 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: with DISTINCT ORDER BY expressions must appear in select list");
    4458        7729 :                         e = rel_project_add_exp(sql, p, e);
    4459        7729 :                         if (r) {
    4460      126861 :                                 for (node *n = p->exps->h ; n ; n = n->next) {
    4461      119134 :                                         sql_exp *ee = n->data;
    4462             : 
    4463      119134 :                                         if (ee->card > r->card) {
    4464           2 :                                                 if (exp_name(ee) && !has_label(ee))
    4465           2 :                                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(ee));
    4466           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    4467             :                                         }
    4468             :                                 }
    4469             :                         }
    4470             :                 }
    4471       68512 :                 return e;
    4472             :         }
    4473          89 :         if (e && r && is_project(r->op)) {
    4474          28 :                 sql_exp *found = exps_find_exp(r->exps, e);
    4475             : 
    4476          28 :                 if (!found) {
    4477           8 :                         if (needs_distinct)
    4478           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: with DISTINCT ORDER BY expressions must appear in select list");
    4479           8 :                         if (!is_simple_project(r->op) && !is_groupby(r->op))
    4480           1 :                                 *R = r = rel_project(sql->sa, r, rel_projections(sql, r, NULL, 1, 0));
    4481           8 :                         append(r->exps, e);
    4482             :                 } else {
    4483             :                         e = found;
    4484             :                 }
    4485          28 :                 e = exp_ref(sql, e);
    4486             :         }
    4487             :         return e;
    4488             : }
    4489             : 
    4490             : static dlist *
    4491      115640 : simple_selection(symbol *sq)
    4492             : {
    4493      115640 :         if (sq->token == SQL_SELECT) {
    4494          13 :                 SelectNode *sn;
    4495          13 :                 sn = (SelectNode *) sq;
    4496             : 
    4497          13 :                 if (!sn->from && !sn->where && !sn->distinct && !sn->window && dlist_length(sn->selection) == 1)
    4498           7 :                         return sn->selection;
    4499             :         }
    4500             :         return NULL;
    4501             : }
    4502             : 
    4503             : static list *
    4504       44952 : rel_order_by(sql_query *query, sql_rel **R, symbol *orderby, int needs_distinct, int f)
    4505             : {
    4506       44952 :         mvc *sql = query->sql;
    4507       44952 :         sql_rel *rel = *R, *or = rel; /* the order by relation */
    4508       44952 :         list *exps = new_exp_list(sql->sa);
    4509             : 
    4510       44952 :         if (!orderby->data.lval) { /* by all */
    4511           6 :                 if (is_sql_orderby(f)) {
    4512           6 :                         assert(is_project(rel->op));
    4513          22 :                         for(node *n = rel->exps->h; n; n = n->next) {
    4514          16 :                                 sql_exp *e = n->data;
    4515          16 :                                 append(exps, exp_ref(sql, e));
    4516             :                         }
    4517             :                         return exps;
    4518             :                 }
    4519             :                 return NULL;
    4520             :         }
    4521             : 
    4522       44946 :         dnode *o = orderby->data.lval->h;
    4523       44946 :         dlist *selection = NULL;
    4524             : 
    4525       44946 :         if (is_sql_orderby(f)) {
    4526       37429 :                 assert(is_project(rel->op));
    4527       37429 :                 rel = rel->l;
    4528             :         }
    4529             : 
    4530      160538 :         for (; o; o = o->next) {
    4531      115640 :                 symbol *order = o->data.sym;
    4532             : 
    4533      115640 :                 if (order->token == SQL_COLUMN || order->token == SQL_IDENT) {
    4534      115640 :                         symbol *col = order->data.lval->h->data.sym;
    4535      115640 :                         int direction = order->data.lval->h->next->data.i_val;
    4536      115640 :                         sql_exp *e = NULL;
    4537             : 
    4538      115640 :                         assert(order->data.lval->h->next->type == type_int);
    4539      115640 :                         if ((selection = simple_selection(col)) != NULL) {
    4540           7 :                                 dnode *o = selection->h;
    4541           7 :                                 order = o->data.sym;
    4542           7 :                                 if (order->data.lval->h->type == type_symbol)
    4543           6 :                                         col = order->data.lval->h->data.sym;
    4544             :                                 /* remove optional name from selection */
    4545           7 :                                 order->data.lval->h->next = NULL;
    4546             :                         }
    4547             : 
    4548      115640 :                         if (col->token == SQL_COLUMN || col->token == SQL_IDENT || col->token == SQL_ATOM) {
    4549      115418 :                                 exp_kind ek = {type_value, card_column, FALSE};
    4550             : 
    4551      115418 :                                 e = rel_value_exp2(query, &rel, col, f, ek);
    4552             : 
    4553      115418 :                                 if (e && e->card <= CARD_ATOM) {
    4554        9376 :                                         sql_subtype *tpe = exp_subtype(e);
    4555             :                                         /* integer atom on the stack */
    4556        9376 :                                         if (!is_sql_window(f) && e->type == e_atom &&
    4557        9268 :                                             tpe->type->eclass == EC_NUM) {
    4558        9259 :                                                 atom *a = e->l;
    4559        9259 :                                                 int nr = (int)atom_get_int(a);
    4560             : 
    4561        9259 :                                                 e = exps_get_exp(rel->exps, nr);
    4562        9259 :                                                 if (!e)
    4563           3 :                                                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: the order by column number (%d) is not in the number of projections range (%d)", nr, list_length(rel->exps));
    4564        9256 :                                                 e = exp_ref(sql, e);
    4565             :                                         }
    4566       37653 :                                 } else if (e && exp_card(e) > rel->card) {
    4567           0 :                                         if (exp_name(e) && !has_label(e))
    4568           0 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(e));
    4569           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    4570             :                                 }
    4571      115415 :                                 if (e && !exp_name(e))
    4572          23 :                                         exp_label(sql->sa, e, ++sql->label);
    4573       47026 :                                 if (e && rel && is_project(rel->op)) {
    4574       40016 :                                         sql_exp *found = exps_find_exp(rel->exps, e);
    4575             : 
    4576       40016 :                                         if (!found) {
    4577          33 :                                                 if (needs_distinct)
    4578           0 :                                                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: with DISTINCT ORDER BY expressions must appear in select list");
    4579          33 :                                                 if (!is_freevar(e))
    4580          16 :                                                         append(rel->exps, e);
    4581             :                                         } else {
    4582             :                                                 e = found;
    4583             :                                         }
    4584       40016 :                                         if (!is_freevar(e))
    4585       39999 :                                                 e = exp_ref(sql, e);
    4586             :                                 }
    4587             :                         }
    4588             : 
    4589      115637 :                         if (rel && !e && sql->session->status != -ERR_AMBIGUOUS) {
    4590             :                                 /* reset error */
    4591       68605 :                                 sql->session->status = 0;
    4592       68605 :                                 sql->errstr[0] = '\0';
    4593             : 
    4594       68605 :                                 e = rel_order_by_column_exp(query, &rel, col, needs_distinct, sql_sel | sql_orderby | (f & sql_group_totals) | (f & sql_window));
    4595             :                         }
    4596       68611 :                         if (!e)
    4597             :                                 return NULL;
    4598      115593 :                         if (!exp_subtype(e))
    4599           1 :                                 return sql_error(sql, 01, SQLSTATE(42000) "Cannot have a parameter (?) for order by column");
    4600      115592 :                         set_direction(e, direction);
    4601      115592 :                         list_append(exps, e);
    4602             :                 } else {
    4603           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: order not of type SQL_COLUMN");
    4604             :                 }
    4605             :         }
    4606       44898 :         if (is_sql_orderby(f) && or != rel)
    4607       37388 :                 or->l = rel;
    4608       44898 :         if (is_sql_window(f))
    4609        3046 :                 *R = rel;
    4610             :         return exps;
    4611             : }
    4612             : 
    4613             : static int
    4614         814 : generate_window_bound(tokens sql_token, bool first_half)
    4615             : {
    4616         814 :         switch (sql_token) {
    4617         408 :                 case SQL_PRECEDING:
    4618         408 :                         return first_half ? BOUND_FIRST_HALF_PRECEDING : BOUND_SECOND_HALF_PRECEDING;
    4619         284 :                 case SQL_FOLLOWING:
    4620         284 :                         return first_half ? BOUND_FIRST_HALF_FOLLOWING : BOUND_SECOND_HALF_FOLLOWING;
    4621         122 :                 case SQL_CURRENT_ROW:
    4622         122 :                         return first_half ? CURRENT_ROW_PRECEDING : CURRENT_ROW_FOLLOWING;
    4623             :                 default:
    4624           0 :                         assert(0);
    4625             :         }
    4626             :         return 0;
    4627             : }
    4628             : 
    4629             : /* window functions */
    4630             : static sql_exp*
    4631         407 : generate_window_bound_call(mvc *sql, sql_exp **estart, sql_exp **eend, sql_exp *pe, sql_exp *e,
    4632             :                                                    sql_exp *start, sql_exp *fend, int frame_type, int excl, tokens t1, tokens t2)
    4633             : {
    4634         407 :         list *rargs1 = sa_list(sql->sa), *rargs2 = sa_list(sql->sa), *targs1 = sa_list(sql->sa), *targs2 = sa_list(sql->sa);
    4635         407 :         sql_subfunc *dc1, *dc2;
    4636         407 :         sql_subtype *it = sql_bind_localtype("int");
    4637             : 
    4638         407 :         if (pe) {
    4639         186 :                 append(targs1, exp_subtype(pe));
    4640         186 :                 append(targs2, exp_subtype(pe));
    4641         186 :                 append(rargs1, exp_copy(sql, pe));
    4642         186 :                 append(rargs2, exp_copy(sql, pe));
    4643             :         }
    4644         407 :         append(rargs1, exp_copy(sql, e));
    4645         407 :         append(rargs2, exp_copy(sql, e));
    4646         407 :         append(targs1, exp_subtype(e));
    4647         407 :         append(targs2, exp_subtype(e));
    4648         407 :         append(targs1, it);
    4649         407 :         append(targs2, it);
    4650         407 :         append(targs1, it);
    4651         407 :         append(targs2, it);
    4652         407 :         append(targs1, it);
    4653         407 :         append(targs2, it);
    4654         407 :         append(targs1, exp_subtype(start));
    4655         407 :         append(targs2, exp_subtype(fend));
    4656             : 
    4657         407 :         dc1 = sql_bind_func_(sql, "sys", "window_bound", targs1, F_ANALYTIC, true, false);
    4658         407 :         dc2 = sql_bind_func_(sql, "sys", "window_bound", targs2, F_ANALYTIC, true, false);
    4659         407 :         if (!dc1 || !dc2)
    4660           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: function 'window_bound' not found");
    4661         407 :         append(rargs1, exp_atom_int(sql->sa, frame_type));
    4662         407 :         append(rargs2, exp_atom_int(sql->sa, frame_type));
    4663         407 :         append(rargs1, exp_atom_int(sql->sa, generate_window_bound(t1, true)));
    4664         407 :         append(rargs2, exp_atom_int(sql->sa, generate_window_bound(t2, false)));
    4665         407 :         append(rargs1, exp_atom_int(sql->sa, excl));
    4666         407 :         append(rargs2, exp_atom_int(sql->sa, excl));
    4667         407 :         append(rargs1, start);
    4668         407 :         append(rargs2, fend);
    4669             : 
    4670         407 :         *estart = exp_op(sql->sa, rargs1, dc1);
    4671         407 :         *eend = exp_op(sql->sa, rargs2, dc2);
    4672         407 :         return e; /* return something to say there were no errors */
    4673             : }
    4674             : 
    4675             : #define EC_NUMERIC(e) (e==EC_NUM||EC_INTERVAL(e)||e==EC_DEC||e==EC_FLT)
    4676             : 
    4677             : static sql_exp*
    4678         601 : calculate_window_bound(sql_query *query, sql_rel *p, tokens token, symbol *bound, sql_exp *ie, int frame_type, int f)
    4679             : {
    4680         601 :         mvc *sql = query->sql;
    4681         601 :         sql_subtype *bt, *bound_tp = sql_bind_localtype("lng"), *iet = exp_subtype(ie);
    4682         601 :         sql_exp *res = NULL;
    4683             : 
    4684         792 :         if ((bound->token == SQL_PRECEDING || bound->token == SQL_FOLLOWING || bound->token == SQL_CURRENT_ROW) && bound->type == type_int) {
    4685         191 :                 atom *a = NULL;
    4686         191 :                 bt = (frame_type == FRAME_ROWS || frame_type == FRAME_GROUPS) ? bound_tp : iet;
    4687             : 
    4688         191 :                 if ((bound->data.i_val == UNBOUNDED_PRECEDING_BOUND || bound->data.i_val == UNBOUNDED_FOLLOWING_BOUND)) {
    4689          69 :                         a = atom_max_value(sql->sa, EC_NUMERIC(bt->type->eclass) ? bt : bound_tp);
    4690         122 :                 } else if (bound->data.i_val == CURRENT_ROW_BOUND) {
    4691         124 :                         a = atom_zero_value(sql->sa, EC_NUMERIC(bt->type->eclass) ? bt : bound_tp);
    4692             :                 } else {
    4693           0 :                         assert(0);
    4694             :                 }
    4695         191 :                 res = exp_atom(sql->sa, a);
    4696             :         } else { /* arbitrary expression case */
    4697         410 :                 exp_kind ek = {type_value, card_column, FALSE};
    4698         410 :                 const char *bound_desc = (token == SQL_PRECEDING) ? "PRECEDING" : "FOLLOWING";
    4699             : 
    4700         410 :                 assert(token == SQL_PRECEDING || token == SQL_FOLLOWING);
    4701         410 :                 if (!(res = rel_value_exp2(query, &p, bound, f, ek)))
    4702           3 :                         return NULL;
    4703         410 :                 if (!(bt = exp_subtype(res))) { /* frame bound is a parameter */
    4704           3 :                         sql_subtype *t = (frame_type == FRAME_ROWS || frame_type == FRAME_GROUPS) ? bound_tp : iet;
    4705           3 :                         if (rel_set_type_param(sql, t, p, res, 0) < 0) /* workaround */
    4706             :                                 return NULL;
    4707           3 :                         bt = exp_subtype(res);
    4708             :                 }
    4709         410 :                 if (exp_is_null_no_value_opt(res))
    4710           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s offset must not be NULL", bound_desc);
    4711         409 :                 if ((frame_type == FRAME_ROWS || frame_type == FRAME_GROUPS) && bt->type->eclass != EC_NUM && !(res = exp_check_type(sql, bound_tp, p, res, type_equal)))
    4712             :                         return NULL;
    4713         409 :                 if (frame_type == FRAME_RANGE) {
    4714          70 :                         sql_class iet_class = iet->type->eclass;
    4715             : 
    4716          70 :                         if (!EC_NUMERIC(iet_class) && !EC_TEMP(iet_class))
    4717           1 :                                 return sql_error(sql, 02, SQLSTATE(42000) "Ranges with arbitrary expressions are available to numeric, interval and temporal types only");
    4718          12 :                         if (EC_NUMERIC(iet_class) && !(res = exp_check_type(sql, iet, p, res, type_equal)))
    4719             :                                 return NULL;
    4720          69 :                         if ((iet_class == EC_TIME || iet_class == EC_TIME_TZ) && bt->type->eclass != EC_SEC) {
    4721           1 :                                 (void) sql_error(sql, 02, SQLSTATE(42000) "For %s input the %s boundary must be an interval type up to the day", subtype2string2(sql->ta, iet), bound_desc);
    4722           1 :                                 sa_reset(sql->ta);
    4723           1 :                                 return NULL;
    4724             :                         }
    4725          68 :                         if (EC_TEMP(iet->type->eclass) && !EC_INTERVAL(bt->type->eclass)) {
    4726           0 :                                 (void) sql_error(sql, 02, SQLSTATE(42000) "For %s input the %s boundary must be an interval type", subtype2string2(sql->ta, iet), bound_desc);
    4727           0 :                                 sa_reset(sql->ta);
    4728           0 :                                 return NULL;
    4729             :                         }
    4730             :                 }
    4731             :         }
    4732             :         return res;
    4733             : }
    4734             : 
    4735             : static dlist*
    4736          16 : get_window_clauses(mvc *sql, char* ident, symbol **partition_by_clause, symbol **order_by_clause, symbol **frame_clause)
    4737             : {
    4738          16 :         dlist *window_specification = NULL;
    4739          16 :         char *window_ident;
    4740          16 :         int pos;
    4741             : 
    4742          16 :         if (mvc_highwater(sql))
    4743           0 :                 return sql_error(sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
    4744             : 
    4745          16 :         if ((window_specification = frame_get_window_def(sql, ident, &pos)) == NULL)
    4746           1 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: window '%s' not found", ident);
    4747             : 
    4748             :         /* avoid infinite lookups */
    4749          15 :         if (frame_check_var_visited(sql, pos))
    4750           1 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: cyclic references to window '%s' found", ident);
    4751          14 :         frame_set_var_visited(sql, pos);
    4752             : 
    4753          14 :         if (window_specification->h->next->data.sym) {
    4754           2 :                 if (*partition_by_clause)
    4755           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: redefinition of PARTITION BY clause from window '%s'", ident);
    4756           2 :                 *partition_by_clause = window_specification->h->next->data.sym;
    4757             :         }
    4758          14 :         if (window_specification->h->next->next->data.sym) {
    4759           4 :                 if (*order_by_clause)
    4760           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: redefinition of ORDER BY clause from window '%s'", ident);
    4761           3 :                 *order_by_clause = window_specification->h->next->next->data.sym;
    4762             :         }
    4763          13 :         if (window_specification->h->next->next->next->data.sym) {
    4764           0 :                 if (*frame_clause)
    4765           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: redefinition of frame clause from window '%s'", ident);
    4766           0 :                 *frame_clause = window_specification->h->next->next->next->data.sym;
    4767             :         }
    4768             : 
    4769          13 :         window_ident = window_specification->h->data.sval;
    4770          13 :         if (window_ident && !get_window_clauses(sql, window_ident, partition_by_clause, order_by_clause, frame_clause))
    4771             :                 return NULL; /* the error was already set */
    4772             : 
    4773             :         return window_specification; /* return something to say there were no errors */
    4774             : }
    4775             : 
    4776             : /*
    4777             :  * select x, y, rank_op() over (partition by x order by y) as, ...
    4778             :                 aggr_op(z) over (partition by y order by x) as, ...
    4779             :  * from table [x,y,z,w,v]
    4780             :  *
    4781             :  * project and order by over x,y / y,x
    4782             :  * a = project( table ) [ x, y, z, w, v ], [ x, y]
    4783             :  * b = project( table ) [ x, y, z, w, v ], [ y, x]
    4784             :  *
    4785             :  * project with order dependent operators, ie combined prev/current value
    4786             :  * aa = project (a) [ x, y, r = rank_op(diff(x) (marks a new partition), rediff(diff(x), y) (marks diff value with in partition)), z, w, v ]
    4787             :  * project(aa) [ aa.x, aa.y, aa.r ] -- only keep current output list
    4788             :  * bb = project (b) [ x, y, a = aggr_op(z, diff(y), rediff(diff(y), x)), z, w, v ]
    4789             :  * project(bb) [ bb.x, bb.y, bb.a ]  -- only keep current output list
    4790             :  */
    4791             : static sql_exp *
    4792       12839 : rel_rankop(sql_query *query, sql_rel **rel, symbol *se, int f)
    4793             : {
    4794       12839 :         mvc *sql = query->sql;
    4795       12839 :         node *n;
    4796       12839 :         dlist *l = se->data.lval, *window_specification = NULL;
    4797       12839 :         symbol *window_function = l->h->data.sym, *partition_by_clause = NULL, *order_by_clause = NULL, *frame_clause = NULL;
    4798       12839 :         char *aname = NULL, *sname = NULL, *window_ident = NULL;
    4799       12839 :         sql_subfunc *wf = NULL;
    4800       12839 :         sql_exp *in = NULL, *pe = NULL, *oe = NULL, *call = NULL, *start = NULL, *eend = NULL, *fstart = NULL, *fend = NULL, *ie = NULL;
    4801       12839 :         sql_rel *p;
    4802       12839 :         list *gbe = NULL, *obe = NULL, *args = NULL, *types = NULL, *fargs = NULL;
    4803       12839 :         dnode *dn = window_function->data.lval->h, *dargs = NULL;
    4804       12839 :         int distinct = 0, frame_type, pos, nf = f, nfargs = 0;
    4805       12839 :         bool is_nth_value, supports_frames = false, found = false;
    4806             : 
    4807       12839 :         frame_clear_visited_flag(sql); /* clear visited flags before iterating */
    4808             : 
    4809       12839 :         if (l->h->next->type == type_list) {
    4810       12823 :                 window_specification = l->h->next->data.lval;
    4811          16 :         } else if (l->h->next->type == type_string) {
    4812          16 :                 const char* window_alias = l->h->next->data.sval;
    4813          16 :                 if ((window_specification = frame_get_window_def(sql, window_alias, &pos)) == NULL)
    4814           1 :                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: window '%s' not found", window_alias);
    4815          15 :                 frame_set_var_visited(sql, pos);
    4816             :         } else {
    4817           0 :                 assert(0);
    4818             :         }
    4819             : 
    4820       12838 :         window_ident = window_specification->h->data.sval;
    4821       12838 :         partition_by_clause = window_specification->h->next->data.sym;
    4822       12838 :         order_by_clause = window_specification->h->next->next->data.sym;
    4823       12838 :         frame_clause = window_specification->h->next->next->next->data.sym;
    4824             : 
    4825       12838 :         if (window_ident && !get_window_clauses(sql, window_ident, &partition_by_clause, &order_by_clause, &frame_clause))
    4826             :                 return NULL;
    4827             : 
    4828       12835 :         frame_type = frame_clause ? frame_clause->data.lval->h->next->next->data.i_val : FRAME_RANGE;
    4829       12835 :         aname = qname_schema_object(dn->data.lval);
    4830       12835 :         sname = qname_schema(dn->data.lval);
    4831             : 
    4832       12835 :         is_nth_value = !strcmp(aname, "nth_value");
    4833       12835 :         bool is_value = is_nth_value || !strcmp(aname, "first_value") || !strcmp(aname, "last_value");
    4834       12835 :         bool rank = false;
    4835       12835 :         if (strcmp(aname, "row_number") == 0 || strcmp(aname, "rank") == 0 || strcmp(aname, "dense_rank") == 0 ||
    4836        2935 :             strcmp(aname, "percent_rank") == 0 || strcmp(aname, "cume_dist") == 0 || strcmp(aname, "ntile") == 0 ||
    4837        2831 :             strcmp(aname, "lag") == 0 || strcmp(aname, "lead") == 0 || strcmp(aname, "fetch") == 0)
    4838       10076 :                 rank = true;
    4839       12835 :         supports_frames = (!rank || is_value);
    4840             : 
    4841       12835 :         if (is_sql_update_set(f) || is_sql_psm(f) || is_sql_values(f) || is_sql_join(f) || is_sql_where(f) || is_sql_groupby(f) || is_sql_having(f) || is_psm_call(f) || is_sql_from(f)) {
    4842          13 :                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4843          13 :                 const char *clause = is_sql_update_set(f)||is_sql_psm(f)?"in SET, WHILE, IF, ELSE, CASE, WHEN, RETURN, ANALYZE clauses (use subquery)":is_sql_values(f)?"on an unique value":
    4844           8 :                                                          is_sql_join(f)?"in JOIN conditions":is_sql_where(f)?"in WHERE clause":is_sql_groupby(f)?"in GROUP BY clause":
    4845           4 :                                                          is_psm_call(f)?"in CALL":is_sql_from(f)?"in functions in FROM":"in HAVING clause";
    4846          13 :                 return sql_error(sql, 02, SQLSTATE(42000) "%s: window function '%s' not allowed %s", toUpperCopy(uaname, aname), aname, clause);
    4847       12822 :         } else if (is_sql_aggr(f)) {
    4848           2 :                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4849           2 :                 return sql_error(sql, 02, SQLSTATE(42000) "%s: window functions not allowed inside aggregation functions", toUpperCopy(uaname, aname));
    4850       12820 :         } else if (is_sql_window(f)) {
    4851           6 :                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4852           6 :                 return sql_error(sql, 02, SQLSTATE(42000) "%s: window functions cannot be nested", toUpperCopy(uaname, aname));
    4853             :         }
    4854       12814 :         if (window_function->token == SQL_AGGR) {
    4855         106 :                 dn->next->next->type = type_list;
    4856         106 :                 dn->next->next->data.lval = dlist_append_symbol(sql->sa, dlist_create( sql->sa ), dn->next->next->data.sym); /* make a list */
    4857             :         }
    4858       12814 :         if (window_function->token == SQL_NOP)
    4859       12708 :                 window_function->token = SQL_AGGR;
    4860       12814 :         if (window_function->token != SQL_RANK && window_function->token != SQL_AGGR) {
    4861           0 :                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4862           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: window function '%s' unknown", toUpperCopy(uaname, aname));
    4863             :         }
    4864             : 
    4865             :         /* window operations are only allowed in the projection */
    4866       12814 :         if (!is_sql_sel(f) && !is_sql_qualify(f))
    4867           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "OVER: only possible within the selection");
    4868             : 
    4869       12814 :         p = *rel;
    4870             :         /* Partition By */
    4871       12814 :         if (partition_by_clause) {
    4872        2339 :                 gbe = rel_partition_groupings(query, &p, partition_by_clause, NULL /* cannot use (selection) column references, as this result is a selection column */, nf | sql_window);
    4873        2339 :                 if (!gbe)
    4874             :                         return NULL;
    4875        4698 :                 for (n = gbe->h ; n ; n = n->next) {
    4876        2367 :                         sql_exp *en = n->data;
    4877             : 
    4878        2367 :                         set_ascending(en);
    4879        2367 :                         set_nulls_first(en);
    4880             :                 }
    4881             :         }
    4882             : 
    4883             :         /* Order By */
    4884       12806 :         if (order_by_clause) {
    4885        3053 :                 obe = rel_order_by(query, &p, order_by_clause, 0, sql_window);
    4886        3053 :                 if (!obe)
    4887             :                         return NULL;
    4888             :         }
    4889             : 
    4890       12799 :         fargs = sa_list(sql->sa);
    4891       12799 :         if (rank) { /* rank function call */
    4892       10062 :                 dlist *dl = dn->next->next->data.lval;
    4893       10062 :                 bool is_lag = !strcmp(aname, "lag"), is_lead = !strcmp(aname, "lead"),
    4894       10062 :                          extra_input = !strcmp(aname, "ntile") || !strcmp(aname, "rank") || !strcmp(aname, "dense_rank") || !strcmp(aname, "row_number") || !strcmp(aname, "percent_rank") || !strcmp(aname, "cume_dist");
    4895             : 
    4896       10062 :                 distinct = dn->next->data.i_val;
    4897       10062 :                 if (extra_input) { /* pass an input column for analytic functions that don't require it */
    4898        9991 :                         sql_subfunc *star = sql_bind_func(sql, "sys", "star", NULL, NULL, F_FUNC, true, true);
    4899        9991 :                         in = exp_op(sql->sa, NULL, star);
    4900        9991 :                         append(fargs, in);
    4901             :                 }
    4902       10062 :                 if (dl)
    4903         324 :                         for (dargs = dl->h ; dargs ; dargs = dargs->next) {
    4904         182 :                                 exp_kind ek = {type_value, card_column, FALSE};
    4905         182 :                                 sql_subtype *empty = sql_bind_localtype("void"), *bte = sql_bind_localtype("bte");
    4906             : 
    4907         182 :                                 in = rel_value_exp2(query, &p, dargs->data.sym, f | sql_window | sql_farg, ek);
    4908         182 :                                 if (!in)
    4909           0 :                                         return NULL;
    4910         182 :                                 if (!exp_subtype(in)) { /* we also do not expect parameters here */
    4911           0 :                                         char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4912           0 :                                         return sql_error(sql, 02, SQLSTATE(42000) "%s: parameters not allowed as arguments to window functions", toUpperCopy(uaname, aname));
    4913             :                                 }
    4914         182 :                                 if (!exp_name(in))
    4915          99 :                                         exp_label(sql->sa, in, ++sql->label);
    4916             : 
    4917             :                                 /* corner case, if the argument is null convert it into something countable such as bte */
    4918         182 :                                 if (subtype_cmp(exp_subtype(in), empty) == 0)
    4919          13 :                                         in = exp_convert(sql, in, empty, bte);
    4920         182 :                                 if ((is_lag || is_lead) && nfargs == 2) { /* lag and lead 3rd arg must have same type as 1st arg */
    4921          10 :                                         sql_exp *first = (sql_exp*) fargs->h->data;
    4922          10 :                                         if (!(in = exp_check_type(sql, exp_subtype(first), p, in, type_equal)))
    4923             :                                                 return NULL;
    4924             :                                 }
    4925         172 :                                 if (!in)
    4926             :                                         return NULL;
    4927             : 
    4928         182 :                                 append(fargs, in);
    4929         182 :                                 in = exp_ref_save(sql, in);
    4930         182 :                                 nfargs++;
    4931             :                         }
    4932             :         } else { /* aggregation function call */
    4933        2737 :                 distinct = dn->next->data.i_val;
    4934        2737 :                 assert(dn->next->next->type == type_list);
    4935        2737 :                 dlist *dl = dn->next->next->data.lval;
    4936        9215 :                 for (dargs = dl?dl->h:NULL; dargs && dargs->data.sym ; dargs = dargs->next) {
    4937        3749 :                         exp_kind ek = {type_value, card_column, FALSE};
    4938        3749 :                         sql_subtype *empty = sql_bind_localtype("void"), *bte = sql_bind_localtype("bte");
    4939             : 
    4940        3749 :                         in = rel_value_exp2(query, &p, dargs->data.sym, f | sql_window | sql_farg, ek);
    4941        3749 :                         if (!in)
    4942           8 :                                 return NULL;
    4943        3742 :                         if (!exp_subtype(in)) { /* we also do not expect parameters here */
    4944           1 :                                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4945           1 :                                 return sql_error(sql, 02, SQLSTATE(42000) "%s: parameters not allowed as arguments to window functions", toUpperCopy(uaname, aname));
    4946             :                         }
    4947        3741 :                         if (!exp_name(in))
    4948        2012 :                                 exp_label(sql->sa, in, ++sql->label);
    4949             : 
    4950             :                         /* corner case, if the argument is null convert it into something countable such as bte */
    4951        3741 :                         if (subtype_cmp(exp_subtype(in), empty) == 0)
    4952          53 :                                 in = exp_convert(sql, in, empty, bte);
    4953        3741 :                         if (!in)
    4954             :                                 return NULL;
    4955             : 
    4956        3741 :                         append(fargs, in);
    4957        3741 :                         in = exp_ref_save(sql, in);
    4958        3741 :                         nfargs++;
    4959             : 
    4960        3741 :                         if (!strcmp(aname, "count"))
    4961          86 :                                 append(fargs, exp_atom_bool(sql->sa, 1)); /* ignore nills */
    4962             :                 }
    4963             : 
    4964        2729 :                 if (!nfargs) { /* count(*) */
    4965         110 :                         if (window_function->token == SQL_AGGR && strcmp(aname, "count") != 0) {
    4966           3 :                                 char *uaname = SA_NEW_ARRAY(sql->ta, char, strlen(aname) + 1);
    4967           3 :                                 return sql_error(sql, 02, SQLSTATE(42000) "%s: unable to perform '%s(*)'", toUpperCopy(uaname, aname), aname);
    4968             :                         }
    4969         107 :                         sql_subfunc *star = sql_bind_func(sql, "sys", "star", NULL, NULL, F_FUNC, true, true);
    4970         107 :                         in = exp_op(sql->sa, NULL, star);
    4971         107 :                         append(fargs, in);
    4972         107 :                         append(fargs, exp_atom_bool(sql->sa, 0)); /* don't ignore nills */
    4973             :                 }
    4974             :         }
    4975             : 
    4976       12788 :         if (distinct)
    4977           4 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: DISTINCT clause is not implemented for window functions");
    4978             : 
    4979             :         /* diff for partitions */
    4980       12784 :         if (gbe) {
    4981        2322 :                 sql_subtype *bt = sql_bind_localtype("bit");
    4982             : 
    4983        4679 :                 for( n = gbe->h; n; n = n->next) {
    4984        2358 :                         sql_subfunc *df;
    4985        2358 :                         sql_exp *e = n->data;
    4986             : 
    4987        2358 :                         if (!exp_subtype(e))
    4988           1 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: parameters not allowed at PARTITION BY clause from window functions");
    4989             : 
    4990        2357 :                         e = exp_copy(sql, e);
    4991        2357 :                         args = sa_list(sql->sa);
    4992        2357 :                         if (pe) {
    4993          36 :                                 df = sql_bind_func(sql, "sys", "diff", bt, exp_subtype(e), F_ANALYTIC, true, true);
    4994          36 :                                 append(args, pe);
    4995             :                         } else {
    4996        2321 :                                 df = sql_bind_func(sql, "sys", "diff", exp_subtype(e), NULL, F_ANALYTIC, true, true);
    4997             :                         }
    4998        2357 :                         if (!df)
    4999           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: function 'diff' not found");
    5000        2357 :                         append(args, e);
    5001        2357 :                         pe = exp_op(sql->sa, args, df);
    5002             :                 }
    5003             :         } else {
    5004       10462 :                 pe = exp_atom_bool(sql->sa, 0);
    5005             :         }
    5006             : 
    5007             :         /* diff for orderby */
    5008       12783 :         if (obe) {
    5009        3044 :                 sql_subtype *bt = sql_bind_localtype("bit");
    5010             : 
    5011        6244 :                 for( n = obe->h; n; n = n->next) {
    5012        3200 :                         sql_subfunc *df;
    5013        3200 :                         sql_exp *e = n->data;
    5014             : 
    5015        3200 :                         if (!exp_subtype(e))
    5016           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: parameters not allowed at ORDER BY clause from window functions");
    5017             : 
    5018        3200 :                         e = exp_copy(sql, e);
    5019        3200 :                         args = sa_list(sql->sa);
    5020        3200 :                         if (oe) {
    5021         156 :                                 df = sql_bind_func(sql, "sys", "diff", bt, exp_subtype(e), F_ANALYTIC, true, true);
    5022         156 :                                 append(args, oe);
    5023             :                         } else {
    5024        3044 :                                 df = sql_bind_func(sql, "sys", "diff", exp_subtype(e), NULL, F_ANALYTIC, true, true);
    5025             :                         }
    5026        3200 :                         if (!df)
    5027           0 :                                 return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: function 'diff' not found");
    5028        3200 :                         append(args, e);
    5029        3200 :                         oe = exp_op(sql->sa, args, df);
    5030             :                 }
    5031             :         } else {
    5032        9739 :                 oe = exp_atom_bool(sql->sa, 0);
    5033             :         }
    5034             : 
    5035       12783 :         if (frame_clause || supports_frames) {
    5036        2726 :                 if (frame_type == FRAME_RANGE)
    5037        2415 :                         ie = obe ? (sql_exp*) obe->t->data : in;
    5038             :                 else
    5039         311 :                         ie = obe ? oe : in;
    5040             :         }
    5041       12783 :         assert(oe && pe);
    5042             : 
    5043       12783 :         types = exp_types(sql->sa, fargs);
    5044       12783 :         wf = bind_func_(sql, sname, aname, types, F_ANALYTIC, false, &found, false);
    5045       12783 :         if (wf && !list_empty(fargs) && !(fargs = check_arguments_and_find_largest_any_type(sql, NULL, fargs, wf, 0, false)))
    5046             :                 wf = NULL;
    5047       12779 :         if (!wf) {
    5048           4 :                 char *arg_list = nfargs ? nary_function_arg_types_2str(sql, types, nfargs) : NULL;
    5049           8 :                 return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: %s window function %s%s%s'%s'(%s)",
    5050           4 :                                                  found ? "insufficient privileges for" : "no such", sname ? "'":"", sname ? sname : "", sname ? "'.":"", aname, arg_list ? arg_list : "");
    5051             :         }
    5052             : 
    5053             :         /* Frame */
    5054       12779 :         if (frame_clause) {
    5055         410 :                 dnode *d = frame_clause->data.lval->h;
    5056         410 :                 symbol *wstart = d->data.sym, *wend = d->next->data.sym, *rstart = wstart->data.lval->h->data.sym,
    5057         410 :                            *rend = wend->data.lval->h->data.sym;
    5058         410 :                 int excl = d->next->next->next->data.i_val;
    5059         410 :                 bool shortcut = false;
    5060             : 
    5061         410 :                 if (!supports_frames)
    5062           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "OVER: frame extend only possible with aggregation and first_value, last_value and nth_value functions");
    5063         406 :                 if (excl != EXCLUDE_NONE)
    5064           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "Only EXCLUDE NO OTHERS exclusion is currently implemented");
    5065         406 :                 if (list_empty(obe) && frame_type == FRAME_GROUPS)
    5066           2 :                         return sql_error(sql, 02, SQLSTATE(42000) "GROUPS frame requires an order by expression");
    5067         404 :                 if (wstart->token == SQL_FOLLOWING && wend->token == SQL_PRECEDING)
    5068           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "FOLLOWING offset must come after PRECEDING offset");
    5069         403 :                 if (wstart->token == SQL_CURRENT_ROW && wend->token == SQL_PRECEDING)
    5070           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "CURRENT ROW offset must come after PRECEDING offset");
    5071         402 :                 if (wstart->token == SQL_FOLLOWING && wend->token == SQL_CURRENT_ROW)
    5072           1 :                         return sql_error(sql, 02, SQLSTATE(42000) "FOLLOWING offset must come after CURRENT ROW offset");
    5073         401 :                 if (wstart->token != SQL_CURRENT_ROW && wend->token != SQL_CURRENT_ROW && wstart->token == wend->token && frame_type != FRAME_ROWS)
    5074           4 :                         return sql_error(sql, 02, SQLSTATE(42000) "Non-centered windows are only supported in row frames");
    5075         367 :                 if (frame_type == FRAME_RANGE) {
    5076          97 :                         if (((wstart->token == SQL_PRECEDING || wstart->token == SQL_FOLLOWING) && rstart->token != SQL_PRECEDING && rstart->token != SQL_CURRENT_ROW && rstart->token != SQL_FOLLOWING) ||
    5077          59 :                                 ((wend->token == SQL_PRECEDING || wend->token == SQL_FOLLOWING) && rend->token != SQL_PRECEDING && rend->token != SQL_CURRENT_ROW && rend->token != SQL_FOLLOWING)) {
    5078          40 :                                 if (list_empty(obe))
    5079           2 :                                         return sql_error(sql, 02, SQLSTATE(42000) "RANGE frame with PRECEDING/FOLLOWING offset requires an order by expression");
    5080          38 :                                 if (list_length(obe) > 1)
    5081           0 :                                         return sql_error(sql, 02, SQLSTATE(42000) "RANGE with offset PRECEDING/FOLLOWING requires exactly one ORDER BY column");
    5082             :                         }
    5083             :                 }
    5084             : 
    5085         395 :                 if (list_empty(obe) && frame_type == FRAME_RANGE) { /* window functions are weird */
    5086             :                         frame_type = FRAME_ALL;
    5087             :                         shortcut = true;
    5088         383 :                 } else if (!is_value && (rstart->token == SQL_PRECEDING || rstart->token == SQL_CURRENT_ROW || rstart->token == SQL_FOLLOWING) && rstart->type == type_int &&
    5089         150 :                         (rend->token == SQL_PRECEDING || rend->token == SQL_CURRENT_ROW || rend->token == SQL_FOLLOWING) && rend->type == type_int) {
    5090             :                          /* special cases, don't calculate bounds */
    5091         146 :                         if (frame_type != FRAME_ROWS && rstart->data.i_val == UNBOUNDED_PRECEDING_BOUND && rend->data.i_val == CURRENT_ROW_BOUND) {
    5092             :                                 frame_type = FRAME_UNBOUNDED_TILL_CURRENT_ROW;
    5093             :                                 shortcut = true;
    5094          21 :                         } else if (frame_type != FRAME_ROWS && rstart->data.i_val == CURRENT_ROW_BOUND && rend->data.i_val == UNBOUNDED_FOLLOWING_BOUND) {
    5095             :                                 frame_type = FRAME_CURRENT_ROW_TILL_UNBOUNDED;
    5096             :                                 shortcut = true;
    5097          86 :                         } else if (rstart->data.i_val == UNBOUNDED_PRECEDING_BOUND && rend->data.i_val == UNBOUNDED_FOLLOWING_BOUND) {
    5098             :                                 frame_type = FRAME_ALL;
    5099             :                                 shortcut = true;
    5100          76 :                         } else if (rstart->data.i_val == CURRENT_ROW_BOUND && rend->data.i_val == CURRENT_ROW_BOUND) {
    5101         395 :                                 frame_type = FRAME_CURRENT_ROW;
    5102         395 :                                 shortcut = true;
    5103             :                         }
    5104             :                 }
    5105         395 :                 if (!shortcut) {
    5106         302 :                         if (!(fstart = calculate_window_bound(query, p, wstart->token, rstart, ie, frame_type, f | sql_window)))
    5107             :                                 return NULL;
    5108         299 :                         if (!(fend = calculate_window_bound(query, p, wend->token, rend, ie, frame_type, f | sql_window)))
    5109             :                                 return NULL;
    5110         471 :                         if (!generate_window_bound_call(sql, &start, &eend, gbe ? pe : NULL, ie, fstart, fend, frame_type, excl,
    5111             :                                                                                         wstart->token, wend->token))
    5112             :                                 return NULL;
    5113             :                 }
    5114       12369 :         } else if (supports_frames) { /* for analytic functions with no frame clause, we use the standard default values */
    5115        2314 :                 if (is_value) {
    5116         108 :                         sql_subtype *bound_tp = sql_bind_localtype("lng"), *bt = (frame_type == FRAME_ROWS || frame_type == FRAME_GROUPS) ? bound_tp : exp_subtype(ie);
    5117         108 :                         unsigned char sclass = bt->type->eclass;
    5118             : 
    5119         136 :                         fstart = exp_atom(sql->sa, atom_max_value(sql->sa, EC_NUMERIC(sclass) ? bt : bound_tp));
    5120         108 :                         fend = order_by_clause ? exp_atom(sql->sa, atom_zero_value(sql->sa, EC_NUMERIC(sclass) ? bt : bound_tp)) :
    5121          23 :                                                                          exp_atom(sql->sa, atom_max_value(sql->sa, EC_NUMERIC(sclass) ? bt : bound_tp));
    5122             : 
    5123         157 :                         if (generate_window_bound_call(sql, &start, &eend, gbe ? pe : NULL, ie, fstart, fend, frame_type, EXCLUDE_NONE, SQL_PRECEDING, SQL_FOLLOWING) == NULL)
    5124             :                                 return NULL;
    5125             :                 } else {
    5126        2206 :                         frame_type = list_empty(obe) ? FRAME_ALL : FRAME_UNBOUNDED_TILL_CURRENT_ROW;
    5127             :                 }
    5128             :         }
    5129             : 
    5130       12761 :         args = sa_list(sql->sa);
    5131       26925 :         for (node *n = fargs->h ; n ; n = n->next)
    5132       14164 :                 list_append(args, n->data);
    5133       12761 :         list_append(args, pe);
    5134       12761 :         list_append(args, oe);
    5135       12761 :         if (supports_frames) {
    5136        2706 :                 list_append(args, exp_atom_int(sql->sa, frame_type));
    5137        2706 :                 list_append(args, start ? start : exp_atom_oid(sql->sa, 1));
    5138        2706 :                 list_append(args, eend ? eend : exp_atom_oid(sql->sa, 1));
    5139             :         }
    5140       12761 :         call = exp_rank_op(sql->sa, list_empty(args) ? NULL : args, gbe, obe, wf);
    5141       12761 :         *rel = p;
    5142       12761 :         return call;
    5143             : }
    5144             : 
    5145             : sql_exp *
    5146     6088359 : rel_value_exp2(sql_query *query, sql_rel **rel, symbol *se, int f, exp_kind ek)
    5147             : {
    5148     6088359 :         mvc *sql = query->sql;
    5149     6088359 :         if (!se)
    5150             :                 return NULL;
    5151             : 
    5152     6088359 :         if (mvc_highwater(sql))
    5153           9 :                 return sql_error(sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
    5154             : 
    5155     6088349 :         if (rel && *rel && (*rel)->card == CARD_AGGR) { /* group by expression case, handle it before */
    5156      137885 :                 sql_exp *exp = NULL;
    5157      137885 :                 if (!is_sql_aggr(f) && !is_sql_window(f))
    5158      137419 :                         exp = frame_get_groupby_expression(sql, se);
    5159      137885 :                 if (sql->errstr[0] != '\0')
    5160             :                         return NULL;
    5161      137885 :                 if (exp) {
    5162          80 :                         sql_exp *res = exp_ref(sql, exp);
    5163          80 :                         res->card = (*rel)->card;
    5164          80 :                         if (se->token == SQL_AGGR) {
    5165           0 :                                 dlist *l = se->data.lval;
    5166           0 :                                 int distinct = l->h->next->data.i_val;
    5167           0 :                                 if (distinct)
    5168           0 :                                         set_distinct(res);
    5169             :                         }
    5170          80 :                         if (!query_has_outer(query) && is_groupby((*rel)->op))
    5171          76 :                                 res = rel_groupby_add_aggr(sql, *rel, res);
    5172          80 :                         return res;
    5173             :                 }
    5174             :         }
    5175             : 
    5176     6088269 :         switch (se->token) {
    5177      593209 :         case SQL_NOP:
    5178      593209 :                 return rel_nop(query, rel, se, f, ek);
    5179       19607 :         case SQL_AGGR:
    5180       19607 :                 return rel_aggr(query, rel, se, f);
    5181       12839 :         case SQL_WINDOW:
    5182       12839 :                 return rel_rankop(query, rel, se, f);
    5183     2392878 :         case SQL_IDENT:
    5184             :         case SQL_COLUMN:
    5185     2392878 :                 return rel_column_ref(query, rel, se, f );
    5186        3055 :         case SQL_NAME: {
    5187        3055 :                 dlist *l = se->data.lval;
    5188        3055 :                 const char *sname = qname_schema(l);
    5189        3055 :                 const char *vname = qname_schema_object(l);
    5190        3055 :                 return rel_exp_variable_on_scope(sql, sname, vname);
    5191             :         }
    5192       51239 :         case SQL_VALUES:
    5193             :         case SQL_WITH:
    5194             :         case SQL_SELECT: {
    5195       51239 :                 sql_rel *r = NULL;
    5196             : 
    5197       51239 :                 if (is_psm_call(f) || is_sql_merge(f))
    5198           8 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: subqueries not supported inside %s", is_psm_call(f) ? "CALL" : "MERGE", is_psm_call(f) ? "CALL statements" : "MERGE conditions");
    5199       51235 :                 if (rel && *rel)
    5200       36948 :                         query_push_outer(query, *rel, f);
    5201       51235 :                 if (se->token == SQL_WITH) {
    5202          36 :                         r = rel_with_query(query, se);
    5203       51199 :                 } else if (se->token == SQL_VALUES) {
    5204         240 :                         r = rel_values(query, se, NULL);
    5205             :                 } else {
    5206       50959 :                         assert(se->token == SQL_SELECT);
    5207       50959 :                         exp_kind nek = ek;
    5208       50959 :                         nek.aggr = is_sql_aggr(f);
    5209       50959 :                         if (is_sql_no_subquery(f))
    5210           3 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery not allowed");
    5211             : 
    5212       50956 :                         r = rel_subquery(query, se, nek);
    5213       50956 :                         if (r)
    5214       50855 :                                 exps_label(sql, r->exps);
    5215             :                 }
    5216       51232 :                 if (rel && *rel) {
    5217       36945 :                         *rel = query_pop_outer(query);
    5218       36945 :                         if (is_sql_join(f) && is_groupby((*rel)->op)) {
    5219           0 :                                 return sql_error(sql, 05, SQLSTATE(42000) "SELECT: aggregate functions not allowed in JOIN conditions");
    5220       36945 :                         } else if (is_sql_where(f) && is_groupby((*rel)->op)) {
    5221           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions not allowed in WHERE clause");
    5222       36945 :                         } else if ((is_sql_update_set(f) || is_sql_psm(f)) && is_groupby((*rel)->op)) {
    5223           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions not allowed in SET, WHILE, IF, ELSE, CASE, WHEN, RETURN, ANALYZE clauses");
    5224             :                         }
    5225             :                 }
    5226       51232 :                 if (!r)
    5227             :                         return NULL;
    5228       51128 :                 if (ek.type == type_value && ek.card <= card_set && is_project(r->op) && list_length(r->exps) > 1)
    5229          17 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery must return only one column");
    5230       51111 :                 if (ek.type == type_relation && is_project(r->op) && list_length(r->exps) != ek.type)
    5231           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery has too %s columns", list_length(r->exps) < ek.type ? "few" : "many");
    5232       51111 :                 if (ek.type == type_value && list_length(r->exps) == 1 && !is_sql_psm(f)) /* for now don't rename multi attribute results */
    5233       38701 :                         r = rel_zero_or_one(sql, r, ek);
    5234       51111 :                 return exp_rel(sql, r);
    5235             :         }
    5236         215 :         case SQL_TABLE: {
    5237             :                 /* turn a subquery into a tabular result */
    5238         215 :                 *rel = rel_selects(query, se->data.sym);
    5239         215 :                 if (*rel)
    5240         213 :                         return lastexp(*rel);
    5241             :                 return NULL;
    5242             :         }
    5243        1611 :         case SQL_PARAMETER: {
    5244        1611 :                 assert(se->type == type_int);
    5245        1611 :                 sql_arg *a = sql_bind_paramnr(sql, se->data.i_val);
    5246        1611 :                 if (sql->emode != m_prepare) {
    5247           1 :                         if (a && a->name && a->name[0])
    5248           1 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: named placeholder ('%s') but named values list is missing", a->name);
    5249             :                         else
    5250           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: parameters ('?') not allowed in normal queries, use PREPARE");
    5251             :                 }
    5252        1610 :                 return exp_atom_ref(sql->sa, se->data.i_val, a?&a->type:NULL);
    5253             :         }
    5254      109527 :         case SQL_NULL:
    5255      109527 :                 return exp_null(sql->sa, sql_bind_localtype("void"));
    5256        1590 :         case SQL_NEXT:
    5257        1590 :                 return rel_next_value_for(sql, se);
    5258      176254 :         case SQL_CAST:
    5259      176254 :                 return rel_cast(query, rel, se, f);
    5260      106443 :         case SQL_CASE:
    5261             :         case SQL_COALESCE:
    5262             :         case SQL_NULLIF:
    5263      106443 :                 return rel_case_exp(query, rel, se, f);
    5264           0 :         case SQL_RANK:
    5265           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: window function %s requires an OVER clause", qname_schema_object(se->data.lval->h->data.lval));
    5266          25 :         case SQL_XMLELEMENT:
    5267             :         case SQL_XMLFOREST:
    5268             :         case SQL_XMLCOMMENT:
    5269             :         case SQL_XMLATTRIBUTE:
    5270             :         case SQL_XMLCONCAT:
    5271             :         case SQL_XMLDOCUMENT:
    5272             :         case SQL_XMLPI:
    5273             :         case SQL_XMLTEXT:
    5274          25 :                 return rel_xml(query, rel, se, f, ek);
    5275     2619777 :         default:
    5276     2619777 :                 return rel_logical_value_exp(query, rel, se, f, ek);
    5277             :         }
    5278             : }
    5279             : 
    5280             : static int exps_has_rank(list *exps);
    5281             : 
    5282             : static int
    5283             : exp_has_rank(sql_exp *e)
    5284             : {
    5285             :         switch(e->type) {
    5286             :         case e_convert:
    5287             :                 return exp_has_rank(e->l);
    5288             :         case e_func:
    5289             :                 if (e->r)
    5290             :                         return 1;
    5291             :                 /* fall through */
    5292             :         case e_aggr:
    5293             :                 return exps_has_rank(e->l);
    5294             :         case e_cmp:
    5295             :                 if (e->flag == cmp_or || e->flag == cmp_filter)
    5296             :                         return exps_has_rank(e->l) || exps_has_rank(e->r);
    5297             :                 if (e->flag == cmp_in || e->flag == cmp_notin)
    5298             :                         return exp_has_rank(e->l) || exps_has_rank(e->r);
    5299             :                 return exp_has_rank(e->l) || exp_has_rank(e->r) || (e->f && exp_has_rank(e->f));
    5300             :         default:
    5301             :                 return 0;
    5302             :         }
    5303             : }
    5304             : 
    5305             : /* TODO create exps_has (list, fptr ) */
    5306             : static int
    5307             : exps_has_rank(list *exps)
    5308             : {
    5309             :         if (!exps || list_empty(exps))
    5310             :                 return 0;
    5311             :         for(node *n = exps->h; n; n=n->next){
    5312             :                 sql_exp *e = n->data;
    5313             : 
    5314             :                 if (exp_has_rank(e))
    5315             :                         return 1;
    5316             :         }
    5317             :         return 0;
    5318             : }
    5319             : 
    5320             : sql_exp *
    5321     4657718 : rel_value_exp(sql_query *query, sql_rel **rel, symbol *se, int f, exp_kind ek)
    5322             : {
    5323     4657718 :         if (!se)
    5324             :                 return NULL;
    5325             : 
    5326     4657718 :         if (mvc_highwater(query->sql))
    5327           1 :                 return sql_error(query->sql, 10, SQLSTATE(42000) "Query too complex: running out of stack space");
    5328             : 
    5329     4657717 :         sql_exp *e = rel_value_exp2(query, rel, se, f, ek);
    5330     4657710 :         if (e && (se->token == SQL_SELECT || se->token == SQL_TABLE) && !exp_is_rel(e)) {
    5331           0 :                 assert(*rel);
    5332           0 :                 return rel_lastexp(query->sql, *rel);
    5333             :         }
    5334             :         return e;
    5335             : }
    5336             : 
    5337             : static sql_exp *
    5338     1352976 : column_exp(sql_query *query, sql_rel **rel, symbol *column_e, int f)
    5339             : {
    5340     1352976 :         dlist *l = column_e->data.lval;
    5341     1352976 :         exp_kind ek = {type_value, card_column, FALSE};
    5342     1352976 :         sql_exp *ve;
    5343             : 
    5344     1352976 :         if (f == sql_sel && rel && *rel && (*rel)->card < CARD_AGGR)
    5345       83228 :                 ek.card = card_value;
    5346     1352976 :         ve = rel_value_exp(query, rel, l->h->data.sym, f, ek);
    5347     1352974 :         if (!ve)
    5348             :                 return NULL;
    5349             :         /* AS name */
    5350     1337122 :         if (ve && l->h->next->data.sval)
    5351      513609 :                 exp_setname(query->sql, ve, NULL, l->h->next->data.sval);
    5352             :         return ve;
    5353             : }
    5354             : 
    5355             : static int
    5356      852851 : exp_is_not_intern(sql_exp *e)
    5357             : {
    5358      852851 :         return is_intern(e)?-1:0;
    5359             : }
    5360             : 
    5361             : static void
    5362      130650 : rel_remove_internal_exp(sql_rel *rel)
    5363             : {
    5364      130650 :         if (rel->exps) {
    5365      130650 :                 list *n_exps = list_select(rel->exps, rel, (fcmp)&exp_is_not_intern, (fdup)NULL);
    5366             : 
    5367      130650 :                 rel->exps = n_exps;
    5368             :         }
    5369      130650 : }
    5370             : 
    5371             : static inline int
    5372          39 : exp_key(sql_exp *e)
    5373             : {
    5374          39 :         if (e->alias.name)
    5375          39 :                 return hash_key(e->alias.name);
    5376             :         return 0;
    5377             : }
    5378             : 
    5379             : static list *
    5380          17 : group_merge_exps(mvc *sql, list *gexps, list *exps)
    5381             : {
    5382          17 :         int nexps = list_length(gexps) + list_length(exps);
    5383             : 
    5384          17 :         sql_hash *ht = hash_new(sql->ta, nexps, (fkeyvalue)&exp_key);
    5385             : 
    5386          27 :         for (node *n = gexps->h; n ; n = n->next) { /* first add grouping expressions */
    5387          10 :                 sql_exp *e = n->data;
    5388          10 :                 int key = ht->key(e);
    5389             : 
    5390          10 :                 hash_add(ht, key, e);
    5391             :         }
    5392             : 
    5393          46 :         for (node *n = exps->h; n ; n = n->next) { /* then test if the new grouping expressions are already there */
    5394          29 :                 sql_exp *e = n->data;
    5395          29 :                 int key = ht->key(e);
    5396          29 :                 sql_hash_e *he = ht->buckets[key&(ht->size-1)];
    5397          29 :                 bool duplicates = false;
    5398             : 
    5399          37 :                 for (; he && !duplicates; he = he->chain) {
    5400           8 :                         sql_exp *f = he->value;
    5401             : 
    5402           8 :                         if (!exp_equal(e, f))
    5403           4 :                                 duplicates = true;
    5404             :                 }
    5405          29 :                 hash_add(ht, key, e);
    5406          29 :                 if (!duplicates) {
    5407          25 :                         list_append(gexps, e);
    5408          25 :                         n->data = exp_ref(sql, e);
    5409             :                 }
    5410             :         }
    5411          17 :         return gexps;
    5412             : }
    5413             : 
    5414             : static list *
    5415       59246 : rel_table_exp(sql_query *query, sql_rel **rel, symbol *column_e, bool single_exp )
    5416             : {
    5417       59246 :         mvc *sql = query->sql;
    5418       59246 :         if (column_e->token == SQL_TABLE && column_e->data.lval->h->type == type_symbol) {
    5419           0 :                 sql_rel *r;
    5420             : 
    5421           0 :                 if (!is_project((*rel)->op))
    5422             :                         return NULL;
    5423           0 :                 r = rel_named_table_function(query, (*rel)->l, column_e, 0, NULL);
    5424           0 :                 if (!r)
    5425             :                         return NULL;
    5426           0 :                 *rel = r;
    5427           0 :                 return sa_list(sql->sa);
    5428             :         }
    5429       59246 :         char *tname = NULL;
    5430       59246 :         if (column_e->token == SQL_TABLE) {
    5431       43405 :                 tname = column_e->data.lval->h->data.sval;
    5432       15841 :         } else if (column_e->token == SQL_COLUMN && column_e->data.lval->h->type == type_symbol) {
    5433       15841 :                 symbol *sym = column_e->data.lval->h->data.sym;
    5434       15841 :                 if (sym->token == SQL_COLUMN)
    5435       14847 :                         tname = sym->data.lval->h->data.sval;
    5436             :                 else
    5437             :                         return NULL;
    5438             :         } else {
    5439             :                 return NULL;
    5440             :         }
    5441             : 
    5442       58252 :         list *exps = NULL;
    5443       58252 :         sql_rel *project = *rel, *groupby = NULL;
    5444             : 
    5445             :         /* if there's a group by relation in the tree, skip it for the '*' case and use the underlying projection */
    5446       58252 :         if (project) {
    5447       67698 :                 while (is_groupby(project->op) || is_select(project->op)) {
    5448        9446 :                         if (is_groupby(project->op))
    5449          37 :                                 groupby = project;
    5450        9446 :                         if (project->l)
    5451        9446 :                                 project = project->l;
    5452             :                 }
    5453             :                 assert(project);
    5454             :         }
    5455             : 
    5456       58252 :         if (project->op == op_project && project->l && project == *rel && !tname && !rel_is_ref(project) && !need_distinct(project) && single_exp) {
    5457         549 :                 sql_rel *l = project->l;
    5458         549 :                 if (!l || !is_project(l->op) || list_length(project->exps) == list_length(l->exps)) {
    5459         542 :                         rel_remove_internal_exp(*rel);
    5460         542 :                         exps = project->exps;
    5461         542 :                         *rel = project->l;
    5462             :                 }
    5463             :         }
    5464       58252 :         if ((exps || (exps = rel_table_projections(sql, project, tname, 0)) != NULL) && !list_empty(exps)) {
    5465       58149 :                 if (!(exps = check_distinct_exp_names(sql, exps)))
    5466           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "Duplicate column names in table%s%s%s projection list", tname ? " '" : "", tname ? tname : "", tname ? "'" : "");
    5467       58149 :                 if (groupby) {
    5468          17 :                         groupby->exps = group_merge_exps(sql, groupby->exps, exps);
    5469          40 :                         for (node *n = groupby->exps->h ; n ; n = n->next) {
    5470          30 :                                 sql_exp *e = n->data;
    5471             : 
    5472          30 :                                 if (e->card > groupby->card) {
    5473           7 :                                         if (exp_name(e) && !has_label(e))
    5474           7 :                                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(e));
    5475           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    5476             :                                 }
    5477             :                         }
    5478             :                 }
    5479       58142 :                 return exps;
    5480             :         }
    5481         103 :         if (!tname)
    5482           2 :                 return sql_error(sql, 02, SQLSTATE(42000) "Table expression without table name");
    5483         101 :         return sql_error(sql, 02, SQLSTATE(42000) "Column expression Table '%s' unknown", tname);
    5484             : }
    5485             : 
    5486             : sql_exp *
    5487     1396381 : rel_column_exp(sql_query *query, sql_rel **rel, symbol *column_e, int f)
    5488             : {
    5489     1396381 :         if (column_e->token == SQL_COLUMN || column_e->token == SQL_IDENT)
    5490     1352976 :                 return column_exp(query, rel, column_e, f);
    5491             :         return NULL;
    5492             : }
    5493             : 
    5494             : static sql_rel*
    5495      377727 : rel_where_groupby_nodes(sql_query *query, sql_rel *rel, SelectNode *sn, int *group_totals)
    5496             : {
    5497      377727 :         mvc *sql = query->sql;
    5498             : 
    5499      377727 :         if (sn->where) {
    5500      147801 :                 rel = rel_logical_exp(query, rel, sn->where, sql_where);
    5501      147801 :                 if (!rel) {
    5502          76 :                         if (sql->errstr[0] == 0)
    5503           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "Subquery result missing");
    5504             :                         return NULL;
    5505             :                 }
    5506             :         }
    5507      377651 :         query_processed(query);
    5508             : 
    5509      377651 :         if (rel && sn->groupby) {
    5510       13553 :                 list *gbe = NULL, *sets = NULL;
    5511       13553 :                 int all = 0;
    5512       13553 :                 if (sn->groupby->data.lval == NULL) { /* ALL */
    5513             :                         all = 1;
    5514             :                 } else {
    5515       50069 :                         for (dnode *o = sn->groupby->data.lval->h; o ; o = o->next) {
    5516       36643 :                                 symbol *grouping = o->data.sym;
    5517       36643 :                                 if (grouping->token == SQL_ROLLUP || grouping->token == SQL_CUBE || grouping->token == SQL_GROUPING_SETS) {
    5518         119 :                                         *group_totals |= sql_group_totals;
    5519         119 :                                         break;
    5520             :                                 }
    5521             :                         }
    5522       13545 :                         gbe = rel_groupings(query, &rel, sn->groupby, sn->selection, sql_sel | sql_groupby | *group_totals, false, &sets);
    5523       13545 :                         if (!gbe)
    5524          20 :                                 return NULL;
    5525             :                 }
    5526       13533 :                 rel = rel_groupby(sql, rel, gbe);
    5527       13533 :                 if (rel && all)
    5528           8 :                         rel->flag = 2;
    5529       13533 :                 if (sets && list_length(sets) > 1) { /* if there is only one combination, there is no reason to generate unions */
    5530         114 :                         prop *p = prop_create(sql->sa, PROP_GROUPINGS, rel->p);
    5531         114 :                         p->value.pval = sets;
    5532         114 :                         rel->p = p;
    5533             :                 }
    5534             :         }
    5535             : 
    5536      377631 :         if (rel && sn->having) {
    5537             :                 /* having implies group by, ie if not supplied do a group by */
    5538        1766 :                 if (rel->op != op_groupby)
    5539         128 :                         rel = rel_groupby(sql, rel, NULL);
    5540             :         }
    5541      377631 :         query_processed(query);
    5542      377631 :         return rel;
    5543             : }
    5544             : 
    5545             : static sql_rel*
    5546      376519 : rel_having_limits_nodes(sql_query *query, sql_rel *rel, SelectNode *sn, exp_kind ek, int group_totals)
    5547             : {
    5548      376519 :         mvc *sql = query->sql;
    5549      376519 :         sql_rel *inner = NULL;
    5550      376519 :         int single_value = 1;
    5551             : 
    5552      376519 :         if (is_project(rel->op) && rel->l) {
    5553      376519 :                 inner = rel->l;
    5554      376519 :                 single_value = 0;
    5555             :         }
    5556             : 
    5557      376519 :         if (sn->having) {
    5558        1759 :                 if (inner && is_groupby(inner->op))
    5559        1759 :                         set_processed(inner);
    5560        1759 :                 if (!(inner = rel_logical_exp(query, inner, sn->having, sql_having | group_totals)))
    5561             :                         return NULL;
    5562        1718 :                 if (inner->exps && exps_card(inner->exps) > CARD_AGGR)
    5563           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: cannot compare sets with values, probably an aggregate function missing");
    5564        1718 :                 if (!single_value)
    5565        1718 :                         rel->l = inner;
    5566             :         }
    5567             : 
    5568      376478 :         if (sn->qualify) {
    5569           0 :                 if (!(rel = rel_logical_exp(query, rel, sn->qualify, sql_qualify)))
    5570             :                         return NULL;
    5571             :         }
    5572             : 
    5573      376478 :         if (rel && sn->distinct)
    5574        1039 :                 rel = rel_distinct(rel);
    5575             : 
    5576      376478 :         if (rel && sn->orderby) {
    5577       37435 :                 list *obe = NULL;
    5578       37435 :                 sql_rel *sel = NULL, *l = rel->l;
    5579             : 
    5580             :                 /* project( select ) */
    5581       37435 :                 if (sn->having && is_select(l->op)) {
    5582          52 :                         sel = l;
    5583          52 :                         rel->l = l->l;
    5584             :                 }
    5585       37435 :                 rel = rel_orderby(sql, rel);
    5586       37435 :                 set_processed(rel);
    5587       37435 :                 obe = rel_order_by(query, &rel, sn->orderby, sn->distinct, sql_orderby | group_totals);
    5588       37435 :                 if (!obe)
    5589             :                         return NULL;
    5590       37394 :                 rel->r = obe;
    5591       37394 :                 if (sel) {
    5592          51 :                         sql_rel *o = rel, *p = o->l;
    5593          51 :                         p->l = sel;
    5594             :                 }
    5595             :         }
    5596      376437 :         if (!rel)
    5597             :                 return NULL;
    5598             : 
    5599      376437 :         if (sn->limit || sn->offset) {
    5600       16628 :                 sql_subtype *lng = sql_bind_localtype("lng");
    5601       16628 :                 list *exps = new_exp_list(sql->sa);
    5602             : 
    5603       16628 :                 if (sn->limit) {
    5604       16602 :                         sql_exp *l = rel_value_exp(query, NULL, sn->limit, 0, ek);
    5605             : 
    5606       16602 :                         if (!l || !(l=exp_check_type(sql, lng, NULL, l, type_equal)))
    5607           0 :                                 return NULL;
    5608       16602 :                         if ((ek.card != card_relation && sn->limit) &&
    5609             :                                 (ek.card == card_value && sn->limit)) {
    5610          12 :                                 sql_subfunc *zero_or_one = sql_bind_func(sql, "sys", "zero_or_one", exp_subtype(l), NULL, F_AGGR, true, true);
    5611          12 :                                 l = exp_aggr1(sql->sa, l, zero_or_one, 0, 0, CARD_ATOM, has_nil(l));
    5612             :                         }
    5613       16602 :                         list_append(exps, l);
    5614             :                 } else
    5615          26 :                         list_append(exps, exp_atom(sql->sa, atom_general(sql->sa, lng, NULL, 0)));
    5616       16628 :                 if (sn->offset) {
    5617          86 :                         sql_exp *o = rel_value_exp( query, NULL, sn->offset, 0, ek);
    5618          86 :                         if (!o || !(o=exp_check_type(sql, lng, NULL, o, type_equal)))
    5619           0 :                                 return NULL;
    5620          86 :                         list_append(exps, o);
    5621             :                 }
    5622       16628 :                 rel = rel_topn(sql->sa, rel, exps);
    5623             :         }
    5624             : 
    5625      376437 :         if (sn->sample || sn->seed) {
    5626          23 :                 list *exps = new_exp_list(sql->sa);
    5627             : 
    5628          23 :                 if (sn->sample) {
    5629          23 :                         sql_exp *s = rel_value_exp(query, NULL, sn->sample, 0, ek);
    5630          23 :                         if (!s)
    5631             :                                 return NULL;
    5632          23 :                         if (!exp_subtype(s) && rel_set_type_param(sql, sql_bind_localtype("lng"), NULL, s, 0) < 0)
    5633             :                                 return NULL;
    5634          23 :                         list_append(exps, s);
    5635             :                 } else {
    5636           0 :                         assert(sn->seed);
    5637           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SEED: cannot have SEED without SAMPLE");
    5638             :                 }
    5639          23 :                 if (sn->seed) {
    5640          12 :                         sql_exp *e = rel_value_exp(query, NULL, sn->seed, 0, ek);
    5641          12 :                         if (!e || !(e=exp_check_type(sql, sql_bind_localtype("int"), NULL, e, type_equal)))
    5642           0 :                                 return NULL;
    5643          12 :                         list_append(exps, e);
    5644             :                 }
    5645          23 :                 rel = rel_sample(sql->sa, rel, exps);
    5646             :         }
    5647             : 
    5648             :         /* after parsing the current query, set the group by relation as processed */
    5649      376437 :         if (!sn->having && inner && is_groupby(inner->op))
    5650       33024 :                 set_processed(inner);
    5651      376437 :         if (rel)
    5652      376437 :                 set_processed(rel);
    5653             :         return rel;
    5654             : }
    5655             : 
    5656             : static sql_rel *
    5657          62 : join_on_column_name(sql_query *query, sql_rel *rel, sql_rel *t1, sql_rel *t2, int op, int l_nil, int r_nil)
    5658             : {
    5659          62 :         mvc *sql = query->sql;
    5660          62 :         int found = 0, full = (op == op_full), right = (op == op_right);
    5661          62 :         list *exps = rel_projections(sql, t1, NULL, 1, 0);
    5662          62 :         list *r_exps = rel_projections(sql, t2, NULL, 1, 0);
    5663          62 :         list *outexps = new_exp_list(sql->sa);
    5664             : 
    5665          62 :         if (!exps || !r_exps)
    5666             :                 return NULL;
    5667         244 :         for (node *n = exps->h; n; n = n->next) {
    5668         185 :                 sql_exp *le = n->data;
    5669         185 :                 int multi = 0;
    5670         185 :                 const char *rname = exp_relname(le), *name = exp_name(le);
    5671         185 :                 sql_exp *re = exps_bind_column(r_exps, name, NULL, &multi, 0);
    5672             : 
    5673         185 :                 if (re) {
    5674         140 :                         if (multi)
    5675           3 :                                 return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "NATURAL JOIN: common column name '%s' appears more than once in right table", rname);
    5676         138 :                         multi = 0;
    5677         138 :                         le = exps_bind_column(exps, name, NULL, &multi, 0);
    5678         138 :                         if (multi)
    5679           0 :                                 return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "NATURAL JOIN: common column name '%s' appears more than once in left table", rname);
    5680             : 
    5681         138 :                         found = 1;
    5682         138 :                         if (!(rel = rel_compare_exp(query, rel, le, re, "=", TRUE, 0, 0, 0, 0)))
    5683             :                                 return NULL;
    5684         137 :                         list_remove_data(r_exps, NULL, re);
    5685         137 :                         if (full) {
    5686          12 :                                 sql_exp *cond = rel_unop_(sql, rel, le, "sys", "isnull", card_value);
    5687          12 :                                 if (!cond)
    5688             :                                         return NULL;
    5689          12 :                                 set_has_no_nil(cond);
    5690          12 :                                 if (rel_convert_types(sql, NULL, NULL, &le, &re, 1, type_equal_no_any) < 0)
    5691             :                                         return NULL;
    5692          12 :                                 if (!(le = rel_nop_(sql, rel, cond, re, le, NULL, "sys", "ifthenelse", card_value)))
    5693             :                                         return NULL;
    5694         125 :                         } else if (right) {
    5695           3 :                                 le = re;
    5696             :                         }
    5697         137 :                         exp_setname(sql, le, rname, name);
    5698         137 :                         set_not_unique(le);
    5699         137 :                         append(outexps, le);
    5700             :                 } else {
    5701          45 :                         if (l_nil)
    5702           5 :                                 set_has_nil(le);
    5703          45 :                         set_not_unique(le);
    5704          45 :                         append(outexps, le);
    5705             :                 }
    5706             :         }
    5707          59 :         if (!found)
    5708           2 :                 return sql_error(sql, 02, SQLSTATE(42000) "JOIN: no columns of tables '%s' and '%s' match", rel_name(t1)?rel_name(t1):"", rel_name(t2)?rel_name(t2):"");
    5709          90 :         for (node *n = r_exps->h; n; n = n->next) {
    5710          33 :                 sql_exp *re = n->data;
    5711          33 :                 if (r_nil)
    5712           3 :                         set_has_nil(re);
    5713          33 :                 set_not_unique(re);
    5714          33 :                 append(outexps, re);
    5715             :         }
    5716          57 :         rel = rel_project(sql->sa, rel, outexps);
    5717          57 :         return rel;
    5718             : }
    5719             : 
    5720             : static sql_rel *
    5721      377727 : rel_select_exp(sql_query *query, sql_rel *rel, SelectNode *sn, exp_kind ek)
    5722             : {
    5723      377727 :         mvc *sql = query->sql;
    5724      377727 :         sql_rel *inner = NULL;
    5725      377727 :         int group_totals = 0;
    5726      377727 :         list *pexps = NULL;
    5727             : 
    5728      377727 :         assert(sn->s.token == SQL_SELECT);
    5729      377727 :         if (!sn->selection)
    5730           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: the selection or from part is missing");
    5731             : 
    5732      377727 :         if (!rel)
    5733       72933 :                 rel = rel_project_exp(sql, exp_atom_bool(sql->sa, 1));
    5734      377727 :         rel = rel_where_groupby_nodes(query, rel, sn, &group_totals);
    5735      377727 :         if (sql->session->status) /* rel might be NULL as input, so we have to check for the session status for errors */
    5736             :                 return NULL;
    5737             : 
    5738      377628 :         inner = rel;
    5739      377628 :         pexps = sa_list(sql->sa);
    5740     1772868 :         for (dnode *n = sn->selection->h; n; n = n->next) {
    5741             :                 /* Here we could get real column expressions
    5742             :                  * (including single atoms) but also table results.
    5743             :                  * Therefore we try both rel_column_exp
    5744             :                  * and rel_table_exp.
    5745             :                  */
    5746     1396345 :                 list *te = NULL;
    5747     2792631 :                 sql_exp *ce = rel_column_exp(query, &inner, n->data.sym, sql_sel | group_totals | (ek.aggr?sql_aggr:0));
    5748             : 
    5749     1396344 :                 if (ce) {
    5750     1337098 :                         if (inner && inner->flag && is_groupby(inner->op)) {
    5751          62 :                                 int found = 0;
    5752          62 :                                 list *gbe = inner->r;
    5753             :                                 /* flag == 2 just add to group by/ aggrs and ref-to pexps*/
    5754             :                                 /* flag == 1 find group by exp referencing this column nr */
    5755          62 :                                 if (inner->flag == 2) {
    5756          25 :                                         if (ce->card > CARD_AGGR) {
    5757           9 :                                                 if (!gbe)
    5758           8 :                                                         inner->r = gbe = sa_list(sql->sa);
    5759           9 :                                                 append(gbe, ce);
    5760           9 :                                                 ce = exp_ref(sql, ce);
    5761           9 :                                                 ce->card = CARD_AGGR;
    5762           9 :                                                 list_append(inner->exps, ce);
    5763           9 :                                                 ce = exp_ref(sql, ce);
    5764           9 :                                                 found = 1;
    5765             :                                         }
    5766             :                                 } else {
    5767          81 :                                         for(node *n = gbe->h; n && !found; n = n->next) {
    5768          44 :                                                 sql_exp *e = n->data;
    5769          44 :                                                 if (is_atom(e->type) && !e->alias.name) {
    5770          29 :                                                         atom *a = e->l;
    5771          29 :                                                         int nr = (int)atom_get_int(a);
    5772          29 :                                                         if (nr == (list_length(pexps) + 1)) {
    5773          27 :                                                                 n->data = ce;
    5774          27 :                                                                 ce = exp_ref(sql, ce);
    5775          27 :                                                                 ce->card = CARD_AGGR;
    5776          27 :                                                                 list_append(inner->exps, ce);
    5777          27 :                                                                 ce = exp_ref(sql, ce);
    5778          27 :                                                                 found = 1;
    5779             :                                                         }
    5780             :                                                 }
    5781             :                                         }
    5782             :                                 }
    5783             :                         }
    5784     1337098 :                         pexps = append(pexps, ce);
    5785     1337098 :                         rel = inner;
    5786     1337098 :                         continue;
    5787             :                 } else {
    5788       63813 :                         te = rel_table_exp(query, &rel, n->data.sym, !list_length(pexps) && !n->next);
    5789             :                 }
    5790       59246 :                 if (!ce && !te) {
    5791        1104 :                         if (sql->errstr[0])
    5792             :                                 return NULL;
    5793           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: subquery result missing");
    5794             :                 }
    5795             :                 /* here we should merge the column expressions we
    5796             :                  * obtained so far with the table expression, ie
    5797             :                  * t1.* or a subquery.
    5798             :                  */
    5799       58142 :                 pexps = list_merge(pexps, te, (fdup)NULL);
    5800             :         }
    5801      376523 :         if (rel && is_groupby(rel->op) && rel->flag) {
    5802          34 :                 list *gbe = rel->r;
    5803          34 :                 if (!list_empty(gbe)) {
    5804          68 :                         for (node *n=gbe->h; n; n = n->next) {
    5805          37 :                                 sql_exp *e = n->data;
    5806          37 :                                 if (rel->flag == 1 && is_atom(e->type) && !e->alias.name) {
    5807           1 :                                         atom *a = e->l;
    5808           1 :                                         int nr = (int)atom_get_int(a);
    5809           1 :                                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: GROUP BY position %d is not in select list", nr);
    5810             :                                 }
    5811          36 :                                 if (exp_has_aggr(rel, e))
    5812           2 :                                         return sql_error(sql, 02, SQLSTATE(42000) "SELECT: aggregate functions are not allowed in GROUP BY");
    5813             :                         }
    5814             :                 }
    5815             :         }
    5816      376520 :         if (rel && is_groupby(rel->op) && (!sn->groupby || rel->flag) && !is_processed(rel)) {
    5817       43640 :                 for (node *n=pexps->h; n; n = n->next) {
    5818       22288 :                         sql_exp *ce = n->data;
    5819       22288 :                         if (rel->card < ce->card && !exp_is_aggr(rel, ce)) {
    5820           0 :                                 if (exp_name(ce) && !has_label(ce))
    5821           0 :                                         return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column '%s' in query results without an aggregate function", exp_name(ce));
    5822           0 :                                 return sql_error(sql, ERR_GROUPBY, SQLSTATE(42000) "SELECT: cannot use non GROUP BY column in query results without an aggregate function");
    5823             :                         }
    5824             :                 }
    5825       21352 :                 set_processed(rel);
    5826             :         }
    5827      376520 :         rel = rel_project(sql->sa, rel, pexps);
    5828             : 
    5829      376519 :         rel = rel_having_limits_nodes(query, rel, sn, ek, group_totals);
    5830      376519 :         return rel;
    5831             : }
    5832             : 
    5833             : static sql_rel*
    5834      130060 : rel_unique_names(mvc *sql, sql_rel *rel)
    5835             : {
    5836      130060 :         list *l;
    5837             : 
    5838      130060 :         if (!is_project(rel->op))
    5839             :                 return rel;
    5840      130060 :         l = sa_list(sql->sa);
    5841      980146 :         for (node *n = rel->exps->h; n; n = n->next) {
    5842      850086 :                 sql_exp *e = n->data;
    5843      850086 :                 const char *name = exp_name(e);
    5844             : 
    5845             :                 /* If there are two identical expression names, there will be ambiguity */
    5846      850086 :                 if (!name || exps_bind_column(l, name, NULL, NULL, 0))
    5847       59145 :                         exp_label(sql->sa, e, ++sql->label);
    5848      850086 :                 append(l,e);
    5849             :         }
    5850      130060 :         rel->exps = l;
    5851      130060 :         return rel;
    5852             : }
    5853             : 
    5854             : static sql_rel *
    5855      378055 : rel_query(sql_query *query, symbol *sq, exp_kind ek)
    5856             : {
    5857      378055 :         mvc *sql = query->sql;
    5858      378055 :         sql_rel *res = NULL;
    5859      378055 :         SelectNode *sn = NULL;
    5860             : 
    5861      378055 :         if (sq->token != SQL_SELECT)
    5862          22 :                 return table_ref(query, sq, 0, NULL);
    5863             : 
    5864             :         /* select ... into is currently not handled here ! */
    5865      378033 :         sn = (SelectNode *) sq;
    5866      378033 :         if (sn->into)
    5867             :                 return NULL;
    5868             : 
    5869      378033 :         if (ek.card != card_relation && sn->orderby)
    5870           2 :                 return sql_error(sql, 01, SQLSTATE(42000) "SELECT: ORDER BY only allowed on outermost SELECT");
    5871             : 
    5872      378031 :         if (sn->window) {
    5873          20 :                 dlist *wl = sn->window->data.lval;
    5874          51 :                 for (dnode *n = wl->h; n ; n = n->next) {
    5875          32 :                         dlist *wd = n->data.sym->data.lval;
    5876          32 :                         const char *name = wd->h->data.sval;
    5877          32 :                         dlist *wdef = wd->h->next->data.lval;
    5878          32 :                         if (frame_get_window_def(sql, name, NULL)) {
    5879           1 :                                 return sql_error(sql, 01, SQLSTATE(42000) "SELECT: Redefinition of window '%s'", name);
    5880          31 :                         } else if (!frame_push_window_def(sql, name, wdef)) {
    5881           0 :                                 return sql_error(sql, 02, SQLSTATE(HY013) MAL_MALLOC_FAIL);
    5882             :                         }
    5883             :                 }
    5884             :         }
    5885             : 
    5886      378030 :         if (sn->from) {
    5887      305097 :                 dlist *fl = sn->from->data.lval;
    5888      305097 :                 sql_rel *fnd = NULL;
    5889      305097 :                 list *refs = new_exp_list(sql->sa); /* Keep list of relation names in order to test for duplicates */
    5890             : 
    5891      758962 :                 for (dnode *n = fl->h; n ; n = n->next) {
    5892      454168 :                         int lateral = check_is_lateral(n->data.sym);
    5893             : 
    5894             :                         /* just used current expression */
    5895      454167 :                         if (lateral && res)
    5896          22 :                                 query_push_outer(query, res, sql_from);
    5897      454167 :                         fnd = table_ref(query, n->data.sym, lateral, refs);
    5898      454168 :                         if (lateral && res)
    5899          22 :                                 res = query_pop_outer(query);
    5900      454168 :                         if (!fnd)
    5901             :                                 break;
    5902      453865 :                         if (res) {
    5903      149059 :                                 res = rel_crossproduct(sql->sa, res, fnd, op_join);
    5904      149059 :                                 if (lateral)
    5905          22 :                                         set_dependent(res);
    5906             :                         } else {
    5907             :                                 res = fnd;
    5908             :                         }
    5909             :                 }
    5910      305097 :                 if (!fnd) {
    5911         303 :                         if (res)
    5912          12 :                                 rel_destroy(res);
    5913         303 :                         return NULL;
    5914             :                 }
    5915       72933 :         } else if (!query_has_outer(query) || !res) {/* only on top level query */
    5916       72933 :                 return rel_select_exp(query, NULL, sn, ek);
    5917             :         }
    5918             : 
    5919      304794 :         sql_rel *rel = NULL;
    5920      304794 :         if (res)
    5921      304794 :                 rel = rel_select_exp(query, res, sn, ek);
    5922      304794 :         if (!rel && res)
    5923        1041 :                 rel_destroy(res);
    5924             :         return rel;
    5925             : }
    5926             : 
    5927             : /* NOTE: does NOT "set" query but instead generate set ops (union, except, intersect) */
    5928             : static sql_rel *
    5929          23 : rel_setquery_corresponding(sql_query *query, sql_rel *l, sql_rel *r, dlist *cols, int op, int outer, bool n_ary_op)
    5930             : {
    5931          23 :         mvc *sql = query->sql;
    5932          23 :         const char *opname = op==SQL_EXCEPT?"EXCEPT":op==SQL_INTERSECT?"INTERSECT":outer?"OUTER UNION":"UNION";
    5933          23 :         list *lexps = sa_list(query->sql->sa), *rexps = sa_list(query->sql->sa);
    5934          23 :         if (!lexps || !rexps)
    5935             :                 return NULL;
    5936          23 :         assert(cols);
    5937          23 :         if (dlist_length(cols)) {
    5938          22 :                 for (dnode *dn = cols->h; dn; dn = dn ->next) {
    5939          13 :                         char *nm = dn->data.sym->data.lval->h->data.sval;
    5940          13 :                         sql_exp *ls, *rs;
    5941             : 
    5942          13 :                         if (!(ls = rel_bind_column(sql, l, nm, sql_where | sql_join, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    5943             :                                 return NULL;
    5944          13 :                         if (!(rs = rel_bind_column(sql, r, nm, sql_where | sql_join, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    5945             :                                 return NULL;
    5946          13 :                         if ((!outer && (!ls || !rs)) || (outer && !ls && !rs))
    5947           2 :                                 return sql_error(sql, 02, SQLSTATE(42000) "%s: tables '%s' and '%s' do not have a matching column '%s'", opname, rel_name(l)?rel_name(l):"", rel_name(r)?rel_name(r):"", nm);
    5948          11 :                         if (outer && !ls)
    5949           4 :                                 ls = exp_null(sql->sa, exp_subtype(rs));
    5950          11 :                         if (outer && !rs)
    5951           4 :                                 rs = exp_null(sql->sa, exp_subtype(ls));
    5952          11 :                         append(lexps, ls);
    5953          11 :                         append(rexps, rs);
    5954             :                 }
    5955             :         } else {
    5956          12 :                 int found = 0;
    5957          12 :                 list *exps = rel_projections(sql, l, NULL, 1, 0), *r_exps = rel_projections(sql, r, NULL, 1, 0);
    5958          12 :                 if (!exps || !r_exps)
    5959             :                         return NULL;
    5960             :                 /* find cols which exist on both sides */
    5961          35 :                 for (node *n = exps->h; n; n = n->next) {
    5962          23 :                         sql_exp *le = n->data;
    5963          23 :                         int multi = 0;
    5964          23 :                         const char *rname = exp_relname(le), *name = exp_name(le);
    5965          23 :                         sql_exp *re = exps_bind_column(r_exps, name, NULL, &multi, 0);
    5966          23 :                         if (re) {
    5967          13 :                                 if (multi)
    5968           0 :                                         return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "%s: common column name '%s' appears more than once in right table", opname, rname);
    5969          13 :                                 multi = 0;
    5970          13 :                                 le = exps_bind_column(exps, name, NULL, &multi, 0);
    5971          13 :                                 if (multi)
    5972           0 :                                         return sql_error(sql, ERR_AMBIGUOUS, SQLSTATE(42000) "%s: common column name '%s' appears more than once in left table", opname, rname);
    5973             : 
    5974          13 :                                 found = 1;
    5975          13 :                                 append(lexps, le);
    5976          13 :                                 append(rexps, re);
    5977          13 :                                 list_remove_data(r_exps, NULL, re);
    5978          10 :                         } else if (outer) {
    5979           3 :                                 append(lexps, le);
    5980           3 :                                 re = exp_null(sql->sa, exp_subtype(le));
    5981           3 :                                 append(rexps, re); /* nils */
    5982             :                         }
    5983             :                 }
    5984          12 :                 if (!found)
    5985           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "%s: no columns of tables '%s' and '%s' match", opname, rel_name(l)?rel_name(l):"", rel_name(r)?rel_name(r):"");
    5986          12 :                 if (outer) {
    5987           7 :                         for (node *n = r_exps->h; n; n = n->next) {
    5988           4 :                                 sql_exp *re = n->data, *le;
    5989           4 :                                 append(rexps, re);
    5990           4 :                                 le = exp_null(sql->sa, exp_subtype(re));
    5991           4 :                                 append(lexps, le); /* nils */
    5992             :                         }
    5993             :                 }
    5994             :         }
    5995          21 :         return n_ary_op ?
    5996          21 :                 rel_setop_n_ary_check_types(sql, l, r, lexps, rexps, (operator_type)op) :
    5997           2 :                 rel_setop_check_types(sql, l, r, lexps, rexps, (operator_type)op);
    5998             : }
    5999             : 
    6000             : static sql_rel *
    6001        2484 : rel_setquery_(sql_query *query, sql_rel *l, sql_rel *r, dlist *cols, int op, int outer)
    6002             : {
    6003        2484 :         mvc *sql = query->sql;
    6004        2484 :         sql_rel *rel;
    6005             : 
    6006        2484 :         if (outer && !cols)
    6007           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "UNION: OUTER needs to be combined with CORRESPONDING [ BY ( column list ) ]");
    6008        2484 :         if (!cols) {
    6009        2482 :                 list *ls, *rs;
    6010             : 
    6011        2482 :                 l = rel_unique_names(sql, l);
    6012        2482 :                 r = rel_unique_names(sql, r);
    6013        2482 :                 ls = rel_projections(sql, l, NULL, 0, 1);
    6014        2482 :                 rs = rel_projections(sql, r, NULL, 0, 1);
    6015        2482 :                 rel = rel_setop_check_types(sql, l, r, ls, rs, (operator_type)op);
    6016             :         } else {
    6017           2 :                 rel = rel_setquery_corresponding(query, l, r, cols, op, outer, false);
    6018             :         }
    6019        2484 :         if (rel) {
    6020        2481 :                 rel_setop_set_exps(sql, rel, rel_projections(sql, rel, NULL, 0, 1), false);
    6021        2481 :                 set_processed(rel);
    6022             :         }
    6023             :         return rel;
    6024             : }
    6025             : 
    6026             : /* Generate n-ary set operator */
    6027             : static sql_rel *
    6028       62569 : rel_setquery_n_ary_(sql_query *query, sql_rel *l, sql_rel *r, dlist *cols, int op, int outer)
    6029             : {
    6030             :         /* even though this is for a general n-ary operators in this phase of the query
    6031             :          * processing we gonna have only two operands (so technically it's binary). In
    6032             :          * general this op supports arbitrary number of operands.
    6033             :          */
    6034             :         // TODO: for now we support only multi-union
    6035       62569 :         assert(op == op_munion);
    6036             : 
    6037       62569 :         mvc *sql = query->sql;
    6038       62569 :         sql_rel *rel;
    6039             : 
    6040       62569 :         if (outer && !cols)
    6041           0 :                         return sql_error(sql, 02, SQLSTATE(42000) "UNION: OUTER needs to be combined with CORRESPONDING [ BY ( column list ) ]");
    6042       62569 :         if (!cols) {
    6043             :                 // TODO: make rel_setop_n_ary_check_types to accept a list of rels
    6044             :                 // and a list of lists of exps
    6045       62548 :                 list *ls, *rs;
    6046             : 
    6047       62548 :                 l = rel_unique_names(sql, l);
    6048       62548 :                 r = rel_unique_names(sql, r);
    6049       62548 :                 ls = rel_projections(sql, l, NULL, 0, 1);
    6050       62548 :                 rs = rel_projections(sql, r, NULL, 0, 1);
    6051       62548 :                 rel = rel_setop_n_ary_check_types(sql, l, r, ls, rs, (operator_type)op);
    6052             :         } else {
    6053          21 :                 rel = rel_setquery_corresponding(query, l, r, cols, op, outer, true);
    6054             :         }
    6055             : 
    6056       62569 :         if (rel) {
    6057       62563 :                 rel_setop_n_ary_set_exps(sql, rel, rel_projections(sql, rel, NULL, 0, 1), false);
    6058       62563 :                 set_processed(rel);
    6059             :         }
    6060             : 
    6061             :         return rel;
    6062             : 
    6063             : }
    6064             : 
    6065             : static sql_rel *
    6066       65066 : rel_setquery(sql_query *query, symbol *q)
    6067             : {
    6068       65066 :         mvc *sql = query->sql;
    6069       65066 :         sql_rel *res = NULL;
    6070       65066 :         dnode *n = q->data.lval->h;
    6071       65066 :         symbol *tab_ref1 = n->data.sym;
    6072       65066 :         int distinct = n->next->data.i_val;
    6073       65066 :         dlist *corresponding = n->next->next->data.lval;
    6074       65066 :         symbol *tab_ref2 = n->next->next->next->data.sym;
    6075       65066 :         sql_rel *t1, *t2;
    6076             : 
    6077       65066 :         assert(n->next->type == type_int);
    6078       65066 :         t1 = table_ref(query, tab_ref1, 0, NULL);
    6079       65066 :         if (!t1)
    6080             :                 return NULL;
    6081       65062 :         t2 = table_ref(query, tab_ref2, 0, NULL);
    6082       65062 :         if (!t2)
    6083             :                 return NULL;
    6084             : 
    6085       65054 :         rel_remove_internal_exp(t1);
    6086       65054 :         rel_remove_internal_exp(t2);
    6087       65054 :         if (!corresponding && list_length(t1->exps) != list_length(t2->exps)) {
    6088           1 :                 int t1nrcols = list_length(t1->exps);
    6089           1 :                 int t2nrcols = list_length(t2->exps);
    6090           1 :                 const char *op = "UNION";
    6091           1 :                 if (q->token == SQL_EXCEPT)
    6092             :                         op = "EXCEPT";
    6093           0 :                 else if (q->token == SQL_INTERSECT)
    6094           0 :                         op = "INTERSECT";
    6095           1 :                 rel_destroy(t1);
    6096           1 :                 rel_destroy(t2);
    6097           1 :                 return sql_error(sql, 02, SQLSTATE(42000) "%s: column counts (%d and %d) do not match", op, t1nrcols, t2nrcols);
    6098             :         }
    6099       65053 :         if ( q->token == SQL_UNION) {
    6100       62569 :                 int outer = n->next->next->next->next->data.i_val;
    6101             :                 /* For EXCEPT/INTERSECT the group by is always done within the implementation */
    6102             :                 /* TODO add those later in an optimizer ! */
    6103       62569 :                 if (t1 && distinct)
    6104        3505 :                         t1 = rel_distinct(t1);
    6105       62569 :                 if (t2 && distinct)
    6106        3505 :                         t2 = rel_distinct(t2);
    6107             :                 // TODO: this has to be fixed
    6108             :                 /*res = rel_setquery_(query, t1, t2, corresponding, op_union, outer);*/
    6109       62569 :                 res = rel_setquery_n_ary_(query, t1, t2, corresponding, op_munion, outer);
    6110        2484 :         } else if ( q->token == SQL_EXCEPT)
    6111        2026 :                 res = rel_setquery_(query, t1, t2, corresponding, op_except, 0);
    6112         458 :         else if ( q->token == SQL_INTERSECT)
    6113         458 :                 res = rel_setquery_(query, t1, t2, corresponding, op_inter, 0);
    6114       65053 :         if (res) {
    6115       65044 :                 set_processed(res);
    6116       65044 :                 if (distinct)
    6117        5863 :                         res = rel_distinct(res);
    6118             :         }
    6119             :         return res;
    6120             : }
    6121             : 
    6122             : static sql_rel *
    6123       60032 : rel_joinquery_(sql_query *query, symbol *tab1, int natural, jt jointype, symbol *tab2, symbol *js, list *refs)
    6124             : {
    6125       60032 :         mvc *sql = query->sql;
    6126       60032 :         operator_type op = op_join;
    6127       60032 :         sql_rel *t1 = NULL, *t2 = NULL, *inner, *rel = NULL;
    6128       60032 :         int l_nil = 0, r_nil = 0, lateral = 0;
    6129             : 
    6130       60032 :         switch(jointype) {
    6131             :         case jt_inner:
    6132             :         case jt_cross: op = op_join;
    6133             :                 break;
    6134             :         case jt_left: op = op_left;
    6135             :                 r_nil = 1;
    6136             :                 break;
    6137             :         case jt_right: op = op_right;
    6138             :                 l_nil = 1;
    6139             :                 break;
    6140             :         case jt_full: op = op_full;
    6141             :                 l_nil = 1;
    6142             :                 r_nil = 1;
    6143             :                 break;
    6144             :         }
    6145             : 
    6146             :         /* a dependent join cannot depend on the right side, so disable lateral check for right and full joins */
    6147       60032 :         lateral = (op == op_join || op == op_left) && check_is_lateral(tab2);
    6148       60032 :         t1 = table_ref(query, tab1, 0, refs);
    6149       60032 :         if (t1) {
    6150       60027 :                 if (!lateral) {
    6151       59992 :                         t2 = table_ref(query, tab2, 0, refs);
    6152             :                 } else {
    6153          35 :                         query_processed(query);
    6154          35 :                         query_push_outer(query, t1, sql_from);
    6155          35 :                         t2 = table_ref(query, tab2, 0, refs);
    6156          35 :                         t1 = query_pop_outer(query);
    6157             :                 }
    6158             :         }
    6159       60032 :         if (!t1 || !t2)
    6160             :                 return NULL;
    6161             : 
    6162       60011 :         query_processed(query);
    6163       60011 :         if (strcmp(rel_name(t1), rel_name(t2)) == 0) {
    6164           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: ERROR:  table name '%s' specified more than once", rel_name(t1));
    6165             :         }
    6166       60011 :         inner = rel = rel_crossproduct(sql->sa, t1, t2, op);
    6167       60011 :         if (!rel)
    6168             :                 return NULL;
    6169       60011 :         if (lateral)
    6170          34 :                 set_dependent(rel);
    6171             : 
    6172       60011 :         assert(jointype != jt_cross || (!natural && !js)); /* there are no natural cross joins, or cross joins with conditions */
    6173       60011 :         if (js && natural)
    6174           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: cannot have a NATURAL JOIN with a join specification (ON or USING)");
    6175       60011 :         if (jointype != jt_cross && !js && !natural)
    6176           0 :                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: must have NATURAL JOIN or a JOIN with a join specification (ON or USING)");
    6177             : 
    6178       60011 :         if (js && js->token != SQL_USING) {  /* On sql_logical_exp */
    6179       59834 :                 rel = rel_logical_exp(query, rel, js, sql_where | sql_join);
    6180         177 :         } else if (js) {        /* using */
    6181          24 :                 char rname[16], *rnme;
    6182          24 :                 dnode *n = js->data.lval->h;
    6183          24 :                 list *outexps = new_exp_list(sql->sa), *exps;
    6184          24 :                 node *m;
    6185             : 
    6186          24 :                 rnme = sa_strdup(sql->sa, number2name(rname, sizeof(rname), ++sql->label));
    6187          54 :                 for (; n; n = n->next) {
    6188          30 :                         char *nm = n->data.sval;
    6189          30 :                         sql_exp *cond, *ls, *rs;
    6190             : 
    6191          30 :                         if (!(ls = rel_bind_column(sql, t1, nm, sql_where | sql_join, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    6192           0 :                                 return NULL;
    6193          30 :                         if (!(rs = rel_bind_column(sql, t2, nm, sql_where | sql_join, 0)) && sql->session->status == -ERR_AMBIGUOUS)
    6194             :                                 return NULL;
    6195          30 :                         if (!ls || !rs)
    6196           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "JOIN: tables '%s' and '%s' do not have a matching column '%s'", rel_name(t1)?rel_name(t1):"", rel_name(t2)?rel_name(t2):"", nm);
    6197          30 :                         if (!(rel = rel_compare_exp(query, rel, ls, rs, "=", TRUE, 0, 0, 0, 0)))
    6198             :                                 return NULL;
    6199          30 :                         if (op != op_join) {
    6200          12 :                                 if (!(cond = rel_unop_(sql, rel, ls, "sys", "isnull", card_value)))
    6201             :                                         return NULL;
    6202          12 :                                 set_has_no_nil(cond);
    6203          12 :                                 if (rel_convert_types(sql, t1, t2, &ls, &rs, 1, type_equal) < 0)
    6204             :                                         return NULL;
    6205          12 :                                 if (!(ls = rel_nop_(sql, rel, cond, rs, ls, NULL, "sys", "ifthenelse", card_value)))
    6206             :                                         return NULL;
    6207             :                         }
    6208          30 :                         exp_setname(sql, ls, rnme, nm);
    6209          30 :                         append(outexps, ls);
    6210          30 :                         if (!rel)
    6211             :                                 return NULL;
    6212             :                 }
    6213          24 :                 exps = rel_projections(sql, t1, NULL, 1, 1);
    6214         207 :                 for (m = exps->h; m; m = m->next) {
    6215         183 :                         const char *nm = exp_name(m->data);
    6216         183 :                         int fnd = 0;
    6217             : 
    6218         433 :                         for (n = js->data.lval->h; n; n = n->next) {
    6219         280 :                                 if (strcmp(nm, n->data.sval) == 0) {
    6220             :                                         fnd = 1;
    6221             :                                         break;
    6222             :                                 }
    6223             :                         }
    6224         183 :                         if (!fnd) {
    6225         153 :                                 sql_exp *ls = m->data;
    6226         153 :                                 if (l_nil)
    6227          18 :                                         set_has_nil(ls);
    6228         153 :                                 set_not_unique(ls);
    6229         153 :                                 append(outexps, ls);
    6230             :                         }
    6231             :                 }
    6232          24 :                 exps = rel_projections(sql, t2, NULL, 1, 1);
    6233         108 :                 for (m = exps->h; m; m = m->next) {
    6234          84 :                         const char *nm = exp_name(m->data);
    6235          84 :                         int fnd = 0;
    6236             : 
    6237         234 :                         for (n = js->data.lval->h; n; n = n->next) {
    6238         180 :                                 if (strcmp(nm, n->data.sval) == 0) {
    6239             :                                         fnd = 1;
    6240             :                                         break;
    6241             :                                 }
    6242             :                         }
    6243          84 :                         if (!fnd) {
    6244          54 :                                 sql_exp *rs = m->data;
    6245          54 :                                 if (r_nil)
    6246          14 :                                         set_has_nil(rs);
    6247          54 :                                 set_not_unique(rs);
    6248          54 :                                 append(outexps, rs);
    6249             :                         }
    6250             :                 }
    6251          24 :                 rel = rel_project(sql->sa, rel, outexps);
    6252         153 :         } else if (jointype != jt_cross) {              /* ! js -> natural join */
    6253          62 :                 rel = join_on_column_name(query, rel, t1, t2, op, l_nil, r_nil);
    6254             :         }
    6255       60011 :         if (!rel)
    6256             :                 return NULL;
    6257       59993 :         if (inner && is_outerjoin(inner->op))
    6258       21103 :                 set_processed(inner);
    6259       59993 :         set_processed(rel);
    6260       59993 :         query_processed(query);
    6261       59993 :         return rel;
    6262             : }
    6263             : 
    6264             : static sql_rel *
    6265       60032 : rel_joinquery(sql_query *query, symbol *q, list *refs)
    6266             : {
    6267       60032 :         dnode *n = q->data.lval->h;
    6268       60032 :         symbol *tab_ref1 = n->data.sym;
    6269       60032 :         int natural = n->next->data.i_val;
    6270       60032 :         jt jointype = (jt) n->next->next->data.i_val;
    6271       60032 :         symbol *tab_ref2 = n->next->next->next->data.sym;
    6272       60032 :         symbol *joinspec = n->next->next->next->next->data.sym;
    6273             : 
    6274       60032 :         assert(n->next->type == type_int);
    6275       60032 :         assert(n->next->next->type == type_int);
    6276       60032 :         return rel_joinquery_(query, tab_ref1, natural, jointype, tab_ref2, joinspec, refs);
    6277             : }
    6278             : 
    6279             : sql_rel *
    6280      378055 : rel_subquery(sql_query *query, symbol *sq, exp_kind ek)
    6281             : {
    6282      378055 :         mvc *sql = query->sql;
    6283             : 
    6284      378055 :         query_processed(query);
    6285      378055 :         if (!stack_push_frame(sql, NULL))
    6286           0 :                 return sql_error(sql, 02, SQLSTATE(HY013) MAL_MALLOC_FAIL);
    6287      378055 :         sql_rel *rel = rel_query(query, sq, ek);
    6288      378055 :         stack_pop_frame(sql);
    6289      378054 :         if (!query_has_outer(query))
    6290      339078 :                 query_processed(query);
    6291      378054 :         if (rel && ek.type == type_relation && ek.card < card_set && rel->card >= CARD_AGGR)
    6292          15 :                 return rel_zero_or_one(sql, rel, ek);
    6293             :         return rel;
    6294             : }
    6295             : 
    6296             : sql_rel *
    6297      238824 : rel_selects(sql_query *query, symbol *s)
    6298             : {
    6299      238824 :         mvc *sql = query->sql;
    6300      238824 :         sql_rel *ret = NULL;
    6301             : 
    6302      238824 :         switch (s->token) {
    6303        4523 :         case SQL_WITH:
    6304        4523 :                 ret = rel_with_query(query, s);
    6305        4523 :                 sql->type = Q_TABLE;
    6306        4523 :                 break;
    6307        2254 :         case SQL_VALUES:
    6308        2254 :                 ret = rel_values(query, s, NULL);
    6309        2254 :                 sql->type = Q_TABLE;
    6310        2254 :                 break;
    6311      210904 :         case SQL_SELECT: {
    6312      210904 :                 exp_kind ek = {type_value, card_relation, TRUE};
    6313      210904 :                 SelectNode *sn = (SelectNode *) s;
    6314             : 
    6315      210904 :                 if (sn->into) {
    6316           1 :                         sql->type = Q_SCHEMA;
    6317           1 :                         ret = rel_select_with_into(query, s);
    6318             :                 } else {
    6319      210903 :                         ret = rel_subquery(query, s, ek);
    6320      210902 :                         sql->type = Q_TABLE;
    6321             :                 }
    6322      210903 :         }       break;
    6323           0 :         case SQL_JOIN:
    6324           0 :                 ret = rel_joinquery(query, s, NULL);
    6325           0 :                 sql->type = Q_TABLE;
    6326           0 :                 break;
    6327       21143 :         case SQL_UNION:
    6328             :         case SQL_EXCEPT:
    6329             :         case SQL_INTERSECT:
    6330       21143 :                 ret = rel_setquery(query, s);
    6331       21143 :                 sql->type = Q_TABLE;
    6332       21143 :                 break;
    6333             :         default:
    6334             :                 return NULL;
    6335             :         }
    6336      238823 :         if (!ret && sql->errstr[0] == 0)
    6337           0 :                 (void) sql_error(sql, 02, SQLSTATE(42000) "relational query without result");
    6338             :         return ret;
    6339             : }
    6340             : 
    6341             : sql_rel *
    6342       78540 : schema_selects(sql_query *query, sql_schema *schema, symbol *s)
    6343             : {
    6344       78540 :         mvc *sql = query->sql;
    6345       78540 :         sql_rel *res;
    6346       78540 :         sql_schema *os = cur_schema(sql);
    6347             : 
    6348       78540 :         sql->session->schema = schema;
    6349       78540 :         res = rel_selects(query, s);
    6350       78540 :         sql->session->schema = os;
    6351       78540 :         return res;
    6352             : }
    6353             : 
    6354             : sql_rel *
    6355          26 : rel_loader_function(sql_query *query, symbol* fcall, list *fexps, sql_subfunc **loader_function)
    6356             : {
    6357          26 :         mvc *sql = query->sql;
    6358          26 :         sql_rel *sq = NULL;
    6359          26 :         dnode *l = fcall->data.lval->h;
    6360          26 :         char *sname = qname_schema(l->data.lval);
    6361          26 :         char *fname = qname_schema_object(l->data.lval);
    6362             : 
    6363          26 :         list *tl = sa_list(sql->sa);
    6364          26 :         list *exps = sa_list(sql->sa);
    6365          26 :         if (l->next)
    6366          26 :                 l = l->next; /* skip distinct */
    6367          26 :         if (l->next) { /* table call with subquery */
    6368          26 :                 if (l->next->type == type_symbol || l->next->type == type_list) {
    6369          26 :                         int count = 0;
    6370          26 :                         symbol *subquery = NULL;
    6371          26 :                         dnode *n = NULL;
    6372             : 
    6373          26 :                         if (l->next->type == type_symbol)
    6374             :                                 n = l->next;
    6375             :                         else
    6376          26 :                                 n = l->next->data.lval?l->next->data.lval->h:NULL;
    6377             : 
    6378          70 :                         for (dnode *m = n; m; m = m->next) {
    6379          44 :                                 if (m->type == type_symbol && m->data.sym->token == SQL_SELECT)
    6380          44 :                                         subquery = m->data.sym;
    6381          44 :                                 count++;
    6382             :                         }
    6383          26 :                         if (subquery && count > 1)
    6384           0 :                                 return sql_error(sql, 02, SQLSTATE(42000) "SELECT: The input for the loader function '%s' must be either a single sub query, or a list of values", fname);
    6385             : 
    6386          26 :                         if (subquery) {
    6387           0 :                                 exp_kind ek = { type_value, card_relation, TRUE };
    6388           0 :                                 if (!(sq = rel_subquery(query, subquery, ek)))
    6389           0 :                                         return NULL;
    6390             :                         } else {
    6391          26 :                                 exp_kind ek = { type_value, card_column, TRUE };
    6392          26 :                                 list *exps = sa_list(sql->sa);
    6393          96 :                                 for ( ; n; n = n->next) {
    6394          44 :                                         sql_exp *e = rel_value_exp(query, NULL, n->data.sym, sql_sel | sql_from, ek);
    6395             : 
    6396          44 :                                         if (!e)
    6397           0 :                                                 return NULL;
    6398          44 :                                         append(exps, e);
    6399             :                                 }
    6400          26 :                                 sq = rel_project(sql->sa, NULL, exps);
    6401             :                         }
    6402             :                 }
    6403          26 :                 if (!sq)
    6404           0 :                         return sql_error(sql, ERR_NOTFOUND, SQLSTATE(42000) "SELECT: no such loader function %s%s%s'%s'", sname ? "'":"", sname ? sname : "", sname ? "'.":"", fname);
    6405          70 :                 for (node *en = sq->exps->h; en; en = en->next) {
    6406          44 :                         sql_exp *e = en->data;
    6407             : 
    6408          44 :                         append(exps, e = exp_ref(sql, e));
    6409          44 :                         append(tl, exp_subtype(e));
    6410             :                 }
    6411             :         }
    6412             : 
    6413          26 :         sql_exp *e = NULL;
    6414          26 :         if (!(e = find_table_function(sql, sname, fname, exps, tl, F_LOADER)))
    6415             :                 return NULL;
    6416          25 :         sql_subfunc *sf = e->f;
    6417          25 :         if (sq) {
    6418          69 :                 for (node *n = sq->exps->h, *m = sf->func->ops->h ; n && m ; n = n->next, m = m->next) {
    6419          44 :                         sql_exp *e = (sql_exp*) n->data;
    6420          44 :                         sql_arg *a = (sql_arg*) m->data;
    6421          44 :                         if (!exp_subtype(e) && rel_set_type_param(sql, &(a->type), sq, e, 0) < 0)
    6422             :                                 return NULL;
    6423             :                 }
    6424             :         }
    6425             : 
    6426          25 :         if (loader_function)
    6427          25 :                 *loader_function = sf;
    6428             : 
    6429          25 :         return rel_table_func(sql->sa, sq, e, fexps, (sq)?TABLE_FROM_RELATION:TABLE_PROD_FUNC);
    6430             : }

Generated by: LCOV version 1.14