/* Copyright (C) 2014 InfiniDB, Inc. Copyright (C) 2019 MariaDB Corporaton This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ /* * $Id: ha_calpont_execplan.cpp 9749 2013-08-15 04:00:39Z zzhu $ */ /** @file */ //#define DEBUG_WALK_COND #include #include #include #include #ifdef _MSC_VER #include #include #else #include #include #endif #include #include #include #include #include #include #include #include #include #include using namespace std; #include #include #include #include #include "errorids.h" using namespace logging; #include "idb_mysql.h" #include "ha_calpont_impl_if.h" #include "ha_subquery.h" //#include "ha_view.h" using namespace cal_impl_if; #include "calpontselectexecutionplan.h" #include "calpontsystemcatalog.h" #include "simplecolumn_int.h" #include "simplecolumn_uint.h" #include "simplecolumn_decimal.h" #include "aggregatecolumn.h" #include "constantcolumn.h" #include "simplefilter.h" #include "constantfilter.h" #include "functioncolumn.h" #include "arithmeticcolumn.h" #include "arithmeticoperator.h" #include "logicoperator.h" #include "predicateoperator.h" #include "rowcolumn.h" #include "selectfilter.h" #include "existsfilter.h" #include "groupconcatcolumn.h" #include "outerjoinonfilter.h" #include "intervalcolumn.h" #include "udafcolumn.h" using namespace execplan; #include "funcexp.h" #include "functor.h" using namespace funcexp; const uint64_t AGG_BIT = 0x01; const uint64_t SUB_BIT = 0x02; const uint64_t AF_BIT = 0x04; const uint64_t CORRELATED = 0x08; // In certain cases, gp_walk is called recursively. When done so, // we need to bookmark the rcWorkStack for those cases where a constant // expression such as 1=1 is used in an if statement or function call. // This is a seriously bad kludge for MariaDB bug 750. // // BM => BookMark // HWM => HighWaterMark class RecursionCounter { private: RecursionCounter() {} public: RecursionCounter(gp_walk_info* gwip) : fgwip(gwip) { ++fgwip->recursionLevel; if (fgwip->recursionLevel > fgwip->recursionHWM) { fgwip->rcBookMarkStack.push(fgwip->rcWorkStack.size()); fgwip->recursionHWM = fgwip->recursionLevel; } } ~RecursionCounter() { --fgwip->recursionLevel; if (fgwip->recursionLevel < fgwip->recursionHWM - 1) { fgwip->rcBookMarkStack.pop(); --fgwip->recursionHWM; } } gp_walk_info* fgwip; }; //#define OUTER_JOIN_DEBUG namespace { string lower(string str) { boost::algorithm::to_lower(str); return str; } } #include "ha_view.h" namespace cal_impl_if { //@bug5228. need to escape backtick ` string escapeBackTick(const char* str) { if (!str) return ""; string ret; for (uint32_t i = 0; str[i] != 0; i++) { if (str[i] == '`') ret.append("``"); else ret.append(1, str[i]); } return ret; } void clearStacks(gp_walk_info& gwi) { while (!gwi.rcWorkStack.empty()) gwi.rcWorkStack.pop(); while (!gwi.ptWorkStack.empty()) gwi.ptWorkStack.pop(); } bool nonConstFunc(Item_func* ifp) { if (strcasecmp(ifp->func_name(), "rand") == 0 || strcasecmp(ifp->func_name(), "sysdate") == 0 || strcasecmp(ifp->func_name(), "idblocalpm") == 0) return true; for (uint32_t i = 0; i < ifp->argument_count(); i++) { if (ifp->arguments()[i]->type() == Item::FUNC_ITEM && nonConstFunc(((Item_func*)ifp->arguments()[i]))) return true; } return false; } /*@brief buildAggFrmTempField- build aggr func from extSELECT list item*/ /*********************************************************** * DESCRIPTION: * Server adds additional aggregation items to extended SELECT list and * references them in projection and HAVING. This f() finds * corresponding item in extSelAggColsItems and builds * ReturnedColumn using the item. * PARAMETERS: * item Item* used to build aggregation * gwi main structure * RETURNS * ReturnedColumn* if corresponding Item has been found * NULL otherwise ***********************************************************/ ReturnedColumn* buildAggFrmTempField(Item* item, gp_walk_info& gwi) { ReturnedColumn* result = NULL; Item_field* ifip = NULL; Item_ref* irip; Item_func_or_sum* isfp; switch ( item->type() ) { case Item::FIELD_ITEM: ifip = reinterpret_cast(item); break; default: irip = reinterpret_cast(item); if ( irip ) ifip = reinterpret_cast(irip->ref[0]); break; } if (ifip && ifip->field) { std::vector::iterator iter = gwi.extSelAggColsItems.begin(); for ( ; iter != gwi.extSelAggColsItems.end(); iter++ ) { //Item* temp_isfp = *iter; isfp = reinterpret_cast(*iter); if ( isfp->type() == Item::SUM_FUNC_ITEM && isfp->result_field == ifip->field ) { ReturnedColumn* rc = buildAggregateColumn(isfp, gwi); if (rc) result = rc; break; } } } return result; } string getViewName(TABLE_LIST* table_ptr) { string viewName = ""; if (!table_ptr) return viewName; TABLE_LIST* view = table_ptr->referencing_view; if (view) { if (!view->derived) viewName = view->alias.str; while ((view = view->referencing_view)) { if (view->derived) continue; viewName = view->alias.str + string(".") + viewName; } } return viewName; } #ifdef DEBUG_WALK_COND void debug_walk(const Item* item, void* arg) { switch (item->type()) { case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)item; cerr << "FIELD_ITEM: " << (ifp->db_name ? ifp->db_name : "") << '.' << bestTableName(ifp) << '.' << ifp->field_name.str << endl; break; } case Item::INT_ITEM: { Item_int* iip = (Item_int*)item; cerr << "INT_ITEM: "; if (iip->name.length) cerr << iip->name.str << " (from name string)" << endl; else cerr << iip->val_int() << endl; break; } case Item::STRING_ITEM: { Item_string* isp = (Item_string*)item; String val, *str = isp->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); cerr << "STRING_ITEM: >" << valStr << '<' << endl; break; } case Item::REAL_ITEM: { cerr << "REAL_ITEM" << endl; break; } case Item::DECIMAL_ITEM: { cerr << "DECIMAL_ITEM" << endl; break; } case Item::FUNC_ITEM: { Item_func* ifp = (Item_func*)item; Item_func_opt_neg* inp; cerr << "FUNC_ITEM: "; switch (ifp->functype()) { case Item_func::UNKNOWN_FUNC: // 0 cerr << ifp->func_name() << " (" << ifp->functype() << ")" << endl; break; case Item_func::GT_FUNC: // 7 cerr << '>' << " (" << ifp->functype() << ")" << endl; break; case Item_func::EQ_FUNC: // 1 cerr << '=' << " (" << ifp->functype() << ")" << endl; break; case Item_func::GE_FUNC: cerr << ">=" << " (" << ifp->functype() << ")" << endl; break; case Item_func::LE_FUNC: cerr << "<=" << " (" << ifp->functype() << ")" << endl; break; case Item_func::LT_FUNC: cerr << '<' << " (" << ifp->functype() << ")" << endl; break; case Item_func::NE_FUNC: cerr << "<>" << " (" << ifp->functype() << ")" << endl; break; case Item_func::NEG_FUNC: // 45 cerr << "unary minus" << " (" << ifp->functype() << ")" << endl; break; case Item_func::IN_FUNC: // 16 inp = (Item_func_opt_neg*)ifp; if (inp->negated) cerr << "not "; cerr << "in" << " (" << ifp->functype() << ")" << endl; break; case Item_func::BETWEEN: inp = (Item_func_opt_neg*)ifp; if (inp->negated) cerr << "not "; cerr << "between" << " (" << ifp->functype() << ")" << endl; break; case Item_func::ISNULL_FUNC: // 10 cerr << "is null" << " (" << ifp->functype() << ")" << endl; break; case Item_func::ISNOTNULL_FUNC: // 11 cerr << "is not null" << " (" << ifp->functype() << ")" << endl; break; case Item_func::NOT_ALL_FUNC: cerr << "not_all" << " (" << ifp->functype() << ")" << endl; break; case Item_func::NOT_FUNC: cerr << "not_func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::TRIG_COND_FUNC: cerr << "trig_cond_func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::ISNOTNULLTEST_FUNC: cerr << "isnotnulltest_func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::MULT_EQUAL_FUNC: { cerr << "mult_equal_func:" << " (" << ifp->functype() << ")" << endl; Item_equal* item_eq = (Item_equal*)ifp; Item_equal_fields_iterator it(*item_eq); Item* item; while ((item = it++)) { Field* equal_field = it.get_curr_field(); cerr << equal_field->field_name.str << endl; } break; } case Item_func::EQUAL_FUNC: cerr << "equal func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::FT_FUNC: cerr << "ft func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::LIKE_FUNC: cerr << "like func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::COND_AND_FUNC: cerr << "cond and func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::COND_OR_FUNC: cerr << "cond or func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::XOR_FUNC: cerr << "xor func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::INTERVAL_FUNC: cerr << "interval func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_EQUALS_FUNC: cerr << "sp equals func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_DISJOINT_FUNC: cerr << "sp disjoint func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_INTERSECTS_FUNC: cerr << "sp intersects func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_TOUCHES_FUNC: cerr << "sp touches func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_CROSSES_FUNC: cerr << "sp crosses func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_WITHIN_FUNC: cerr << "sp within func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_CONTAINS_FUNC: cerr << "sp contains func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_OVERLAPS_FUNC: cerr << "sp overlaps func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_STARTPOINT: cerr << "sp startpoint func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_ENDPOINT: cerr << "sp endpoint func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_EXTERIORRING: cerr << "sp exteriorring func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_POINTN: cerr << "sp pointn func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_GEOMETRYN: cerr << "sp geometryn func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_INTERIORRINGN: cerr << "sp exteriorringn func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SP_RELATE_FUNC: cerr << "sp relate func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::NOW_FUNC: cerr << "now func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::SUSERVAR_FUNC: cerr << "suservar func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::GUSERVAR_FUNC: cerr << "guservar func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::COLLATE_FUNC: cerr << "collate func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::EXTRACT_FUNC: cerr << "extract func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::CHAR_TYPECAST_FUNC: cerr << "char typecast func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::FUNC_SP: cerr << "func sp func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::UDF_FUNC: cerr << "udf func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::GSYSVAR_FUNC: cerr << "gsysvar func" << " (" << ifp->functype() << ")" << endl; break; case Item_func::DYNCOL_FUNC: cerr << "dyncol func" << " (" << ifp->functype() << ")" << endl; break; default: cerr << "type=" << ifp->functype() << endl; break; } break; } case Item::COND_ITEM: { Item_cond* icp = (Item_cond*)item; cerr << "COND_ITEM: " << icp->func_name() << endl; break; } case Item::SUM_FUNC_ITEM: { Item_sum* isp = (Item_sum*)item; char* item_name = const_cast(item->name.str); // MCOL-1052 This is an extended SELECT list item if (!item_name && isp->get_arg_count() && isp->get_arg(0)->name.length) { item_name = const_cast(isp->get_arg(0)->name.str); } else if (!item_name && isp->get_arg_count() && isp->get_arg(0)->type() == Item::INT_ITEM) { item_name = (char*)"INT||*"; } else if (!item_name) { item_name = (char*)""; } switch (isp->sum_func()) { case Item_sum::SUM_FUNC: cerr << "SUM_FUNC: " << item_name << endl; break; case Item_sum::SUM_DISTINCT_FUNC: cerr << "SUM_DISTINCT_FUNC: " << item_name << endl; break; case Item_sum::AVG_FUNC: cerr << "AVG_FUNC: " << item_name << endl; break; case Item_sum::COUNT_FUNC: cerr << "COUNT_FUNC: " << item_name << endl; break; case Item_sum::COUNT_DISTINCT_FUNC: cerr << "COUNT_DISTINCT_FUNC: " << item_name << endl; break; case Item_sum::MIN_FUNC: cerr << "MIN_FUNC: " << item_name << endl; break; case Item_sum::MAX_FUNC: cerr << "MAX_FUNC: " << item_name << endl; break; case Item_sum::UDF_SUM_FUNC: cerr << "UDAF_FUNC: " << item_name << endl; break; default: cerr << "SUM_FUNC_ITEM type=" << isp->sum_func() << endl; break; } break; } case Item::SUBSELECT_ITEM: { Item_subselect* sub = (Item_subselect*)item; cerr << "SUBSELECT Item: "; switch (sub->substype()) { case Item_subselect::EXISTS_SUBS: cerr << "EXISTS"; break; case Item_subselect::IN_SUBS: cerr << "IN"; break; default: cerr << sub->substype(); break; } cerr << endl; JOIN* join = sub->get_select_lex()->join; if (join) { Item_cond* cond = reinterpret_cast(join->conds); if (cond) cond->traverse_cond(debug_walk, arg, Item::POSTFIX); } cerr << "Finish subselect item traversing" << endl; break; } case Item::REF_ITEM: { Item_ref* ref = (Item_ref*)item; if (ref->real_item()->type() == Item::CACHE_ITEM) { Item* field = ((Item_cache*)ref->real_item())->get_example(); if (field->type() == Item::FIELD_ITEM) { Item_field* ifp = (Item_field*)field; //ifp->cached_table->select_lex->select_number gives the select level. // could be used on alias. // could also be used to tell correlated join (equal level). cerr << "CACHED REF FIELD_ITEM: " << ifp->db_name << '.' << bestTableName(ifp) << '.' << ifp->field_name.str << endl; break; } else if (field->type() == Item::FUNC_ITEM) { Item_func* ifp = (Item_func*)field; cerr << "CACHED REF FUNC_ITEM " << ifp->func_name() << endl; } else if (field->type() == Item::REF_ITEM) { Item_ref* ifr = (Item_ref*)field; string refType; string realType; switch (ifr->ref_type()) { case Item_ref::REF: refType = "REF"; break; case Item_ref::DIRECT_REF: refType = "DIRECT_REF"; break; case Item_ref::VIEW_REF: refType = "VIEW_REF"; break; case Item_ref::OUTER_REF: refType = "OUTER_REF"; break; case Item_ref::AGGREGATE_REF: refType = "AGGREGATE_REF"; break; default: refType = "UNKNOWN"; break; } switch (ifr->real_type()) { case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)(*(ifr->ref)); realType = "FIELD_ITEM "; realType += ifp->db_name; realType += '.'; realType += bestTableName(ifp); realType += '.'; realType += ifp->field_name.str; break; } case Item::SUM_FUNC_ITEM: { Item_sum* isp = (Item_sum*)(*(ifr->ref)); if (isp->sum_func() == Item_sum::GROUP_CONCAT_FUNC) realType = "GROUP_CONCAT_FUNC"; else realType = "SUM_FUNC_ITEM"; break; } case Item::REF_ITEM: // Need recursion here realType = "REF_ITEM"; break; case Item::FUNC_ITEM: { Item_func* ifp = (Item_func*)(*(ifr->ref)); realType = "FUNC_ITEM "; realType += ifp->func_name(); break; } default: { realType = "UNKNOWN"; } } cerr << "CACHED REF_ITEM: ref type " << refType.c_str() << " real type " << realType.c_str() << endl; break; } else { cerr << "REF_ITEM with CACHE_ITEM type unknown " << field->type() << endl; } } else if (ref->real_item()->type() == Item::FIELD_ITEM) { Item_field* ifp = (Item_field*)ref->real_item(); // MCOL-1052 The field referenced presumable came from // extended SELECT list. if ( !ifp->field_name.str ) { cerr << "REF extra FIELD_ITEM: " << ifp->name.str << endl; } else { cerr << "REF FIELD_ITEM: " << ifp->db_name << '.' << bestTableName(ifp) << '.' << ifp->field_name.str << endl; } break; } else if (ref->real_item()->type() == Item::FUNC_ITEM) { Item_func* ifp = (Item_func*)ref->real_item(); cerr << "REF FUNC_ITEM " << ifp->func_name() << endl; } else if (ref->real_item()->type() == Item::WINDOW_FUNC_ITEM) { Item_window_func* ifp = (Item_window_func*)ref->real_item(); cerr << "REF WINDOW_FUNC_ITEM " << ifp->window_func()->func_name() << endl; } else { cerr << "UNKNOWN REF ITEM type " << ref->real_item()->type() << endl; } break; } case Item::ROW_ITEM: { Item_row* row = (Item_row*)item; cerr << "ROW_ITEM: " << endl; for (uint32_t i = 0; i < row->cols(); i++) debug_walk(row->element_index(i), 0); break; } case Item::EXPR_CACHE_ITEM: { cerr << "Expr Cache Item" << endl; ((Item_cache_wrapper*)item)->get_orig_item()->traverse_cond(debug_walk, arg, Item::POSTFIX); break; } case Item::CACHE_ITEM: { Item_cache* isp = (Item_cache*)item; // MCOL-46 isp->val_str() can cause a call to execute a subquery. We're not set up // to execute yet. #if 0 switch (item->result_type()) { case STRING_RESULT: cerr << "CACHE_STRING_ITEM" << endl; break; case REAL_RESULT: cerr << "CACHE_REAL_ITEM " << isp->val_real() << endl; break; case INT_RESULT: cerr << "CACHE_INT_ITEM " << isp->val_int() << endl; break; case ROW_RESULT: cerr << "CACHE_ROW_ITEM" << endl; break; case DECIMAL_RESULT: cerr << "CACHE_DECIMAL_ITEM " << isp->val_decimal() << endl; break; default: cerr << "CACHE_UNKNOWN_ITEM" << endl; break; } #endif Item* field = isp->get_example(); if (field->type() == Item::FIELD_ITEM) { Item_field* ifp = (Item_field*)field; //ifp->cached_table->select_lex->select_number gives the select level. // could be used on alias. // could also be used to tell correlated join (equal level). cerr << "CACHED FIELD_ITEM: " << ifp->db_name << '.' << bestTableName(ifp) << '.' << ifp->field_name.str << endl; break; } else if (field->type() == Item::REF_ITEM) { Item_ref* ifr = (Item_ref*)field; string refType; string realType; switch (ifr->ref_type()) { case Item_ref::REF: refType = "REF"; break; case Item_ref::DIRECT_REF: refType = "DIRECT_REF"; break; case Item_ref::VIEW_REF: refType = "VIEW_REF"; break; case Item_ref::OUTER_REF: refType = "OUTER_REF"; break; case Item_ref::AGGREGATE_REF: refType = "AGGREGATE_REF"; break; default: refType = "UNKNOWN"; break; } switch (ifr->real_type()) { case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)(*(ifr->ref)); realType = "FIELD_ITEM "; realType += ifp->db_name; realType += '.'; realType += bestTableName(ifp); realType += '.'; realType += ifp->field_name.str; break; } case Item::SUM_FUNC_ITEM: { Item_sum* isp = (Item_sum*)(*(ifr->ref)); if (isp->sum_func() == Item_sum::GROUP_CONCAT_FUNC) realType = "GROUP_CONCAT_FUNC"; else realType = "SUM_FUNC_ITEM"; break; } case Item::REF_ITEM: // Need recursion here realType = "REF_ITEM"; break; case Item::FUNC_ITEM: { Item_func* ifp = (Item_func*)(*(ifr->ref)); realType = "FUNC_ITEM "; realType += ifp->func_name(); break; } default: { realType = "UNKNOWN"; } } cerr << "CACHE_ITEM ref type " << refType.c_str() << " real type " << realType.c_str() << endl; break; } else if (field->type() == Item::FUNC_ITEM) { Item_func* ifp = (Item_func*)field; cerr << "CACHE_ITEM FUNC_ITEM " << ifp->func_name() << endl; break; } else { cerr << "CACHE_ITEM type unknown " << field->type() << endl; } break; } case Item::DATE_ITEM: { String val, *str = NULL; Item_temporal_literal* itp = (Item_temporal_literal*)item; str = itp->val_str(&val); cerr << "DATE ITEM: "; if (str) cerr << ": (" << str->ptr() << ')' << endl; else cerr << ": " << endl; break; } case Item::WINDOW_FUNC_ITEM: { Item_window_func* ifp = (Item_window_func*)item; cerr << "Window Function Item " << ifp->window_func()->func_name() << endl; break; } case Item::NULL_ITEM: { cerr << "NULL item" << endl; break; } default: { cerr << "UNKNOWN_ITEM type " << item->type() << endl; break; } } } #endif void buildNestedTableOuterJoin(gp_walk_info& gwi, TABLE_LIST* table_ptr) { TABLE_LIST* table; List_iterator li(table_ptr->nested_join->join_list); while ((table = li++)) { gwi.innerTables.clear(); if (table->outer_join) { CalpontSystemCatalog::TableAliasName ta = make_aliasview( (table->db.length ? table->db.str : ""), (table->table_name.length ? table->table_name.str : ""), (table->alias.length ? table->alias.str : ""), getViewName(table)); gwi.innerTables.insert(ta); } if (table->nested_join) { TABLE_LIST* tab; List_iterator li(table->nested_join->join_list); while ((tab = li++)) { CalpontSystemCatalog::TableAliasName ta = make_aliasview( (tab->db.length ? tab->db.str : ""), (tab->table_name.length ? tab->table_name.str : ""), (tab->alias.length ? tab->alias.str : ""), getViewName(tab)); gwi.innerTables.insert(ta); } } if (table->on_expr) { Item_cond* expr = reinterpret_cast(table->on_expr); #ifdef DEBUG_WALK_COND expr->traverse_cond(debug_walk, &gwi, Item::POSTFIX); #endif expr->traverse_cond(gp_walk, &gwi, Item::POSTFIX); } if (table->nested_join && &(table->nested_join->join_list)) { buildNestedTableOuterJoin(gwi, table); } } } uint32_t buildOuterJoin(gp_walk_info& gwi, SELECT_LEX& select_lex) { // check non-collapsed outer join // this set contains all processed embedded joins. duplicate joins are ignored set embeddingSet; TABLE_LIST* table_ptr = select_lex.get_table_list(); gp_walk_info gwi_outer = gwi; gwi_outer.subQuery = NULL; gwi_outer.hasSubSelect = false; vector tmpVec; for (; table_ptr; table_ptr = table_ptr->next_local) { gwi_outer.innerTables.clear(); clearStacks(gwi_outer); gwi_outer.subQuery = NULL; gwi_outer.hasSubSelect = false; // View is already processed in view::transform // @bug5319. view is sometimes treated as derived table and // fromSub::transform does not build outer join filters. if (!table_ptr->derived && table_ptr->view) continue; CalpontSystemCatalog:: TableAliasName tan = make_aliasview( (table_ptr->db.length ? table_ptr->db.str : ""), (table_ptr->table_name.length ? table_ptr->table_name.str : ""), (table_ptr->alias.length ? table_ptr->alias.str : ""), getViewName(table_ptr)); if (table_ptr->outer_join && table_ptr->on_expr) { Item_cond* expr = reinterpret_cast(table_ptr->on_expr); gwi_outer.innerTables.insert(tan); if (table_ptr->nested_join && &(table_ptr->nested_join->join_list)) { TABLE_LIST* table; List_iterator li(table_ptr->nested_join->join_list); while ((table = li++)) { CalpontSystemCatalog::TableAliasName ta = make_aliasview( (table->db.length ? table->db.str : ""), (table->table_name.length ? table->table_name.str : ""), (table->alias.length ? table->alias.str : ""), getViewName(table)); gwi_outer.innerTables.insert(ta); } } #ifdef DEBUG_WALK_COND if (table_ptr->alias.length) cerr << table_ptr->alias.str; else if (table_ptr->alias.length) cerr << table_ptr->alias.str; cerr << " outer table expression: " << endl; expr->traverse_cond(debug_walk, &gwi_outer, Item::POSTFIX); #endif expr->traverse_cond(gp_walk, &gwi_outer, Item::POSTFIX); } else if (table_ptr->nested_join && &(table_ptr->nested_join->join_list)) { buildNestedTableOuterJoin(gwi_outer, table_ptr); } // this part is ambiguous. Not quite sure how MySQL's lay out the outer join filters in the structure else if (table_ptr->embedding && table_ptr->embedding->outer_join && table_ptr->embedding->on_expr) { // all the tables in nested_join are inner tables. TABLE_LIST* table; List_iterator li(table_ptr->embedding->nested_join->join_list); gwi_outer.innerTables.clear(); while ((table = li++)) { CalpontSystemCatalog:: TableAliasName ta = make_aliasview( (table->db.length ? table->db.str : ""), (table->table_name.length ? table->table_name.str : ""), (table->alias.length ? table->alias.str : ""), getViewName(table)); gwi_outer.innerTables.insert(ta); } if (embeddingSet.find(table_ptr->embedding) != embeddingSet.end()) continue; embeddingSet.insert(table_ptr->embedding); Item_cond* expr = reinterpret_cast(table_ptr->embedding->on_expr); #ifdef DEBUG_WALK_COND cerr << "inner tables: " << endl; set::const_iterator it; for (it = gwi_outer.innerTables.begin(); it != gwi_outer.innerTables.end(); ++it) cerr << (*it) << " "; cerr << endl; expr->traverse_cond(debug_walk, &gwi_outer, Item::POSTFIX); #endif expr->traverse_cond(gp_walk, &gwi_outer, Item::POSTFIX); } // @bug 2849 else if (table_ptr->embedding && table_ptr->embedding->nested_join) { // if this is dervied table process phase, mysql may have not developed the plan // completely. Return and let it finish. It will come to rnd_init again. if (table_ptr->embedding->is_natural_join && table_ptr->derived) { if (gwi.thd->derived_tables_processing) { gwi.thd->infinidb_vtable.isUnion = false; gwi.thd->infinidb_vtable.isUpdateWithDerive = true; return -1; } } if (embeddingSet.find(table_ptr->embedding) != embeddingSet.end()) continue; gwi_outer.innerTables.clear(); gwi_outer.innerTables.insert(tan); embeddingSet.insert(table_ptr->embedding); List* inners = &(table_ptr->embedding->nested_join->join_list); List_iterator_fast li(*inners); TABLE_LIST* curr; while ((curr = li++)) { if (curr->on_expr) { if (!curr->outer_join) // only handle nested JOIN for now { gwi_outer.innerTables.clear(); Item_cond* expr = reinterpret_cast(curr->on_expr); #ifdef DEBUG_WALK_COND expr->traverse_cond(debug_walk, &gwi_outer, Item::POSTFIX); #endif expr->traverse_cond(gp_walk, &gwi_outer, Item::POSTFIX); } } } } // Error out subquery in outer join on filter for now if (gwi_outer.hasSubSelect) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_OUTER_JOIN_SUBSELECT); setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText); return -1; } // build outerjoinon filter ParseTree* filters = NULL, *ptp = NULL, /**rhs = NULL*/*lhs = NULL; while (!gwi_outer.ptWorkStack.empty()) { filters = gwi_outer.ptWorkStack.top(); gwi_outer.ptWorkStack.pop(); if (gwi_outer.ptWorkStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); ptp->right(filters); lhs = gwi_outer.ptWorkStack.top(); gwi_outer.ptWorkStack.pop(); ptp->left(lhs); gwi_outer.ptWorkStack.push(ptp); } // should have only 1 pt left in stack. if (filters) { SPTP on_sp(filters); OuterJoinOnFilter* onFilter = new OuterJoinOnFilter(on_sp); ParseTree* pt = new ParseTree(onFilter); gwi.ptWorkStack.push(pt); } } embeddingSet.clear(); return 0; } ParseTree* buildRowPredicate(RowColumn* lhs, RowColumn* rhs, string predicateOp) { PredicateOperator* po = new PredicateOperator(predicateOp); boost::shared_ptr sop(po); LogicOperator* lo = NULL; if (predicateOp == "=") lo = new LogicOperator("and"); else lo = new LogicOperator("or"); ParseTree* pt = new ParseTree(lo); sop->setOpType(lhs->columnVec()[0]->resultType(), rhs->columnVec()[0]->resultType()); SimpleFilter* sf = new SimpleFilter(sop, lhs->columnVec()[0].get(), rhs->columnVec()[0].get()); pt->left(new ParseTree(sf)); for (uint32_t i = 1; i < lhs->columnVec().size(); i++) { sop.reset(po->clone()); sop->setOpType(lhs->columnVec()[i]->resultType(), rhs->columnVec()[i]->resultType()); SimpleFilter* sf = new SimpleFilter(sop, lhs->columnVec()[i].get(), rhs->columnVec()[i].get()); pt->right(new ParseTree(sf)); if (i + 1 < lhs->columnVec().size()) { ParseTree* lpt = pt; pt = new ParseTree(lo->clone()); pt->left(lpt); } } return pt; } bool buildRowColumnFilter(gp_walk_info* gwip, RowColumn* rhs, RowColumn* lhs, Item_func* ifp) { if (ifp->functype() == Item_func::EQ_FUNC || ifp->functype() == Item_func::NE_FUNC) { // (c1,c2,..) = (v1,v2,...) transform to: c1=v1 and c2=v2 and ... assert (!lhs->columnVec().empty() && lhs->columnVec().size() == rhs->columnVec().size()); gwip->ptWorkStack.push(buildRowPredicate(rhs, lhs, ifp->func_name())); } else if (ifp->functype() == Item_func::IN_FUNC) { // (c1,c2,...) in ((v11,v12,...),(v21,v22,...)...) transform to: // ((c1 = v11 and c2 = v12 and ...) or (c1 = v21 and c2 = v22 and ...) or ...) // and c1 in (v11,v21,...) and c2 in (v12,v22,...) => to favor CP Item_func_opt_neg* inp = (Item_func_opt_neg*)ifp; string predicateOp, logicOp; if (inp->negated) { predicateOp = "<>"; logicOp = "and"; } else { predicateOp = "="; logicOp = "or"; } boost::scoped_ptr lo(new LogicOperator(logicOp)); // 1st round. build the equivalent filters // two entries have been popped from the stack already: lhs and rhs stack tmpStack; vector valVec; tmpStack.push(rhs); tmpStack.push(lhs); assert (gwip->rcWorkStack.size() >= ifp->argument_count() - 2); for (uint32_t i = 2; i < ifp->argument_count(); i++) { tmpStack.push(gwip->rcWorkStack.top()); if (!gwip->rcWorkStack.empty()) gwip->rcWorkStack.pop(); } RowColumn* columns = dynamic_cast(tmpStack.top()); tmpStack.pop(); RowColumn* vals = dynamic_cast(tmpStack.top()); valVec.push_back(vals); tmpStack.pop(); ParseTree* pt = buildRowPredicate(columns, vals, predicateOp); while (!tmpStack.empty()) { ParseTree* pt1 = new ParseTree(lo->clone()); pt1->left(pt); vals = dynamic_cast(tmpStack.top()); valVec.push_back(vals); tmpStack.pop(); pt1->right(buildRowPredicate(columns->clone(), vals, predicateOp)); pt = pt1; } gwip->ptWorkStack.push(pt); // done for NOTIN clause if (predicateOp == "<>") return true; // 2nd round. add the filter to favor casual partition for IN clause boost::shared_ptr sop; boost::shared_ptr ssc; stack tmpPtStack; for (uint32_t i = 0; i < columns->columnVec().size(); i++) { ConstantFilter* cf = new ConstantFilter(); sop.reset(lo->clone()); cf->op(sop); SimpleColumn* sc = dynamic_cast(columns->columnVec()[i].get()); // no optimization for non-simple column because CP won't apply if (!sc) continue; ssc.reset(sc->clone()); cf->col(ssc); uint32_t j = 0; for (; j < valVec.size(); j++) { sop.reset(new PredicateOperator(predicateOp)); ConstantColumn* cc = dynamic_cast(valVec[j]->columnVec()[i].get()); // no optimization for non-constant value because CP won't apply if (!cc) break; sop->setOpType(sc->resultType(), valVec[j]->resultType()); cf->pushFilter(new SimpleFilter(sop, sc->clone(), valVec[j]->columnVec()[i]->clone())); } if (j < valVec.size()) continue; tmpPtStack.push(new ParseTree(cf)); } // "and" all the filters together ParseTree* ptp = NULL; pt = NULL; while (!tmpPtStack.empty()) { pt = tmpPtStack.top(); tmpPtStack.pop(); if (tmpPtStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); ptp->left(pt); pt = tmpPtStack.top(); tmpPtStack.pop(); ptp->right(pt); tmpPtStack.push(ptp); } if (pt) { tmpPtStack.push(pt); // AND with the pt built from the first round if (!gwip->ptWorkStack.empty() && !tmpPtStack.empty()) { pt = new ParseTree(new LogicOperator("and")); pt->left(gwip->ptWorkStack.top()); gwip->ptWorkStack.pop(); pt->right(tmpPtStack.top()); tmpPtStack.pop(); } // Push the final pt tree for this IN clause gwip->ptWorkStack.push(pt); } } else { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FUNC_MULTI_COL); return false; } return true; } bool buildPredicateItem(Item_func* ifp, gp_walk_info* gwip) { boost::shared_ptr sop(new PredicateOperator(ifp->func_name())); if (ifp->functype() == Item_func::LIKE_FUNC) { // Starting with MariaDB 10.2, LIKE uses a negated flag instead of FUNC_NOT // Further processing is done below as before for LIKE if (((Item_func_like*)ifp)->negated) { sop->reverseOp(); } } if (!(gwip->thd->infinidb_vtable.cal_conn_info)) gwip->thd->infinidb_vtable.cal_conn_info = (void*)(new cal_connection_info()); cal_connection_info* ci = reinterpret_cast(gwip->thd->infinidb_vtable.cal_conn_info); if (ifp->functype() == Item_func::BETWEEN) { idbassert (gwip->rcWorkStack.size() >= 3); ReturnedColumn* rhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); ReturnedColumn* lhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); ReturnedColumn* filterCol = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); // pop gwip->scsp; Item_func_opt_neg* inp = (Item_func_opt_neg*)ifp; SimpleFilter* sfr = 0; SimpleFilter* sfl = 0; if (inp->negated) { sop.reset(new PredicateOperator(">")); sop->setOpType(filterCol->resultType(), rhs->resultType()); sfr = new SimpleFilter(sop, filterCol, rhs); sop.reset(new PredicateOperator("<")); sop->setOpType(filterCol->resultType(), lhs->resultType()); sfl = new SimpleFilter(sop, filterCol->clone(), lhs); ParseTree* ptp = new ParseTree(new LogicOperator("or")); ptp->left(sfr); ptp->right(sfl); gwip->ptWorkStack.push(ptp); } else { sop.reset(new PredicateOperator("<=")); sop->setOpType(filterCol->resultType(), rhs->resultType()); sfr = new SimpleFilter(sop, filterCol, rhs); sop.reset(new PredicateOperator(">=")); sop->setOpType(filterCol->resultType(), lhs->resultType()); sfl = new SimpleFilter(sop, filterCol->clone(), lhs); ParseTree* ptp = new ParseTree(new LogicOperator("and")); ptp->left(sfr); ptp->right(sfl); gwip->ptWorkStack.push(ptp); } return true; } else if (ifp->functype() == Item_func::IN_FUNC) { idbassert(gwip->rcWorkStack.size() >= 2); ReturnedColumn* rhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); ReturnedColumn* lhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); // @bug3038 RowColumn* rrhs = dynamic_cast(rhs); RowColumn* rlhs = dynamic_cast(lhs); if (rrhs && rlhs) { return buildRowColumnFilter(gwip, rrhs, rlhs, ifp); } ConstantColumn* crhs = dynamic_cast(rhs); ConstantColumn* clhs = dynamic_cast(lhs); if (!crhs || !clhs) { gwip->fatalParseError = true; gwip->parseErrorText = "non constant value in IN clause"; return false; } string eqop; string cmbop; Item_func_opt_neg* inp = (Item_func_opt_neg*)ifp; if (inp->negated) { eqop = "<>"; cmbop = "and"; } else { eqop = "="; cmbop = "or"; } sop.reset(new PredicateOperator(eqop)); sop->setOpType(gwip->scsp->resultType(), rhs->resultType()); ConstantFilter* cf = 0; cf = new ConstantFilter(sop, gwip->scsp->clone(), rhs); sop.reset(new LogicOperator(cmbop)); cf->op(sop); sop.reset(new PredicateOperator(eqop)); sop->setOpType(gwip->scsp->resultType(), lhs->resultType()); cf->pushFilter(new SimpleFilter(sop, gwip->scsp->clone(), lhs)); while (!gwip->rcWorkStack.empty()) { lhs = gwip->rcWorkStack.top(); if (dynamic_cast(lhs) == 0) break; gwip->rcWorkStack.pop(); sop.reset(new PredicateOperator(eqop)); sop->setOpType(gwip->scsp->resultType(), lhs->resultType()); cf->pushFilter(new SimpleFilter(sop, gwip->scsp->clone(), lhs)); } if (!gwip->rcWorkStack.empty()) gwip->rcWorkStack.pop(); // pop gwip->scsp if (cf->filterList().size() < inp->argument_count() - 1) { gwip->fatalParseError = true; gwip->parseErrorText = "non constant value in IN clause"; delete cf; return false; } cf->functionName(gwip->funcName); String str; // @bug5811. This filter string is for cross engine to use. // Use real table name. ifp->print(&str, QT_INFINIDB_DERIVED); IDEBUG(cerr << str.ptr() << endl); if (str.ptr()) cf->data(str.c_ptr()); ParseTree* ptp = new ParseTree(cf); gwip->ptWorkStack.push(ptp); } else if (ifp->functype() == Item_func::ISNULL_FUNC || ifp->functype() == Item_func::ISNOTNULL_FUNC) { ReturnedColumn* rhs = NULL; if (!gwip->rcWorkStack.empty() && !gwip->inCaseStmt) { rhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); } else { rhs = buildReturnedColumn(ifp->arguments()[0], *gwip, gwip->fatalParseError); } if (rhs && !gwip->fatalParseError) buildConstPredicate(ifp, rhs, gwip); else if (!rhs) // @bug 3802 { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); return false; } } else if (ifp->functype() == Item_func::GUSERVAR_FUNC) { Item_func_get_user_var* udf = reinterpret_cast(ifp); String buf; if (udf->result_type() == INT_RESULT) { if (udf->unsigned_flag) { gwip->rcWorkStack.push(new ConstantColumn((uint64_t)udf->val_uint())); } else { gwip->rcWorkStack.push(new ConstantColumn((int64_t)udf->val_int())); } } else { //const String* str = udf->val_str(&buf); udf->val_str(&buf); if (!buf.ptr()) { ostringstream oss; oss << "Unknown user variable: " << udf->name.str; gwip->parseErrorText = oss.str(); gwip->fatalParseError = true; return false; } if (udf->result_type() == STRING_RESULT) gwip->rcWorkStack.push(new ConstantColumn(buf.ptr())); else { gwip->rcWorkStack.push(new ConstantColumn(buf.ptr(), ConstantColumn::NUM)); } return false; } } #if 0 else if (ifp->functype() == Item_func::NEG_FUNC) { //peek at the (hopefully) ConstantColumn on the top of stack, negate it in place ConstantColumn* ccp = dynamic_cast(gwip->rcWorkStack.top()); if (!ccp) { ostringstream oss; oss << "Attempt to negate a non-constant column"; gwip->parseErrorText = oss.str(); gwip->fatalParseError = true; return false; } string cval = ccp->constval(); string newval; if (cval[0] == '-') newval.assign(cval, 1, string::npos); else newval = "-" + cval; ccp->constval(newval); } #endif else if (ifp->functype() == Item_func::NOT_FUNC) { if (gwip->condPush && ifp->next->type() == Item::SUBSELECT_ITEM) return false; if (ifp->next && ifp->next->type() == Item::SUBSELECT_ITEM && gwip->lastSub) { gwip->lastSub->handleNot(); return false; } idbassert(ifp->argument_count() == 1); ParseTree* ptp = 0; if (((Item_func*)(ifp->arguments()[0]))->functype() == Item_func::EQUAL_FUNC) { // negate it in place // Note that an EQUAL_FUNC ( a <=> b) was converted to // ( a = b OR ( a is null AND b is null) ) // NOT of the above expression is: ( a != b AND (a is not null OR b is not null ) if (!gwip->ptWorkStack.empty()) ptp = gwip->ptWorkStack.top(); if (ptp) { ParseTree* or_ptp = ptp; ParseTree* and_ptp = or_ptp->right(); ParseTree* equal_ptp = or_ptp->left(); ParseTree* nullck_left_ptp = and_ptp->left(); ParseTree* nullck_right_ptp = and_ptp->right(); SimpleFilter *sf_left_nullck = dynamic_cast(nullck_left_ptp->data()); SimpleFilter *sf_right_nullck = dynamic_cast(nullck_right_ptp->data()); SimpleFilter *sf_equal = dynamic_cast(equal_ptp->data()); if (sf_left_nullck && sf_right_nullck && sf_equal) { // Negate the null checks sf_left_nullck->op()->reverseOp(); sf_right_nullck->op()->reverseOp(); sf_equal->op()->reverseOp(); // Rehook the nodes ptp = and_ptp; ptp->left(equal_ptp); ptp->right(or_ptp); or_ptp->left(nullck_left_ptp); or_ptp->right(nullck_right_ptp); gwip->ptWorkStack.pop(); gwip->ptWorkStack.push(ptp); } else { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_ASSERTION_FAILURE); return false; } } } else if (isPredicateFunction(ifp->arguments()[0], gwip) || ifp->arguments()[0]->type() == Item::COND_ITEM) { // negate it in place if (!gwip->ptWorkStack.empty()) ptp = gwip->ptWorkStack.top(); SimpleFilter* sf = 0; if (ptp) { sf = dynamic_cast(ptp->data()); if (sf) sf->op()->reverseOp(); } } else { // transfrom the not item to item = 0 ReturnedColumn* rhs = 0; if (!gwip->rcWorkStack.empty()) { rhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); } else { rhs = buildReturnedColumn(ifp->arguments()[0], *gwip, gwip->fatalParseError); } if (rhs && !gwip->fatalParseError) return buildConstPredicate(ifp, rhs, gwip); else if (!rhs) // @bug3802 { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); return false; } } } else if (ifp->functype() == Item_func::MULT_EQUAL_FUNC) { #if 0 // MYSQL_5.6 Item_equal* item_eq = (Item_equal*)ifp; // This code is for mysql 5.6. Need to convert to MariaDB 10.1 List_iterator_fast it(item_eq->fields); idbassert(item_eq->fields.elements == 2); // @todo handle more than 2 equal fields Item* item_field = it++; ReturnedColumn* lhs = buildReturnedColumn(item_field, *gwip, gwip->fatalParseError); item_field = it++; ReturnedColumn* rhs = buildReturnedColumn(item_field, *gwip, gwip->fatalParseError); if (!rhs || !lhs) { gwip->fatalParseError = true; gwip->parseErrorText = "Unsupport elements in MULT_EQUAL item"; delete rhs; delete lhs; return false; } PredicateOperator* po = new PredicateOperator("="); boost::shared_ptr sop(po); sop->setOpType(lhs->resultType(), rhs->resultType()); SimpleFilter* sf = new SimpleFilter(sop, lhs, rhs); ParseTree* pt = new ParseTree(sf); gwip->ptWorkStack.push(pt); #endif } else if (ifp->functype() == Item_func::EQUAL_FUNC) { // Convert "a <=> b" to (a = b OR (a IS NULL AND b IS NULL))" idbassert (gwip->rcWorkStack.size() >= 2); ReturnedColumn* rhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); ReturnedColumn* lhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); SimpleFilter* sfn1 = 0; SimpleFilter* sfn2 = 0; SimpleFilter* sfo = 0; // b IS NULL ConstantColumn* nlhs1 = new ConstantColumn("", ConstantColumn::NULLDATA); sop.reset(new PredicateOperator("isnull")); sop->setOpType(lhs->resultType(), rhs->resultType()); sfn1 = new SimpleFilter(sop, rhs, nlhs1); ParseTree* ptpl = new ParseTree(sfn1); // a IS NULL ConstantColumn* nlhs2 = new ConstantColumn("", ConstantColumn::NULLDATA); sop.reset(new PredicateOperator("isnull")); sop->setOpType(lhs->resultType(), rhs->resultType()); sfn2 = new SimpleFilter(sop, lhs, nlhs2); ParseTree* ptpr = new ParseTree(sfn2); // AND them both ParseTree* ptpn = new ParseTree(new LogicOperator("and")); ptpn->left(ptpl); ptpn->right(ptpr); // a = b sop.reset(new PredicateOperator("=")); sop->setOpType(lhs->resultType(), rhs->resultType()); sfo = new SimpleFilter(sop, lhs->clone(), rhs->clone()); // OR with the NULL comparison tree ParseTree* ptp = new ParseTree(new LogicOperator("or")); ptp->left(sfo); ptp->right(ptpn); gwip->ptWorkStack.push(ptp); return true; } else //std rel ops (incl "like") { if (gwip->rcWorkStack.size() < 2) { idbassert(ifp->argument_count() == 2); if (isPredicateFunction(ifp->arguments()[0], gwip) || ifp->arguments()[0]->type() == Item::COND_ITEM || isPredicateFunction(ifp->arguments()[1], gwip) || ifp->arguments()[1]->type() == Item::COND_ITEM) { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); } return false; } ReturnedColumn* rhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); ReturnedColumn* lhs = gwip->rcWorkStack.top(); gwip->rcWorkStack.pop(); // @bug3038. rowcolumn filter RowColumn* rrhs = dynamic_cast(rhs); RowColumn* rlhs = dynamic_cast(lhs); if (rrhs && rlhs) { return buildRowColumnFilter(gwip, rrhs, rlhs, ifp); } // push the column that is associated with the correlated column to the returned // column list, so the materialized view have the complete projection list. // e.g. tout.c1 in (select tin.c1 from tin where tin.c2=tout.c2); // the projetion list of subquery will have tin.c1, tin.c2. ReturnedColumn* correlatedCol = NULL; ReturnedColumn* localCol = NULL; if (rhs->joinInfo() & JOIN_CORRELATED) { correlatedCol = rhs; localCol = lhs; } else if (lhs->joinInfo() & JOIN_CORRELATED) { correlatedCol = lhs; localCol = rhs; } if (correlatedCol && localCol) { ConstantColumn* cc = dynamic_cast(localCol); if ((!cc || (cc && ifp->functype() == Item_func::EQ_FUNC)) && !(localCol->joinInfo() & JOIN_CORRELATED)) { localCol->sequence(gwip->returnedCols.size()); localCol->expressionId(ci->expressionId++); ReturnedColumn* rc = localCol->clone(); rc->colSource(rc->colSource() | CORRELATED_JOIN); gwip->additionalRetCols.push_back(SRCP(rc)); gwip->localCols.push_back(localCol); if (rc->hasWindowFunc()) gwip->windowFuncList.push_back(rc); } // push the correlated join partner to the group by list only when there's aggregate // and we don't push aggregate column to the group by // @bug4756. mysql does not always give correct information about whether there is // aggregate on the SELECT list. Need to figure that by ourselves and then decide // to add the group by or not. if (gwip->subQuery) { if (!localCol->hasAggregate() && !localCol->hasWindowFunc()) gwip->subGroupByCols.push_back(SRCP(localCol->clone())); } if (sop->op() == OP_EQ) { if (gwip->subSelectType == CalpontSelectExecutionPlan::IN_SUBS || gwip->subSelectType == CalpontSelectExecutionPlan::EXISTS_SUBS) correlatedCol->joinInfo(correlatedCol->joinInfo() | JOIN_SEMI); else if (gwip->subSelectType == CalpontSelectExecutionPlan::NOT_IN_SUBS || gwip->subSelectType == CalpontSelectExecutionPlan::NOT_EXISTS_SUBS) correlatedCol->joinInfo(correlatedCol->joinInfo() | JOIN_ANTI); } } SimpleFilter* sf = new SimpleFilter(); //@bug 2101 for when there are only constants in a delete or update where clause (eg "where 5 < 6"). //There will be no field column and it will get here only if the comparison is true. if (gwip->columnMap.empty() && ((current_thd->lex->sql_command == SQLCOM_UPDATE) || (current_thd->lex->sql_command == SQLCOM_UPDATE_MULTI) || (current_thd->lex->sql_command == SQLCOM_DELETE) || (current_thd->lex->sql_command == SQLCOM_DELETE_MULTI))) { IDEBUG( cerr << "deleted func with 2 const columns" << endl ); delete rhs; delete lhs; return false; } // handle noop (only for table mode) if (rhs->data() == "noop" || lhs->data() == "noop") { sop.reset(new Operator("noop")); } else { for (uint32_t i = 0; i < ifp->argument_count(); i++) { if (isPredicateFunction(ifp->arguments()[i], gwip)) { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_SUB_EXPRESSION); } } } sf->op(sop); sf->lhs(lhs); sf->rhs(rhs); sop->setOpType(lhs->resultType(), rhs->resultType()); sop->resultType(sop->operationType()); if (sop->op() == OP_EQ) { CalpontSystemCatalog::TableAliasName tan_lhs; CalpontSystemCatalog::TableAliasName tan_rhs; set::const_iterator it; bool outerjoin = (rhs->singleTable(tan_rhs) && lhs->singleTable(tan_lhs)); // @bug 1632. Alias should be taken account to the identity of tables for selfjoin to work if (outerjoin && tan_lhs != tan_rhs) // join { if (!gwip->condPush) // vtable { if (!gwip->innerTables.empty()) { bool notInner = true; for (it = gwip->innerTables.begin(); it != gwip->innerTables.end(); ++it) { if (tan_lhs.alias == it->alias && tan_lhs.view == it->view) notInner = false; } if (notInner) { lhs->returnAll(true); IDEBUG( cerr << "setting returnAll on " << tan_lhs << endl); } } if (!gwip->innerTables.empty()) { bool notInner = true; for (it = gwip->innerTables.begin(); it != gwip->innerTables.end(); ++it) { if (tan_rhs.alias == it->alias && tan_rhs.view == it->view) notInner = false; } if (notInner) { rhs->returnAll(true); IDEBUG( cerr << "setting returnAll on " << tan_rhs << endl ); } } ParseTree* ptp = new ParseTree(sf); gwip->ptWorkStack.push(ptp); } } else { ParseTree* ptp = new ParseTree(sf); gwip->ptWorkStack.push(ptp); } } else { ParseTree* ptp = new ParseTree(sf); gwip->ptWorkStack.push(ptp); } } return true; } bool buildConstPredicate(Item_func* ifp, ReturnedColumn* rhs, gp_walk_info* gwip) { SimpleFilter* sf = new SimpleFilter(); boost::shared_ptr sop(new PredicateOperator(ifp->func_name())); ConstantColumn* lhs = 0; if (ifp->functype() == Item_func::ISNULL_FUNC) { lhs = new ConstantColumn("", ConstantColumn::NULLDATA); sop.reset(new PredicateOperator("isnull")); } else if (ifp->functype() == Item_func::ISNOTNULL_FUNC) { lhs = new ConstantColumn("", ConstantColumn::NULLDATA); sop.reset(new PredicateOperator("isnotnull")); } else //if (ifp->functype() == Item_func::NOT_FUNC) { lhs = new ConstantColumn((int64_t)0, ConstantColumn::NUM); sop.reset(new PredicateOperator("=")); } CalpontSystemCatalog::ColType opType = rhs->resultType(); if ( (opType.colDataType == CalpontSystemCatalog::CHAR && opType.colWidth <= 8) || (opType.colDataType == CalpontSystemCatalog::VARCHAR && opType.colWidth < 8) || (opType.colDataType == CalpontSystemCatalog::VARBINARY && opType.colWidth < 8) ) { opType.colDataType = execplan::CalpontSystemCatalog::BIGINT; opType.colWidth = 8; } sop->operationType(opType); sf->op(sop); //yes, these are backwards assert (lhs); sf->lhs(rhs); sf->rhs(lhs); ParseTree* ptp = new ParseTree(sf); gwip->ptWorkStack.push(ptp); return true; } SimpleColumn* buildSimpleColFromDerivedTable(gp_walk_info& gwi, Item_field* ifp) { SimpleColumn* sc = NULL; // view name string viewName = getViewName(ifp->cached_table); // Check from the end because local scope resolve is preferred for (int32_t i = gwi.tbList.size() - 1; i >= 0; i--) { if (sc) break; if (!gwi.tbList[i].schema.empty() && !gwi.tbList[i].table.empty() && (!ifp->table_name || strcasecmp(ifp->table_name, gwi.tbList[i].alias.c_str()) == 0)) { CalpontSystemCatalog::TableName tn(gwi.tbList[i].schema, gwi.tbList[i].table); CalpontSystemCatalog::RIDList oidlist = gwi.csc->columnRIDs(tn, true); for (unsigned int j = 0; j < oidlist.size(); j++) { CalpontSystemCatalog::TableColName tcn = gwi.csc->colName(oidlist[j].objnum); CalpontSystemCatalog::ColType ct = gwi.csc->colType(oidlist[j].objnum); if (strcasecmp(ifp->field_name.str, tcn.column.c_str()) == 0) { // @bug4827. Remove the checking because outside tables could be the same // name as inner tables. This function is to identify column from a table, // as long as it matches the next step in predicate parsing will tell the scope // of the column. /*if (sc) { gwi.fatalParseError = true; Message::Args args; args.add(ifp->name); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_AMBIGUOUS_COL, args); return NULL; }*/ sc = new SimpleColumn(); sc->columnName(tcn.column); sc->tableName(tcn.table); sc->schemaName(tcn.schema); sc->oid(oidlist[j].objnum); // @bug 3003. Keep column alias if it has. sc->alias(ifp->is_autogenerated_name ? tcn.column : ifp->name.str); sc->tableAlias(lower(gwi.tbList[i].alias)); sc->viewName(lower(viewName)); sc->resultType(ct); break; } } } } if (sc) return sc; // Check from the end because local scope resolve is preferred for (int32_t i = gwi.derivedTbList.size() - 1; i >= 0; i--) { if (sc) break; CalpontSelectExecutionPlan* csep = dynamic_cast(gwi.derivedTbList[i].get()); string derivedName = csep->derivedTbAlias(); if (!ifp->table_name || strcasecmp(ifp->table_name, derivedName.c_str()) == 0) { CalpontSelectExecutionPlan::ReturnedColumnList cols = csep->returnedCols(); for (uint32_t j = 0; j < cols.size(); j++) { // @bug 3167 duplicate column alias is full name SimpleColumn* col = dynamic_cast(cols[j].get()); string alias = cols[j]->alias(); if (strcasecmp(ifp->field_name.str, alias.c_str()) == 0 || (col && alias.find(".") != string::npos && (strcasecmp(ifp->field_name.str, col->columnName().c_str()) == 0 || strcasecmp(ifp->field_name.str, (alias.substr(alias.find_last_of(".") + 1)).c_str()) == 0))) //@bug6066 { // @bug4827. Remove the checking because outside tables could be the same // name as inner tables. This function is to identify column from a table, // as long as it matches the next step in predicate parsing will tell the scope // of the column. /*if (sc) { gwi.fatalParseError = true; Message::Args args; args.add(ifp->name); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_AMBIGUOUS_COL, args); return NULL; }*/ sc = new SimpleColumn(); if (!col) sc->columnName(cols[j]->alias()); else sc->columnName(col->columnName()); // @bug 3003. Keep column alias if it has. sc->alias(ifp->is_autogenerated_name ? cols[j]->alias() : ifp->name.str); sc->tableName(csep->derivedTbAlias()); sc->colPosition(j); string tableAlias(csep->derivedTbAlias()); sc->tableAlias(lower(tableAlias)); sc->viewName(lower(viewName)); sc->resultType(cols[j]->resultType()); sc->hasAggregate(cols[j]->hasAggregate()); if (col) sc->isInfiniDB(col->isInfiniDB()); // @bug5634, @bug5635. mark used derived col on derived table. // outer join inner table filter can not be moved in // MariaDB 10.1: cached_table is never available for derived tables. // Find the uncached object in table_list TABLE_LIST* tblList = ifp->context ? ifp->context->table_list : NULL; while (tblList) { if (strcasecmp(tblList->alias.str, ifp->table_name) == 0) { if (!tblList->outer_join) { sc->derivedTable(derivedName); sc->derivedRefCol(cols[j].get()); } break; } tblList = tblList->next_local; } cols[j]->incRefCount(); break; } } } } if (!sc) { gwi.fatalParseError = true; Message::Args args; string name; if (ifp->db_name) name += string(ifp->db_name) + "."; if (ifp->table_name) name += string(ifp->table_name) + "."; if (ifp->name.length) name += ifp->name.str; args.add(name); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_UNKNOWN_COL, args); } if (ifp->depended_from) { sc->joinInfo(sc->joinInfo() | JOIN_CORRELATED); if (gwi.subQuery) gwi.subQuery->correlated(true); // for error out non-support select filter case (comparison outside semi join tables) gwi.correlatedTbNameVec.push_back(make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias())); // imply semi for scalar for now. if (gwi.subSelectType == CalpontSelectExecutionPlan::SINGLEROW_SUBS) sc->joinInfo(sc->joinInfo() | JOIN_SCALAR | JOIN_SEMI); if (gwi.subSelectType == CalpontSelectExecutionPlan::SELECT_SUBS) sc->joinInfo(sc->joinInfo() | JOIN_SCALAR | JOIN_OUTER_SELECT); } return sc; } // for FROM clause subquery. get all the columns from real tables and derived tables. void collectAllCols(gp_walk_info& gwi, Item_field* ifp) { // view name string viewName = getViewName(ifp->cached_table); if (gwi.derivedTbList.empty()) { gwi.fatalParseError = true; Message::Args args; args.add("*"); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_UNKNOWN_COL, args); } string tableName = (ifp->table_name ? string(ifp->table_name) : ""); CalpontSelectExecutionPlan::SelectList::const_iterator it = gwi.derivedTbList.begin(); for (uint32_t i = 0; i < gwi.tbList.size(); i++) { SRCP srcp; // derived table if (gwi.tbList[i].schema.empty()) { CalpontSelectExecutionPlan* csep = dynamic_cast((*it).get()); ++it; if (!tableName.empty() && strcasecmp(tableName.c_str(), csep->derivedTbAlias().c_str()) != 0) continue; CalpontSelectExecutionPlan::ReturnedColumnList cols = csep->returnedCols(); for (uint32_t j = 0; j < cols.size(); j++) { SimpleColumn* sc = new SimpleColumn(); sc->columnName(cols[j]->alias()); sc->colPosition(j); string tableAlias(csep->derivedTbAlias()); sc->tableAlias(lower(tableAlias)); sc->viewName(lower(gwi.tbList[i].view)); sc->resultType(cols[j]->resultType()); // @bug5634 derived table optimization cols[j]->incRefCount(); sc->derivedTable(tableAlias); sc->derivedRefCol(cols[j].get()); srcp.reset(sc); gwi.returnedCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(sc->columnName(), srcp)); } } // real tables else { CalpontSystemCatalog::TableName tn(gwi.tbList[i].schema, gwi.tbList[i].table); if (!tableName.empty() && (strcasecmp(tableName.c_str(), tn.table.c_str()) != 0 && strcasecmp(tableName.c_str(), gwi.tbList[i].alias.c_str()) != 0 )) continue; CalpontSystemCatalog::RIDList oidlist = gwi.csc->columnRIDs(tn, true); for (unsigned int j = 0; j < oidlist.size(); j++) { SimpleColumn* sc = new SimpleColumn(); CalpontSystemCatalog::TableColName tcn = gwi.csc->colName(oidlist[j].objnum); CalpontSystemCatalog::ColType ct = gwi.csc->colType(oidlist[j].objnum); sc->columnName(tcn.column); sc->tableName(tcn.table); sc->schemaName(tcn.schema); sc->oid(oidlist[j].objnum); sc->alias(tcn.column); sc->resultType(ct); sc->tableAlias(lower(gwi.tbList[i].alias)); sc->viewName(lower(viewName)); srcp.reset(sc); gwi.returnedCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(sc->columnName(), srcp)); } } } } void buildSubselectFunc(Item_func* ifp, gp_walk_info* gwip) { // @bug 3035 if (!isPredicateFunction(ifp, gwip)) { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_FUNC_SUB); return; } WhereSubQuery* subquery = NULL; for (uint32_t i = 0; i < ifp->argument_count(); i++) { #if MYSQL_VERSION_ID >= 50172 // changes comply with mysql 5.1.72 if (ifp->arguments()[i]->type() == Item::FUNC_ITEM && string(((Item_func*)ifp->arguments()[i])->func_name()) == "") { if (ifp->functype() == Item_func::NOT_FUNC) { if (gwip->lastSub) gwip->lastSub->handleNot(); } } #endif if (ifp->arguments()[i]->type() == Item::SUBSELECT_ITEM) { Item_subselect* sub = (Item_subselect*)ifp->arguments()[i]; switch (sub->substype()) { case Item_subselect::SINGLEROW_SUBS: subquery = new ScalarSub(*gwip, ifp); break; case Item_subselect::IN_SUBS: subquery = new InSub(*gwip, ifp); break; case Item_subselect::EXISTS_SUBS: // exists sub has been handled earlier. here is for not function if (ifp->functype() == Item_func::NOT_FUNC) { if (gwip->lastSub) gwip->lastSub->handleNot(); } break; default: Message::Args args; gwip->fatalParseError = true; gwip->parseErrorText = "non supported subquery"; return; } } } if (subquery) { gwip->hasSubSelect = true; SubQuery* orig = gwip->subQuery; gwip->subQuery = subquery; // no need to check NULL for now. error will be handled in gp_walk gwip->ptWorkStack.push(subquery->transform()); // recover original sub. Save current sub for Not handling. gwip->lastSub = subquery; gwip->subQuery = orig; } return; } bool isPredicateFunction(Item* item, gp_walk_info* gwip) { if (item->type() == Item::COND_ITEM) return true; if (item->type() != Item::FUNC_ITEM) return false; Item_func* ifp = (Item_func*)item; return ( ifp->functype() == Item_func::EQ_FUNC || ifp->functype() == Item_func::NE_FUNC || ifp->functype() == Item_func::LT_FUNC || ifp->functype() == Item_func::LE_FUNC || ifp->functype() == Item_func::GE_FUNC || ifp->functype() == Item_func::GT_FUNC || ifp->functype() == Item_func::LIKE_FUNC || ifp->functype() == Item_func::BETWEEN || ifp->functype() == Item_func::IN_FUNC || (ifp->functype() == Item_func::ISNULL_FUNC && (gwip->clauseType == WHERE || gwip->clauseType == HAVING)) || (ifp->functype() == Item_func::ISNOTNULL_FUNC && (gwip->clauseType == WHERE || gwip->clauseType == HAVING)) || ifp->functype() == Item_func::NOT_FUNC || ifp->functype() == Item_func::ISNOTNULLTEST_FUNC || ifp->functype() == Item_func::TRIG_COND_FUNC || string(ifp->func_name()) == ""/* || string(ifp->func_name()) == "xor"*/); } void setError(THD* thd, uint32_t errcode, string errmsg) { thd->get_stmt_da()->set_overwrite_status(true); if (errmsg.empty()) errmsg = "Unknown error"; if (errcode < ER_ERROR_FIRST || errcode > ER_ERROR_LAST) { errcode = ER_UNKNOWN_ERROR; } thd->raise_error_printf(errcode, errmsg.c_str()); thd->infinidb_vtable.isNewQuery = true; thd->infinidb_vtable.override_largeside_estimate = false; // reset expressionID if (!(thd->infinidb_vtable.cal_conn_info)) thd->infinidb_vtable.cal_conn_info = (void*)(new cal_connection_info()); cal_connection_info* ci = reinterpret_cast(thd->infinidb_vtable.cal_conn_info); ci->expressionId = 0; } void setError(THD* thd, uint32_t errcode, string errmsg, gp_walk_info& gwi) { setError(thd, errcode, errmsg); clearStacks(gwi); } const string bestTableName(const Item_field* ifp) { idbassert(ifp); if (!ifp->table_name) return ""; if (!ifp->field) return ifp->table_name; string table_name(ifp->table_name); string field_table_table_name; if (ifp->cached_table) field_table_table_name = ifp->cached_table->table_name.str; else if (ifp->field->table && ifp->field->table->s && ifp->field->table->s->table_name.str) field_table_table_name = ifp->field->table->s->table_name.str; string tn; if (!field_table_table_name.empty()) tn = field_table_table_name; else tn = table_name; return tn; } uint32_t setAggOp(AggregateColumn* ac, Item_sum* isp) { Item_sum::Sumfunctype agg_type = isp->sum_func(); uint32_t rc = 0; switch (agg_type) { case Item_sum::COUNT_FUNC: ac->aggOp(AggregateColumn::COUNT); return rc; case Item_sum::SUM_FUNC: ac->aggOp(AggregateColumn::SUM); return rc; case Item_sum::AVG_FUNC: ac->aggOp(AggregateColumn::AVG); return rc; case Item_sum::MIN_FUNC: ac->aggOp(AggregateColumn::MIN); return rc; case Item_sum::MAX_FUNC: ac->aggOp(AggregateColumn::MAX); return rc; case Item_sum::COUNT_DISTINCT_FUNC: ac->aggOp(AggregateColumn::DISTINCT_COUNT); ac->distinct(true); return rc; case Item_sum::SUM_DISTINCT_FUNC: ac->aggOp(AggregateColumn::DISTINCT_SUM); ac->distinct(true); return rc; case Item_sum::AVG_DISTINCT_FUNC: ac->aggOp(AggregateColumn::DISTINCT_AVG); ac->distinct(true); return rc; case Item_sum::STD_FUNC: { Item_sum_variance* var = (Item_sum_variance*)isp; if (var->sample) ac->aggOp(AggregateColumn::STDDEV_SAMP); else ac->aggOp(AggregateColumn::STDDEV_POP); return rc; } case Item_sum::VARIANCE_FUNC: { Item_sum_variance* var = (Item_sum_variance*)isp; if (var->sample) ac->aggOp(AggregateColumn::VAR_SAMP); else ac->aggOp(AggregateColumn::VAR_POP); return rc; } case Item_sum::GROUP_CONCAT_FUNC: { Item_func_group_concat* gc = (Item_func_group_concat*)isp; ac->aggOp(AggregateColumn::GROUP_CONCAT); ac->distinct(gc->isDistinct()); return rc; } case Item_sum::SUM_BIT_FUNC: { string funcName = isp->func_name(); if (funcName.compare("bit_and(") == 0) ac->aggOp(AggregateColumn::BIT_AND); else if (funcName.compare("bit_or(") == 0) ac->aggOp(AggregateColumn::BIT_OR); else if (funcName.compare("bit_xor(") == 0) ac->aggOp(AggregateColumn::BIT_XOR); else return ER_CHECK_NOT_IMPLEMENTED; return rc; } case Item_sum::UDF_SUM_FUNC: ac->aggOp(AggregateColumn::UDAF); return rc; default: return ER_CHECK_NOT_IMPLEMENTED; } } /* get the smallest column of a table. Used for filling columnmap */ SimpleColumn* getSmallestColumn(boost::shared_ptr csc, CalpontSystemCatalog::TableName& tn, CalpontSystemCatalog::TableAliasName& tan, TABLE* table, gp_walk_info& gwi) { // derived table if (tan.schema.empty()) { for (uint32_t i = 0; i < gwi.derivedTbList.size(); i++) { CalpontSelectExecutionPlan* csep = dynamic_cast(gwi.derivedTbList[i].get()); if (tan.alias == csep->derivedTbAlias()) { assert (!csep->returnedCols().empty()); ReturnedColumn* rc = dynamic_cast(csep->returnedCols()[0].get()); SimpleColumn* sc = new SimpleColumn(); sc->columnName(rc->alias()); sc->sequence(0); sc->tableAlias(lower(tan.alias)); // @bug5634 derived table optimization. rc->incRefCount(); sc->derivedTable(csep->derivedTbAlias()); sc->derivedRefCol(rc); return sc; } } throw runtime_error ("getSmallestColumn: Internal error."); } // check engine type if (!tan.fIsInfiniDB) { // get the first column to project. @todo optimization to get the smallest one for foreign engine. Field* field = *(table->field); SimpleColumn* sc = new SimpleColumn(table->s->db.str, table->s->table_name.str, field->field_name.str, tan.fIsInfiniDB, gwi.sessionid); string alias(table->alias.ptr()); sc->tableAlias(lower(alias)); sc->isInfiniDB(false); sc->resultType(fieldType_MysqlToIDB(field)); sc->oid(field->field_index + 1); return sc; } CalpontSystemCatalog::RIDList oidlist = csc->columnRIDs(tn, true); CalpontSystemCatalog::TableColName tcn; int minColWidth = -1; int minWidthColOffset = 0; for (unsigned int j = 0; j < oidlist.size(); j++) { CalpontSystemCatalog::ColType ct = csc->colType(oidlist[j].objnum); if (ct.colDataType == CalpontSystemCatalog::VARBINARY) continue; if (minColWidth == -1 || ct.colWidth < minColWidth) { minColWidth = ct.colWidth; minWidthColOffset = j; } } tcn = csc->colName(oidlist[minWidthColOffset].objnum); SimpleColumn* sc = new SimpleColumn(tcn.schema, tcn.table, tcn.column, csc->sessionID()); sc->tableAlias(lower(tan.alias)); sc->viewName(lower(tan.view)); sc->resultType(csc->colType(oidlist[minWidthColOffset].objnum)); return sc; } CalpontSystemCatalog::ColType fieldType_MysqlToIDB (const Field* field) { CalpontSystemCatalog::ColType ct; ct.precision = 4; switch (field->result_type()) { case INT_RESULT: ct.colDataType = CalpontSystemCatalog::BIGINT; ct.colWidth = 8; break; case STRING_RESULT: ct.colDataType = CalpontSystemCatalog::VARCHAR; ct.colWidth = field->field_length; break; case DECIMAL_RESULT: { Field_decimal* idp = (Field_decimal*)field; ct.colDataType = CalpontSystemCatalog::DECIMAL; ct.colWidth = 8; ct.scale = idp->dec; if (ct.scale == 0) ct.precision = idp->field_length - 1; else ct.precision = idp->field_length - idp->dec; break; } case REAL_RESULT: ct.colDataType = CalpontSystemCatalog::DOUBLE; ct.colWidth = 8; break; default: IDEBUG( cerr << "fieldType_MysqlToIDB:: Unknown result type of MySQL " << field->result_type() << endl ); break; } return ct; } CalpontSystemCatalog::ColType colType_MysqlToIDB (const Item* item) { CalpontSystemCatalog::ColType ct; ct.precision = 4; switch (item->result_type()) { case INT_RESULT: if (item->unsigned_flag) { ct.colDataType = CalpontSystemCatalog::UBIGINT; } else { ct.colDataType = CalpontSystemCatalog::BIGINT; } ct.colWidth = 8; break; case STRING_RESULT: ct.colDataType = CalpontSystemCatalog::VARCHAR; // MCOL-697 the longest TEXT we deal with is 2100000000 so // limit to that. Otherwise for LONGBLOB ct.colWidth ends // up being -1 and demons eat your data and first born // (or you just get truncated to 20 bytes with the 'if' // below) if (item->max_length < 2100000000) ct.colWidth = item->max_length; else ct.colWidth = 2100000000; // force token if (item->type() == Item::FUNC_ITEM) { if (ct.colWidth < 20) ct.colWidth = 20; // for infinidb date length } // @bug5083. MySQL gives string type for date/datetime column. // need to adjust here. if (item->type() == Item::FIELD_ITEM) { if (item->field_type() == MYSQL_TYPE_DATE) { ct.colDataType = CalpontSystemCatalog::DATE; ct.colWidth = 4; } else if (item->field_type() == MYSQL_TYPE_DATETIME || item->field_type() == MYSQL_TYPE_DATETIME2 || item->field_type() == MYSQL_TYPE_TIMESTAMP || item->field_type() == MYSQL_TYPE_TIMESTAMP2 ) { ct.colDataType = CalpontSystemCatalog::DATETIME; ct.colWidth = 8; } else if (item->field_type() == MYSQL_TYPE_TIME) { ct.colDataType = CalpontSystemCatalog::TIME; ct.colWidth = 8; } if (item->field_type() == MYSQL_TYPE_BLOB) { ct.colDataType = CalpontSystemCatalog::BLOB; } } break; /* FIXME: case xxxBINARY_RESULT: ct.colDataType = CalpontSystemCatalog::VARBINARY; ct.colWidth = item->max_length; // force token if (item->type() == Item::FUNC_ITEM) { if (ct.colWidth < 20) ct.colWidth = 20; // for infinidb date length } break; */ case DECIMAL_RESULT: { Item_decimal* idp = (Item_decimal*)item; ct.colDataType = CalpontSystemCatalog::DECIMAL; ct.colWidth = 8; ct.scale = idp->decimals; if (ct.scale == 0) ct.precision = idp->max_length - 1; else ct.precision = idp->max_length - idp->decimals; break; } case REAL_RESULT: ct.colDataType = CalpontSystemCatalog::DOUBLE; ct.colWidth = 8; break; default: IDEBUG( cerr << "colType_MysqlToIDB:: Unknown result type of MySQL " << item->result_type() << endl ); break; } return ct; } ReturnedColumn* buildReturnedColumn(Item* item, gp_walk_info& gwi, bool& nonSupport, bool pushdownHand) { ReturnedColumn* rc = NULL; if ( gwi.thd) { //if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI )) { if ( !item->fixed) { item->fix_fields(gwi.thd, (Item**)&item); } } } switch (item->type()) { case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)item; return buildSimpleColumn(ifp, gwi); } case Item::INT_ITEM: case Item::VARBIN_ITEM: { String val, *str = item->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); if (item->unsigned_flag) { //cc = new ConstantColumn(valStr, (uint64_t)item->val_uint(), ConstantColumn::NUM); // It seems that str at this point is crap if val_uint() is > MAX_BIGINT. By using // this constructor, ConstantColumn is built with the proper string. For whatever reason, // ExeMgr converts the fConstval member to numeric, rather than using the existing numeric // values available, so it's important to have fConstval correct. rc = new ConstantColumn((uint64_t)item->val_uint(), ConstantColumn::NUM); } else { rc = new ConstantColumn(valStr, (int64_t)item->val_int(), ConstantColumn::NUM); } //return cc; break; } case Item::STRING_ITEM: { String val, *str = item->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); rc = new ConstantColumn(valStr); break; } case Item::REAL_ITEM: { String val, *str = item->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); rc = new ConstantColumn(valStr, item->val_real()); break; } case Item::DECIMAL_ITEM: { rc = buildDecimalColumn(item, gwi); break; } case Item::FUNC_ITEM: { Item_func* ifp = (Item_func*)item; string func_name = ifp->func_name(); // try to evaluate const F&E. only for select clause vector tmpVec; //bool hasAggColumn = false; uint16_t parseInfo = 0; parse_item(ifp, tmpVec, gwi.fatalParseError, parseInfo); if (parseInfo & SUB_BIT) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_SELECT_SUB); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); break; } if (!gwi.fatalParseError && !nonConstFunc(ifp) && !(parseInfo & AF_BIT) && (tmpVec.size() == 0)) { String val, *str = ifp->val_str(&val); string valStr; if (str) { valStr.assign(str->ptr(), str->length()); } if (!str) { rc = new ConstantColumn("", ConstantColumn::NULLDATA); } else if (ifp->result_type() == STRING_RESULT) { rc = new ConstantColumn(valStr, ConstantColumn::LITERAL); rc->resultType(colType_MysqlToIDB(item)); } else if (ifp->result_type() == DECIMAL_RESULT) rc = buildDecimalColumn(ifp, gwi); else { rc = new ConstantColumn(valStr, ConstantColumn::NUM); rc->resultType(colType_MysqlToIDB(item)); } break; } if (func_name == "+" || func_name == "-" || func_name == "*" || func_name == "/" ) return buildArithmeticColumn(ifp, gwi, nonSupport, pushdownHand); else return buildFunctionColumn(ifp, gwi, nonSupport, pushdownHand); } case Item::SUM_FUNC_ITEM: { return buildAggregateColumn(item, gwi); } case Item::REF_ITEM: { Item_ref* ref = (Item_ref*)item; switch ((*(ref->ref))->type()) { case Item::SUM_FUNC_ITEM: return buildAggregateColumn(*(ref->ref), gwi); case Item::FIELD_ITEM: return buildReturnedColumn(*(ref->ref), gwi, nonSupport); case Item::REF_ITEM: return buildReturnedColumn(*(((Item_ref*)(*(ref->ref)))->ref), gwi, nonSupport); case Item::FUNC_ITEM: return buildFunctionColumn((Item_func*)(*(ref->ref)), gwi, nonSupport); case Item::WINDOW_FUNC_ITEM: return buildWindowFunctionColumn(*(ref->ref), gwi, nonSupport); default: gwi.fatalParseError = true; gwi.parseErrorText = "Unknown REF item"; break; } } case Item::NULL_ITEM: { if (gwi.condPush) return new SimpleColumn("noop"); return new ConstantColumn("", ConstantColumn::NULLDATA); } case Item::CACHE_ITEM: { Item* col = ((Item_cache*)item)->get_example(); rc = buildReturnedColumn(col, gwi, nonSupport); if (rc) { ConstantColumn* cc = dynamic_cast(rc); if (!cc) { rc->joinInfo(rc->joinInfo() | JOIN_CORRELATED); if (gwi.subQuery) gwi.subQuery->correlated(true); } } break; } case Item::EXPR_CACHE_ITEM: { // TODO: item is a Item_cache_wrapper printf("EXPR_CACHE_ITEM in buildReturnedColumn\n"); cerr << "EXPR_CACHE_ITEM in buildReturnedColumn" << endl; break; } case Item::DATE_ITEM: { String val, *str = item->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); rc = new ConstantColumn(valStr); break; } case Item::WINDOW_FUNC_ITEM: { return buildWindowFunctionColumn(item, gwi, nonSupport); } #if INTERVAL_ITEM case Item::INTERVAL_ITEM: { Item_interval* interval = (Item_interval*)item; SRCP srcp; srcp.reset(buildReturnedColumn(interval->item, gwi, nonSupport)); if (!srcp) return NULL; rc = new IntervalColumn(srcp, (int)interval->interval); rc->resultType(srcp->resultType()); break; } #endif case Item::SUBSELECT_ITEM: { gwi.hasSubSelect = true; break; } case Item::COND_ITEM: { // MCOL-1196: Allow COND_ITEM thru. They will be picked up // by further logic. It may become desirable to add code here. break; } default: { gwi.fatalParseError = true; gwi.parseErrorText = "Unknown item type"; break; } } if (rc && item->name.length) rc->alias(item->name.str); return rc; } ArithmeticColumn* buildArithmeticColumn( Item_func* item, gp_walk_info& gwi, bool& nonSupport, bool pushdownHand) { if (!(gwi.thd->infinidb_vtable.cal_conn_info)) gwi.thd->infinidb_vtable.cal_conn_info = (void*)(new cal_connection_info()); cal_connection_info* ci = reinterpret_cast(gwi.thd->infinidb_vtable.cal_conn_info); ArithmeticColumn* ac = new ArithmeticColumn(); Item** sfitempp = item->arguments(); ArithmeticOperator* aop = new ArithmeticOperator(item->func_name()); ParseTree* pt = new ParseTree(aop); //ReturnedColumn *lhs = 0, *rhs = 0; ParseTree* lhs = 0, *rhs = 0; SRCP srcp; if (item->name.length) ac->alias(item->name.str); // argument_count() should generally be 2, except negate expression if (item->argument_count() == 2) { if (gwi.clauseType == SELECT || /*gwi.clauseType == HAVING || */gwi.clauseType == GROUP_BY || gwi.clauseType == FROM) // select list { lhs = new ParseTree(buildReturnedColumn(sfitempp[0], gwi, nonSupport, pushdownHand)); if (!lhs->data() && (sfitempp[0]->type() == Item::FUNC_ITEM)) { delete lhs; Item_func* ifp = (Item_func*)sfitempp[0]; lhs = buildParseTree(ifp, gwi, nonSupport); } else if(pushdownHand && !lhs->data() && (sfitempp[0]->type() == Item::REF_ITEM)) { // There must be an aggregation column in extended SELECT // list so find the corresponding column. // Could have it set if there are aggregation funcs as this function arguments. gwi.fatalParseError = false; ReturnedColumn* rc = buildAggFrmTempField(sfitempp[0], gwi); if(rc) lhs = new ParseTree(rc); } rhs = new ParseTree(buildReturnedColumn(sfitempp[1], gwi, nonSupport, pushdownHand)); if (!rhs->data() && (sfitempp[1]->type() == Item::FUNC_ITEM)) { delete rhs; Item_func* ifp = (Item_func*)sfitempp[1]; rhs = buildParseTree(ifp, gwi, nonSupport); } else if(pushdownHand && !rhs->data() && (sfitempp[1]->type() == Item::REF_ITEM)) { // There must be an aggregation column in extended SELECT // list so find the corresponding column. // Could have it set if there are aggregation funcs as this function arguments. gwi.fatalParseError = false; ReturnedColumn* rc = buildAggFrmTempField(sfitempp[1], gwi); if(rc) rhs = new ParseTree(rc); } } else // where clause { if (isPredicateFunction(sfitempp[1], &gwi)) { if (gwi.ptWorkStack.empty()) { rhs = new ParseTree(buildReturnedColumn(sfitempp[1], gwi, nonSupport)); } else { rhs = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); } } else { if (gwi.rcWorkStack.empty()) { rhs = new ParseTree(buildReturnedColumn(sfitempp[1], gwi, nonSupport)); } else { rhs = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } } if (isPredicateFunction(sfitempp[0], &gwi)) { if (gwi.ptWorkStack.empty()) { lhs = new ParseTree(buildReturnedColumn(sfitempp[0], gwi, nonSupport)); } else { lhs = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); } } else { if (gwi.rcWorkStack.empty()) { lhs = new ParseTree(buildReturnedColumn(sfitempp[0], gwi, nonSupport)); } else { lhs = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } } } if (nonSupport || !lhs->data() || !rhs->data()) { gwi.fatalParseError = true; if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Un-recognized Arithmetic Operand"; delete lhs; delete rhs; return NULL; } //aop->operationType(lhs->resultType(), rhs->resultType()); pt->left(lhs); pt->right(rhs); } else { ConstantColumn* cc = new ConstantColumn(string("0"), (int64_t)0); if (gwi.clauseType == SELECT || gwi.clauseType == HAVING || gwi.clauseType == GROUP_BY) // select clause { rhs = new ParseTree(buildReturnedColumn(sfitempp[0], gwi, nonSupport)); } else { if (gwi.rcWorkStack.empty()) { rhs = new ParseTree(buildReturnedColumn(sfitempp[0], gwi, nonSupport)); } else { rhs = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } } if (nonSupport || !rhs->data()) { gwi.fatalParseError = true; if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Un-recognized Arithmetic Operand"; delete rhs; return NULL; } pt->left(cc); pt->right(rhs); } //aop->resultType(colType_MysqlToIDB(item)); // @bug5715. Use InfiniDB adjusted coltype for result type. // decimal arithmetic operation gives double result when the session variable is set. //idbassert(pt->left() && pt->right() && pt->left()->data() && pt->right()->data()); CalpontSystemCatalog::ColType mysql_type = colType_MysqlToIDB(item); if (current_thd->variables.infinidb_double_for_decimal_math == 1) aop->adjustResultType(mysql_type); else aop->resultType(mysql_type); // adjust decimal result type according to internalDecimalScale if (gwi.internalDecimalScale >= 0 && aop->resultType().colDataType == CalpontSystemCatalog::DECIMAL) { CalpontSystemCatalog::ColType ct = aop->resultType(); ct.scale = gwi.internalDecimalScale; aop->resultType(ct); } aop->operationType(aop->resultType()); ac->expression(pt); ac->resultType(aop->resultType()); ac->operationType(aop->operationType()); ac->expressionId(ci->expressionId++); // @3391. optimization. try to associate expression ID to the expression on the select list if (gwi.clauseType != SELECT) { for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if ((!ac->alias().empty()) && strcasecmp(ac->alias().c_str(), gwi.returnedCols[i]->alias().c_str()) == 0) { ac->expressionId(gwi.returnedCols[i]->expressionId()); break; } } } // For function join. If any argument has non-zero joininfo, set it to the function. ac->setSimpleColumnList(); std::vector simpleColList = ac->simpleColumnList(); for (uint i = 0; i < simpleColList.size(); i++) { if (simpleColList[i]->joinInfo() != 0) { ac->joinInfo(simpleColList[i]->joinInfo()); break; } } return ac; } ReturnedColumn* buildFunctionColumn( Item_func* ifp, gp_walk_info& gwi, bool& nonSupport, bool pushdownHand) { if (!(gwi.thd->infinidb_vtable.cal_conn_info)) gwi.thd->infinidb_vtable.cal_conn_info = (void*)(new cal_connection_info()); cal_connection_info* ci = reinterpret_cast(gwi.thd->infinidb_vtable.cal_conn_info); string funcName = ifp->func_name(); FuncExp* funcExp = FuncExp::instance(); Func* functor; FunctionColumn* fc = NULL; // Pseudocolumn uint32_t pseudoType = PSEUDO_UNKNOWN; if (ifp->functype() == Item_func::UDF_FUNC) pseudoType = PseudoColumn::pseudoNameToType(funcName); if (pseudoType != PSEUDO_UNKNOWN) { ReturnedColumn* rc = buildPseudoColumn(ifp, gwi, gwi.fatalParseError, pseudoType); if (!rc || gwi.fatalParseError) { if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Unsupported function."; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return NULL; } return rc; } // TODO MariaDB 10.1: Until we figure out what to do with this if (funcName == "multiple equal") return NULL; // Arithmetic exp if (funcName == "+" || funcName == "-" || funcName == "*" || funcName == "/" ) { ArithmeticColumn* ac = buildArithmeticColumn(ifp, gwi, nonSupport, pushdownHand); return ac; } else if (funcName == "case") { fc = buildCaseFunction(ifp, gwi, nonSupport); } else if ((functor = funcExp->getFunctor(funcName))) { // where clause isnull still treated as predicate operator // MCOL-686: between also a predicate operator so that extent elimination can happen if ((funcName == "isnull" || funcName == "isnotnull" || funcName == "between") && (gwi.clauseType == WHERE || gwi.clauseType == HAVING)) return NULL; if (funcName == "in" || funcName == " IN " || funcName == "between") { // if F&E involved, build function. otherwise, fall to the old path. // @todo need more checks here if (ifp->arguments()[0]->type() == Item::ROW_ITEM) { return NULL; } if (ifp->arguments()[0]->type() == Item::FIELD_ITEM || (ifp->arguments()[0]->type() == Item::REF_ITEM && (*(((Item_ref*)ifp->arguments()[0])->ref))->type() == Item::FIELD_ITEM)) { bool fe = false; for (uint32_t i = 1; i < ifp->argument_count(); i++) { if (!(ifp->arguments()[i]->type() == Item::INT_ITEM || ifp->arguments()[i]->type() == Item::STRING_ITEM || ifp->arguments()[i]->type() == Item::REAL_ITEM || ifp->arguments()[i]->type() == Item::DECIMAL_ITEM || ifp->arguments()[i]->type() == Item::NULL_ITEM)) { if (ifp->arguments()[i]->type() == Item::FUNC_ITEM) { // try to identify const F&E. fall to primitive if parms are constant F&E. vector tmpVec; uint16_t parseInfo = 0; parse_item(ifp->arguments()[i], tmpVec, gwi.fatalParseError, parseInfo); if (!gwi.fatalParseError && !(parseInfo & AF_BIT) && tmpVec.size() == 0) continue; } fe = true; break; } } if (!fe) return NULL; } Item_func_opt_neg* inp = (Item_func_opt_neg*)ifp; if (inp->negated) funcName = "not" + funcName; } // @todo non-support function as argument. need to do post process. Assume all support for now fc = new FunctionColumn(); fc->data(funcName); FunctionParm funcParms; SPTP sptp; ClauseType clauseType = gwi.clauseType; if (gwi.clauseType == SELECT || /*gwi.clauseType == HAVING || */gwi.clauseType == GROUP_BY) // select clause { for (uint32_t i = 0; i < ifp->argument_count(); i++) { // group by clause try to see if the arguments are alias if (gwi.clauseType == GROUP_BY && ifp->arguments()[i]->name.length) { uint32_t j = 0; for (; j < gwi.returnedCols.size(); j++) { if (string (ifp->arguments()[i]->name.str) == gwi.returnedCols[j]->alias()) { ReturnedColumn* rc = gwi.returnedCols[j]->clone(); rc->orderPos(j); sptp.reset(new ParseTree(rc)); funcParms.push_back(sptp); break; } } if (j != gwi.returnedCols.size()) continue; } // special handling for function that takes a filter arguments, like if(). // @todo. merge this logic to buildParseTree(). if ((funcName == "if" && i == 0) || funcName == "xor") { // make sure the rcWorkStack is cleaned. gwi.clauseType = WHERE; sptp.reset(buildParseTree((Item_func*)(ifp->arguments()[i]), gwi, nonSupport)); gwi.clauseType = clauseType; if (!sptp) { nonSupport = true; return NULL; } funcParms.push_back(sptp); continue; } // @bug 3039 //if (isPredicateFunction(ifp->arguments()[i], &gwi) || ifp->arguments()[i]->with_subquery()) if (ifp->arguments()[i]->with_subquery()) { nonSupport = true; gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_SUB_EXPRESSION); return NULL; } ReturnedColumn* rc = buildReturnedColumn(ifp->arguments()[i], gwi, nonSupport, pushdownHand); // MCOL-1510 It must be a temp table field, so find the corresponding column. if (!rc && pushdownHand && ifp->arguments()[i]->type() == Item::REF_ITEM) { gwi.fatalParseError = false; rc = buildAggFrmTempField(ifp->arguments()[i], gwi); } if (!rc || nonSupport) { nonSupport = true; return NULL; } sptp.reset(new ParseTree(rc)); funcParms.push_back(sptp); } } else // where clause { stack tmpPtStack; for (int32_t i = ifp->argument_count() - 1; i >= 0; i--) { if (isPredicateFunction((ifp->arguments()[i]), &gwi) && !gwi.ptWorkStack.empty()) { sptp.reset(gwi.ptWorkStack.top()); tmpPtStack.push(sptp); gwi.ptWorkStack.pop(); } else if (!isPredicateFunction((ifp->arguments()[i]), &gwi) && !gwi.rcWorkStack.empty()) { sptp.reset(new ParseTree(gwi.rcWorkStack.top())); tmpPtStack.push(sptp); gwi.rcWorkStack.pop(); } else { nonSupport = true; return NULL; } } while (!tmpPtStack.empty()) { funcParms.push_back(tmpPtStack.top()); tmpPtStack.pop(); } } // the followings are special treatment of some functions if (funcName == "week" && funcParms.size() == 1) { THD* thd = current_thd; sptp.reset(new ParseTree(new ConstantColumn(static_cast(thd->variables.default_week_format)))); funcParms.push_back(sptp); } // add the keyword unit argument for interval function if (funcName == "date_add_interval" || funcName == "extract" || funcName == "timestampdiff") { addIntervalArgs(ifp, funcParms); } // check for unsupported arguments add the keyword unit argument for extract functions if (funcName == "extract") { Item_date_add_interval* idai = (Item_date_add_interval*)ifp; switch (idai->int_type) { case INTERVAL_DAY_MICROSECOND: { nonSupport = true; gwi.fatalParseError = true; Message::Args args; string info = funcName + " with DAY_MICROSECOND parameter"; args.add(info); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); return NULL; } case INTERVAL_HOUR_MICROSECOND: { nonSupport = true; gwi.fatalParseError = true; Message::Args args; string info = funcName + " with HOUR_MICROSECOND parameter"; args.add(info); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); return NULL; } case INTERVAL_MINUTE_MICROSECOND: { nonSupport = true; gwi.fatalParseError = true; Message::Args args; string info = funcName + " with MINUTE_MICROSECOND parameter"; args.add(info); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); return NULL; } default: break; } } // add the keyword unit argument and char length for cast functions if (funcName == "cast_as_char" ) { castCharArgs(ifp, funcParms); } // add the length and scale arguments if (funcName == "decimal_typecast" ) { castDecimalArgs(ifp, funcParms); } // add the type argument if (funcName == "get_format") { castTypeArgs(ifp, funcParms); } // add my_time_zone if (funcName == "unix_timestamp") { #ifndef _MSC_VER time_t tmp_t = 1; struct tm tmp; localtime_r(&tmp_t, &tmp); sptp.reset(new ParseTree(new ConstantColumn(static_cast(tmp.tm_gmtoff), ConstantColumn::NUM))); #else //FIXME: Get GMT offset (in seconds east of GMT) in Windows... sptp.reset(new ParseTree(new ConstantColumn(static_cast(0), ConstantColumn::NUM))); #endif funcParms.push_back(sptp); } // add the default seed to rand function without arguments if (funcName == "rand") { if (funcParms.size() == 0) { sptp.reset(new ParseTree(new ConstantColumn((int64_t)gwi.thd->rand.seed1, ConstantColumn::NUM))); funcParms.push_back(sptp); sptp.reset(new ParseTree(new ConstantColumn((int64_t)gwi.thd->rand.seed2, ConstantColumn::NUM))); funcParms.push_back(sptp); gwi.no_parm_func_list.push_back(fc); } else { ConstantColumn* cc = dynamic_cast(funcParms[0]->data()); if (cc) gwi.no_parm_func_list.push_back(fc); } } if (funcName == "sysdate") { gwi.no_parm_func_list.push_back(fc); } // func name is addtime/subtime in 10.3.9 // note: this means get_time() can now go away in our server fork if ((funcName == "addtime") || (funcName == "subtime")) { int64_t sign = 1; if (funcName == "subtime") { sign = -1; } sptp.reset(new ParseTree(new ConstantColumn(sign))); funcParms.push_back(sptp); } fc->functionName(funcName); fc->functionParms(funcParms); fc->resultType(colType_MysqlToIDB(ifp)); // MySQL give string result type for date function, but has the flag set. // we should set the result type to be datetime for comparision. if (ifp->field_type() == MYSQL_TYPE_DATETIME || ifp->field_type() == MYSQL_TYPE_DATETIME2 || ifp->field_type() == MYSQL_TYPE_TIMESTAMP || ifp->field_type() == MYSQL_TYPE_TIMESTAMP2) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::DATETIME; ct.colWidth = 8; fc->resultType(ct); } else if (ifp->field_type() == MYSQL_TYPE_DATE) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::DATE; ct.colWidth = 4; fc->resultType(ct); } else if (ifp->field_type() == MYSQL_TYPE_TIME) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::TIME; ct.colWidth = 8; fc->resultType(ct); } #if 0 if (is_temporal_type_with_date(ifp->field_type())) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::DATETIME; ct.colWidth = 8; fc->resultType(ct); } if (funcName == "cast_as_date") { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::DATE; ct.colWidth = 4; fc->resultType(ct); } #endif fc->operationType(functor->operationType(funcParms, fc->resultType())); fc->expressionId(ci->expressionId++); } else if (ifp->type() == Item::COND_ITEM || ifp->functype() == Item_func::EQ_FUNC || ifp->functype() == Item_func::NE_FUNC || ifp->functype() == Item_func::LT_FUNC || ifp->functype() == Item_func::LE_FUNC || ifp->functype() == Item_func::GE_FUNC || ifp->functype() == Item_func::GT_FUNC || ifp->functype() == Item_func::LIKE_FUNC || ifp->functype() == Item_func::BETWEEN || ifp->functype() == Item_func::IN_FUNC || ifp->functype() == Item_func::ISNULL_FUNC || ifp->functype() == Item_func::ISNOTNULL_FUNC || ifp->functype() == Item_func::NOT_FUNC || ifp->functype() == Item_func::EQUAL_FUNC) { return NULL; } else { nonSupport = true; gwi.fatalParseError = true; Message::Args args; args.add(funcName); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); return NULL; } // adjust decimal result type according to internalDecimalScale if (!fc) return NULL; if (gwi.internalDecimalScale >= 0 && fc->resultType().colDataType == CalpontSystemCatalog::DECIMAL) { CalpontSystemCatalog::ColType ct = fc->resultType(); ct.scale = gwi.internalDecimalScale; fc->resultType(ct); } if (ifp->name.length) fc->alias(ifp->name.str); // @3391. optimization. try to associate expression ID to the expression on the select list if (gwi.clauseType != SELECT) { for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if ((!fc->alias().empty()) && strcasecmp(fc->alias().c_str(), gwi.returnedCols[i]->alias().c_str()) == 0) fc->expressionId(gwi.returnedCols[i]->expressionId()); } } // For function join. If any argument has non-zero joininfo, set it to the function. fc->setSimpleColumnList(); std::vector simpleColList = fc->simpleColumnList(); for (uint i = 0; i < simpleColList.size(); i++) { if (simpleColList[i]->joinInfo() != 0) { fc->joinInfo(simpleColList[i]->joinInfo()); break; } } return fc; } FunctionColumn* buildCaseFunction(Item_func* item, gp_walk_info& gwi, bool& nonSupport) { if (!(gwi.thd->infinidb_vtable.cal_conn_info)) gwi.thd->infinidb_vtable.cal_conn_info = (void*)(new cal_connection_info()); cal_connection_info* ci = reinterpret_cast(gwi.thd->infinidb_vtable.cal_conn_info); FunctionColumn* fc = new FunctionColumn(); FunctionParm funcParms; SPTP sptp; stack tmpPtStack; FuncExp* funcexp = FuncExp::instance(); string funcName = "case_simple"; if (item->functype() == Item_func::CASE_SEARCHED_FUNC) { funcName = "case_searched"; } /* if (dynamic_cast(item)) { funcName = "case_searched"; }*/ funcParms.reserve(item->argument_count()); // so buildXXXcolumn function will not pop stack. ClauseType realClauseType = gwi.clauseType; gwi.clauseType = SELECT; // We ought to be able to just build from the stack, and would // be able to if there were any way to know which stack had the // next case item. Unfortunately, parameters may have been pushed // onto the ptWorkStack or rcWorkStack or neither, depending on type // and position. We can't tell which at this point, so we // rebuild the item from the arguments directly and then try to // figure what to pop, if anything, in order to sync the stacks. // // MCOL-1341 - With MariaDB 10.2.14 onwards CASE is now in the order: // [case,]when1,when2,...,then1,then2,...[,else] // See server commit bf1ca14ff3f3faa9f7a018097b25aa0f66d068cd for more // information. int32_t arg_offset = 0; if ((item->argument_count() - 1) % 2) { arg_offset = (item->argument_count() - 1) / 2; } else { arg_offset = item->argument_count() / 2; } for (int32_t i = item->argument_count() - 1; i >= 0; i--) { // For case_searched, we know the items for the WHEN clause will // not be ReturnedColumns. We do this separately just to save // some cpu cycles trying to build a ReturnedColumn as below. // Every even numbered arg is a WHEN. In between are the THEN. // An odd number of args indicates an ELSE residing in the last spot. if ((item->functype() == Item_func::CASE_SEARCHED_FUNC) && (i < arg_offset)) { // MCOL-1472 Nested CASE with an ISNULL predicate. We don't want the predicate // to pull off of rcWorkStack, so we set this inCaseStmt flag to tell it // not to. gwi.inCaseStmt = true; sptp.reset(buildParseTree((Item_func*)(item->arguments()[i]), gwi, nonSupport)); gwi.inCaseStmt = false; if (!gwi.ptWorkStack.empty() && *gwi.ptWorkStack.top()->data() == sptp->data()) { gwi.ptWorkStack.pop(); } } else { // First try building a ReturnedColumn. It may or may not succeed // depending on the types involved. There's also little correlation // between buildReturnedColumn and the existance of the item on // rwWorkStack or ptWorkStack. // For example, simple predicates, such as 1=1 or 1=0, land in the // ptWorkStack but other stuff might land in the rwWorkStack ReturnedColumn* parm = buildReturnedColumn(item->arguments()[i], gwi, nonSupport); if (parm) { sptp.reset(new ParseTree(parm)); // We need to pop whichever stack is holding it, if any. if ((!gwi.rcWorkStack.empty()) && *gwi.rcWorkStack.top() == parm) { gwi.rcWorkStack.pop(); } else if (!gwi.ptWorkStack.empty()) { ReturnedColumn* ptrc = dynamic_cast(gwi.ptWorkStack.top()->data()); if (ptrc && *ptrc == *parm) gwi.ptWorkStack.pop(); } } else { sptp.reset(buildParseTree((Item_func*)(item->arguments()[i]), gwi, nonSupport)); // We need to pop whichever stack is holding it, if any. if ((!gwi.ptWorkStack.empty()) && *gwi.ptWorkStack.top()->data() == sptp->data()) { gwi.ptWorkStack.pop(); } else if (!gwi.rcWorkStack.empty()) { // Probably won't happen, but it might have been on the // rcWorkStack all along. ReturnedColumn* ptrc = dynamic_cast(sptp->data()); if (ptrc && *ptrc == *gwi.rcWorkStack.top()) { gwi.rcWorkStack.pop(); } } } } funcParms.insert(funcParms.begin(), sptp); } // recover clause type gwi.clauseType = realClauseType; if (gwi.fatalParseError) { setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return NULL; } Func* functor = funcexp->getFunctor(funcName); fc->resultType(colType_MysqlToIDB(item)); fc->operationType(functor->operationType(funcParms, fc->resultType())); fc->functionName(funcName); fc->functionParms(funcParms); fc->expressionId(ci->expressionId++); // For function join. If any argument has non-zero joininfo, set it to the function. fc->setSimpleColumnList(); std::vector simpleColList = fc->simpleColumnList(); for (uint i = 0; i < simpleColList.size(); i++) { if (simpleColList[i]->joinInfo() != 0) { fc->joinInfo(simpleColList[i]->joinInfo()); break; } } return fc; } ConstantColumn* buildDecimalColumn(Item* item, gp_walk_info& gwi) { Item_decimal* idp = (Item_decimal*)item; IDB_Decimal infinidb_decimal; String val, *str = item->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); ostringstream infinidb_decimal_val; uint32_t i = 0; if (str->ptr()[0] == '+' || str->ptr()[0] == '-') { infinidb_decimal_val << str->ptr()[0]; i = 1; } for (; i < str->length(); i++) { if (str->ptr()[i] == '.') continue; infinidb_decimal_val << str->ptr()[i]; } infinidb_decimal.value = strtoll(infinidb_decimal_val.str().c_str(), 0, 10); if (gwi.internalDecimalScale >= 0 && idp->decimals > (uint)gwi.internalDecimalScale) { infinidb_decimal.scale = gwi.internalDecimalScale; double val = (double)(infinidb_decimal.value / pow((double)10, idp->decimals - gwi.internalDecimalScale)); infinidb_decimal.value = (int64_t)(val > 0 ? val + 0.5 : val - 0.5); } else infinidb_decimal.scale = idp->decimals; infinidb_decimal.precision = idp->max_length - idp->decimals; return new ConstantColumn(valStr, infinidb_decimal); } SimpleColumn* buildSimpleColumn(Item_field* ifp, gp_walk_info& gwi) { if (!gwi.csc) { gwi.csc = CalpontSystemCatalog::makeCalpontSystemCatalog(gwi.sessionid); gwi.csc->identity(CalpontSystemCatalog::FE); } bool isInformationSchema = false; // @bug5523 if (ifp->cached_table && strcmp(ifp->cached_table->db.str, "information_schema") == 0) isInformationSchema = true; // support FRPM subquery. columns from the derived table has no definition if ((!ifp->field || !ifp->db_name || strlen(ifp->db_name) == 0) && !isInformationSchema) return buildSimpleColFromDerivedTable(gwi, ifp); CalpontSystemCatalog::ColType ct; bool infiniDB = true; try { // check foreign engine if (ifp->cached_table && ifp->cached_table->table) infiniDB = isInfiniDB(ifp->cached_table->table); // @bug4509. ifp->cached_table could be null for myisam sometimes else if (ifp->field && ifp->field->table) infiniDB = isInfiniDB(ifp->field->table); if (infiniDB) { ct = gwi.csc->colType( gwi.csc->lookupOID(make_tcn(ifp->db_name, bestTableName(ifp), ifp->field_name.str))); } else { ct = colType_MysqlToIDB(ifp); } } catch (std::exception& ex ) { gwi.fatalParseError = true; gwi.parseErrorText = ex.what(); return NULL; } SimpleColumn* sc = NULL; switch (ct.colDataType) { case CalpontSystemCatalog::TINYINT: if (ct.scale == 0) sc = new SimpleColumn_INT<1>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); else { sc = new SimpleColumn_Decimal<1>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); ct.colDataType = CalpontSystemCatalog::DECIMAL; } break; case CalpontSystemCatalog::SMALLINT: if (ct.scale == 0) sc = new SimpleColumn_INT<2>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); else { sc = new SimpleColumn_Decimal<2>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); ct.colDataType = CalpontSystemCatalog::DECIMAL; } break; case CalpontSystemCatalog::INT: case CalpontSystemCatalog::MEDINT: if (ct.scale == 0) sc = new SimpleColumn_INT<4>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); else { sc = new SimpleColumn_Decimal<4>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); ct.colDataType = CalpontSystemCatalog::DECIMAL; } break; case CalpontSystemCatalog::BIGINT: if (ct.scale == 0) sc = new SimpleColumn_INT<8>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); else { sc = new SimpleColumn_Decimal<8>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); ct.colDataType = CalpontSystemCatalog::DECIMAL; } break; case CalpontSystemCatalog::UTINYINT: sc = new SimpleColumn_UINT<1>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); break; case CalpontSystemCatalog::USMALLINT: sc = new SimpleColumn_UINT<2>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); break; case CalpontSystemCatalog::UINT: case CalpontSystemCatalog::UMEDINT: sc = new SimpleColumn_UINT<4>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); break; case CalpontSystemCatalog::UBIGINT: sc = new SimpleColumn_UINT<8>(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); break; default: sc = new SimpleColumn(ifp->db_name, bestTableName(ifp), ifp->field_name.str, infiniDB, gwi.sessionid); } sc->resultType(ct); string tbname(ifp->table_name); if (isInformationSchema) { sc->schemaName("information_schema"); sc->tableName(tbname); } sc->tableAlias(lower(tbname)); // view name sc->viewName(lower(getViewName(ifp->cached_table))); sc->alias(ifp->name.str); sc->isInfiniDB(infiniDB); if (!infiniDB && ifp->field) sc->oid(ifp->field->field_index + 1); // ExeMgr requires offset started from 1 if (ifp->depended_from) { sc->joinInfo(sc->joinInfo() | JOIN_CORRELATED); if (gwi.subQuery) gwi.subQuery->correlated(true); // for error out non-support select filter case (comparison outside semi join tables) gwi.correlatedTbNameVec.push_back(make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias())); // imply semi for scalar for now. if (gwi.subSelectType == CalpontSelectExecutionPlan::SINGLEROW_SUBS) sc->joinInfo(sc->joinInfo() | JOIN_SCALAR | JOIN_SEMI); if (gwi.subSelectType == CalpontSelectExecutionPlan::SELECT_SUBS) sc->joinInfo(sc->joinInfo() | JOIN_SCALAR | JOIN_OUTER_SELECT); } return sc; } ParseTree* buildParseTree(Item_func* item, gp_walk_info& gwi, bool& nonSupport) { ParseTree* pt = 0; Item_cond* icp = (Item_cond*)item; #ifdef DEBUG_WALK_COND // debug cerr << "Build Parsetree: " << endl; icp->traverse_cond(debug_walk, &gwi, Item::POSTFIX); #endif //@bug5044. PPSTFIX walking should always be treated as WHERE clause filter ClauseType clauseType = gwi.clauseType; gwi.clauseType = WHERE; icp->traverse_cond(gp_walk, &gwi, Item::POSTFIX); gwi.clauseType = clauseType; if (gwi.fatalParseError) return NULL; // bug 2840. if the filter/function is constant, result is in rcWorkStack if (!gwi.ptWorkStack.empty()) { pt = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); } else if (!gwi.rcWorkStack.empty()) { pt = new ParseTree (gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } return pt; } ReturnedColumn* buildAggregateColumn(Item* item, gp_walk_info& gwi) { // MCOL-1201 For UDAnF multiple parameters vector selCols; vector orderCols; bool bIsConst = false; if (!(gwi.thd->infinidb_vtable.cal_conn_info)) gwi.thd->infinidb_vtable.cal_conn_info = (void*)(new cal_connection_info()); cal_connection_info* ci = reinterpret_cast(gwi.thd->infinidb_vtable.cal_conn_info); Item_sum* isp = reinterpret_cast(item); Item** sfitempp = isp->get_orig_args(); // Item** sfitempp = isp->arguments(); SRCP parm; // @bug4756 if (gwi.clauseType == SELECT) gwi.aggOnSelect = true; // N.B. argument_count() is the # of formal parms to the agg fcn. InifniDB only supports 1 argument // TODO: Support more than one parm #if 0 if (isp->argument_count() != 1 && isp->sum_func() != Item_sum::GROUP_CONCAT_FUNC && isp->sum_func() != Item_sum::UDF_SUM_FUNC) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_MUL_ARG_AGG); return NULL; } #endif AggregateColumn* ac = NULL; if (isp->sum_func() == Item_sum::GROUP_CONCAT_FUNC) { ac = new GroupConcatColumn(gwi.sessionid); } else if (isp->sum_func() == Item_sum::UDF_SUM_FUNC) { ac = new UDAFColumn(gwi.sessionid); } else { ac = new AggregateColumn(gwi.sessionid); } if (isp->name.length) ac->alias(isp->name.str); if ((setAggOp(ac, isp))) { gwi.fatalParseError = true; gwi.parseErrorText = "Non supported aggregate type on the select clause"; if (ac) delete ac; return NULL; } try { // special parsing for group_concat if (isp->sum_func() == Item_sum::GROUP_CONCAT_FUNC) { Item_func_group_concat* gc = (Item_func_group_concat*)isp; vector orderCols; RowColumn* rowCol = new RowColumn(); vector selCols; uint32_t select_ctn = gc->count_field(); ReturnedColumn* rc = NULL; for (uint32_t i = 0; i < select_ctn; i++) { rc = buildReturnedColumn(sfitempp[i], gwi, gwi.fatalParseError); if (!rc || gwi.fatalParseError) { if (ac) delete ac; return NULL; } selCols.push_back(SRCP(rc)); } ORDER** order_item, **end; for (order_item = gc->get_order(), end = order_item + gc->order_field(); order_item < end; order_item++) { Item* ord_col = *(*order_item)->item; if (ord_col->type() == Item::INT_ITEM) { Item_int* id = (Item_int*)ord_col; if (id->val_int() > (int)selCols.size()) { gwi.fatalParseError = true; if (ac) delete ac; return NULL; } rc = selCols[id->val_int() - 1]->clone(); rc->orderPos(id->val_int() - 1); } else { rc = buildReturnedColumn(ord_col, gwi, gwi.fatalParseError); if (!rc || gwi.fatalParseError) { if (ac) delete ac; return NULL; } } // 10.2 TODO: direction is now a tri-state flag rc->asc((*order_item)->direction == ORDER::ORDER_ASC ? true : false); orderCols.push_back(SRCP(rc)); } rowCol->columnVec(selCols); (dynamic_cast(ac))->orderCols(orderCols); parm.reset(rowCol); ac->aggParms().push_back(parm); if (gc->str_separator()) { string separator; separator.assign(gc->str_separator()->ptr(), gc->str_separator()->length()); (dynamic_cast(ac))->separator(separator); } } else { for (uint32_t i = 0; i < isp->argument_count(); i++) { Item* sfitemp = sfitempp[i]; Item::Type sfitype = sfitemp->type(); switch (sfitype) { case Item::FIELD_ITEM: { Item_field* ifp = reinterpret_cast(sfitemp); SimpleColumn* sc = buildSimpleColumn(ifp, gwi); if (!sc) { gwi.fatalParseError = true; break; } parm.reset(sc); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), parm)); TABLE_LIST* tmp = (ifp->cached_table ? ifp->cached_table : 0); gwi.tableMap[make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias(), sc->isInfiniDB())] = make_pair(1, tmp); break; } case Item::INT_ITEM: case Item::STRING_ITEM: case Item::REAL_ITEM: case Item::DECIMAL_ITEM: { // treat as count(*) if (ac->aggOp() == AggregateColumn::COUNT) ac->aggOp(AggregateColumn::COUNT_ASTERISK); parm.reset(buildReturnedColumn(sfitemp, gwi, gwi.fatalParseError)); ac->constCol(parm); bIsConst = true; break; } case Item::NULL_ITEM: { parm.reset(new ConstantColumn("", ConstantColumn::NULLDATA)); ac->constCol(SRCP(buildReturnedColumn(sfitemp, gwi, gwi.fatalParseError))); break; } case Item::FUNC_ITEM: { Item_func* ifp = (Item_func*)sfitemp; ReturnedColumn* rc = 0; // check count(1+1) case vector tmpVec; uint16_t parseInfo = 0; parse_item(ifp, tmpVec, gwi.fatalParseError, parseInfo); if (parseInfo & SUB_BIT) { gwi.fatalParseError = true; break; } else if (!gwi.fatalParseError && !(parseInfo & AGG_BIT) && !(parseInfo & AF_BIT) && tmpVec.size() == 0) { rc = buildFunctionColumn(ifp, gwi, gwi.fatalParseError); FunctionColumn* fc = dynamic_cast(rc); if ((fc && fc->functionParms().empty()) || !fc) { //ac->aggOp(AggregateColumn::COUNT_ASTERISK); ReturnedColumn* rc = buildReturnedColumn(sfitemp, gwi, gwi.fatalParseError); if (dynamic_cast(rc)) { //@bug5229. handle constant function on aggregate argument ac->constCol(SRCP(rc)); break; } } } // MySQL carelessly allows correlated aggregate function on the WHERE clause. // Here is the work around to deal with that inconsistence. // e.g., SELECT (SELECT t.c FROM t1 AS t WHERE t.b=MAX(t1.b + 0)) FROM t1; ClauseType clauseType = gwi.clauseType; if (gwi.clauseType == WHERE) gwi.clauseType = HAVING; // @bug 3603. for cases like max(rand()). try to build function first. if (!rc) rc = buildFunctionColumn(ifp, gwi, gwi.fatalParseError); parm.reset(rc); gwi.clauseType = clauseType; if (gwi.fatalParseError) break; break; } case Item::REF_ITEM: { ReturnedColumn* rc = buildReturnedColumn(sfitemp, gwi, gwi.fatalParseError); if (rc) { parm.reset(rc); break; } } default: { gwi.fatalParseError = true; //gwi.parseErrorText = "Non-supported Item in Aggregate function"; } } if (gwi.fatalParseError) { if (gwi.parseErrorText.empty()) { Message::Args args; if (item->name.length) args.add(item->name.str); else args.add(""); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_AGG_ARGS, args); } if (ac) delete ac; return NULL; } if (parm) { // MCOL-1201 multi-argument aggregate ac->aggParms().push_back(parm); } } } // Get result type // Modified for MCOL-1201 multi-argument aggregate if (!bIsConst && ac->aggParms().size() > 0) { // These are all one parm functions, so we can safely // use the first parm for result type. parm = ac->aggParms()[0]; if (isp->sum_func() == Item_sum::AVG_FUNC || isp->sum_func() == Item_sum::AVG_DISTINCT_FUNC) { CalpontSystemCatalog::ColType ct = parm->resultType(); ct.colDataType = CalpontSystemCatalog::LONGDOUBLE; ct.colWidth = sizeof(long double); ct.scale += 4; ct.precision = -1; ac->resultType(ct); } else if (isp->sum_func() == Item_sum::COUNT_FUNC || isp->sum_func() == Item_sum::COUNT_DISTINCT_FUNC) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::BIGINT; ct.colWidth = 8; ct.scale = parm->resultType().scale; ac->resultType(ct); } else if (isp->sum_func() == Item_sum::SUM_FUNC || isp->sum_func() == Item_sum::SUM_DISTINCT_FUNC) { CalpontSystemCatalog::ColType ct = parm->resultType(); ct.colDataType = CalpontSystemCatalog::LONGDOUBLE; ct.colWidth = sizeof(long double); ct.precision = -1; ac->resultType(ct); } else if (isp->sum_func() == Item_sum::STD_FUNC || isp->sum_func() == Item_sum::VARIANCE_FUNC) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::DOUBLE; ct.colWidth = 8; ct.scale = 0; ac->resultType(ct); } else if (isp->sum_func() == Item_sum::SUM_BIT_FUNC) { CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::BIGINT; ct.colWidth = 8; ct.scale = 0; ct.precision = -16; // borrowed to indicate skip null value check on connector ac->resultType(ct); } else if (isp->sum_func() == Item_sum::GROUP_CONCAT_FUNC) { //Item_func_group_concat* gc = (Item_func_group_concat*)isp; CalpontSystemCatalog::ColType ct; ct.colDataType = CalpontSystemCatalog::VARCHAR; ct.colWidth = isp->max_length; ct.precision = 0; ac->resultType(ct); } else { // UDAF result type will be set below. ac->resultType(parm->resultType()); } } else { ac->resultType(colType_MysqlToIDB(isp)); } // adjust decimal result type according to internalDecimalScale if (gwi.internalDecimalScale >= 0 && ac->resultType().colDataType == CalpontSystemCatalog::DECIMAL) { CalpontSystemCatalog::ColType ct = ac->resultType(); ct.scale = gwi.internalDecimalScale; ac->resultType(ct); } // check for same aggregate on the select list ac->expressionId(ci->expressionId++); if (gwi.clauseType != SELECT) { for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if (*ac == gwi.returnedCols[i].get()) ac->expressionId(gwi.returnedCols[i]->expressionId()); } } // @bug5977 @note Temporary fix to avoid mysqld crash. The permanent fix will // be applied in ExeMgr. When the ExeMgr fix is available, this checking // will be taken out. if (isp->sum_func() != Item_sum::UDF_SUM_FUNC) { if (ac->constCol() && gwi.tbList.empty() && gwi.derivedTbList.empty()) { gwi.fatalParseError = true; gwi.parseErrorText = "No project column found for aggregate function"; if (ac) delete ac; return NULL; } else if (ac->constCol()) { gwi.count_asterisk_list.push_back(ac); } } // For UDAF, populate the context and call the UDAF init() function. // The return type is (should be) set in context by init(). if (isp->sum_func() == Item_sum::UDF_SUM_FUNC) { UDAFColumn* udafc = dynamic_cast(ac); if (udafc) { mcsv1Context& context = udafc->getContext(); context.setName(isp->func_name()); // Set up the return type defaults for the call to init() context.setResultType(udafc->resultType().colDataType); context.setColWidth(udafc->resultType().colWidth); context.setScale(udafc->resultType().scale); context.setPrecision(udafc->resultType().precision); context.setParamCount(udafc->aggParms().size()); ColumnDatum colType; ColumnDatum colTypes[udafc->aggParms().size()]; // Build the column type vector. // Modified for MCOL-1201 multi-argument aggregate for (uint32_t i = 0; i < udafc->aggParms().size(); ++i) { const execplan::CalpontSystemCatalog::ColType& resultType = udafc->aggParms()[i]->resultType(); colType.dataType = resultType.colDataType; colType.precision = resultType.precision; colType.scale = resultType.scale; colTypes[i] = colType; } // Call the user supplied init() mcsv1sdk::mcsv1_UDAF* udaf = context.getFunction(); if (!udaf) { gwi.fatalParseError = true; gwi.parseErrorText = "Aggregate Function " + context.getName() + " doesn't exist in the ColumnStore engine"; if (ac) delete ac; return NULL; } if (udaf->init(&context, colTypes) == mcsv1_UDAF::ERROR) { gwi.fatalParseError = true; gwi.parseErrorText = udafc->getContext().getErrorMessage(); if (ac) delete ac; return NULL; } // UDAF_OVER_REQUIRED means that this function is for Window // Function only. Reject it here in aggregate land. if (udafc->getContext().getRunFlag(UDAF_OVER_REQUIRED)) { gwi.fatalParseError = true; gwi.parseErrorText = logging::IDBErrorInfo::instance()->errorMsg(logging::ERR_WINDOW_FUNC_ONLY, context.getName()); if (ac) delete ac; return NULL; } // Set the return type as set in init() CalpontSystemCatalog::ColType ct; ct.colDataType = context.getResultType(); ct.colWidth = context.getColWidth(); ct.scale = context.getScale(); ct.precision = context.getPrecision(); udafc->resultType(ct); } } } catch (std::logic_error e) { gwi.fatalParseError = true; gwi.parseErrorText = "error building Aggregate Function: "; gwi.parseErrorText += e.what(); if (ac) delete ac; return NULL; } catch (...) { gwi.fatalParseError = true; gwi.parseErrorText = "error building Aggregate Function: Unspecified exception"; if (ac) delete ac; return NULL; } return ac; } void addIntervalArgs(Item_func* ifp, FunctionParm& functionParms) { string funcName = ifp->func_name(); int interval_type = -1; if (funcName == "date_add_interval") interval_type = ((Item_date_add_interval*)ifp)->int_type; else if (funcName == "timestampdiff") interval_type = ((Item_func_timestamp_diff*)ifp)->int_type; else if (funcName == "extract") interval_type = ((Item_extract*)ifp)->int_type; functionParms.push_back(getIntervalType(interval_type)); SPTP sptp; if (funcName == "date_add_interval") { if (((Item_date_add_interval*)ifp)->date_sub_interval) { sptp.reset(new ParseTree(new ConstantColumn((int64_t)OP_SUB))); functionParms.push_back(sptp); } else { sptp.reset(new ParseTree(new ConstantColumn((int64_t)OP_ADD))); functionParms.push_back(sptp); } } } SPTP getIntervalType(int interval_type) { SPTP sptp; sptp.reset(new ParseTree(new ConstantColumn((int64_t)interval_type))); return sptp; } void castCharArgs(Item_func* ifp, FunctionParm& functionParms) { Item_char_typecast* idai = (Item_char_typecast*)ifp; SPTP sptp; sptp.reset(new ParseTree(new ConstantColumn((int64_t)idai->castLength()))); functionParms.push_back(sptp); } void castDecimalArgs(Item_func* ifp, FunctionParm& functionParms) { Item_decimal_typecast* idai = (Item_decimal_typecast*)ifp; SPTP sptp; sptp.reset(new ParseTree(new ConstantColumn((int64_t)idai->decimals))); functionParms.push_back(sptp); // max length including sign and/or decimal points if (idai->decimals == 0) sptp.reset(new ParseTree(new ConstantColumn((int64_t)idai->max_length - 1))); else sptp.reset(new ParseTree(new ConstantColumn((int64_t)idai->max_length - 2))); functionParms.push_back(sptp); } void castTypeArgs(Item_func* ifp, FunctionParm& functionParms) { Item_func_get_format* get_format = (Item_func_get_format*)ifp; SPTP sptp; if (get_format->type == MYSQL_TIMESTAMP_DATE) sptp.reset(new ParseTree(new ConstantColumn("DATE"))); else sptp.reset(new ParseTree(new ConstantColumn("DATETIME"))); functionParms.push_back(sptp); } void gp_walk(const Item* item, void* arg) { gp_walk_info* gwip = reinterpret_cast(arg); idbassert(gwip); //Bailout... if (gwip->fatalParseError) return; RecursionCounter r(gwip); // Increments and auto-decrements upon exit. Item::Type itype = item->type(); // Allow to process XOR(which is Item_func) like other logical operators (which are Item_cond) if (itype == Item::FUNC_ITEM && ((Item_func*)item)->functype() == Item_func::XOR_FUNC ) itype = Item::COND_ITEM; switch (itype) { case Item::CACHE_ITEM: { // The item or condition is cached as per MariaDB server view but // for InfiniDB it need to be parsed and executed. // MCOL-1188 and MCOL-1029 Item* orig_item = ((Item_cache*)item)->get_example(); orig_item->traverse_cond(gp_walk, gwip, Item::POSTFIX); break; } case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)item; if (ifp) { SimpleColumn* scp = buildSimpleColumn(ifp, *gwip); if (!scp) break; string aliasTableName(scp->tableAlias()); scp->tableAlias(lower(aliasTableName)); gwip->rcWorkStack.push(scp->clone()); //gwip->rcWorkStack.push(scp); boost::shared_ptr scsp(scp); //boost::shared_ptr scsp(scp->clone()); gwip->scsp = scsp; gwip->funcName.clear(); gwip->columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), scsp)); //@bug4636 take where clause column as dummy projection column, but only on local column. // varbinary aggregate is not supported yet, so rule it out if (!((scp->joinInfo() & JOIN_CORRELATED) || scp->colType().colDataType == CalpontSystemCatalog::VARBINARY)) { TABLE_LIST* tmp = (ifp->cached_table ? ifp->cached_table : 0); gwip->tableMap[make_aliastable(scp->schemaName(), scp->tableName(), scp->tableAlias(), scp->isInfiniDB())] = make_pair(1, tmp); } } break; } case Item::INT_ITEM: { Item_int* iip = (Item_int*)item; gwip->rcWorkStack.push(buildReturnedColumn(iip, *gwip, gwip->fatalParseError)); break; } case Item::STRING_ITEM: { Item_string* isp = (Item_string*)item; if (isp) { if (isp->result_type() == STRING_RESULT) { String val, *str = isp->val_str(&val); string cval; if (str->ptr()) { cval.assign(str->ptr(), str->length()); } gwip->rcWorkStack.push(new ConstantColumn(cval)); break; } gwip->rcWorkStack.push(buildReturnedColumn(isp, *gwip, gwip->fatalParseError)); } break; } case Item::REAL_ITEM: { Item_float* ifp = (Item_float*)item; gwip->rcWorkStack.push(buildReturnedColumn(ifp, *gwip, gwip->fatalParseError)); break; } case Item::DECIMAL_ITEM: { Item_decimal* idp = (Item_decimal*)item; gwip->rcWorkStack.push(buildReturnedColumn(idp, *gwip, gwip->fatalParseError)); break; } case Item::VARBIN_ITEM: { Item_hex_string* hdp = (Item_hex_string*)item; gwip->rcWorkStack.push(buildReturnedColumn(hdp, *gwip, gwip->fatalParseError)); break; } case Item::NULL_ITEM: { if (gwip->condPush) { // push noop for unhandled item SimpleColumn* rc = new SimpleColumn("noop"); gwip->rcWorkStack.push(rc); break; } gwip->rcWorkStack.push(new ConstantColumn("", ConstantColumn::NULLDATA)); break; } case Item::FUNC_ITEM: { Item_func* ifp = (Item_func*)item; string funcName = ifp->func_name(); if (!gwip->condPush) { if (ifp->with_subquery() || funcName == "") { buildSubselectFunc(ifp, gwip); return; } if (ifp->argument_count() > 0 && ifp->arguments()) { for (uint32_t i = 0; i < ifp->argument_count(); i++) { if (ifp->arguments()[i]->type() == Item::SUBSELECT_ITEM) { // This is probably NOT IN subquery with derived table in it. // for some reason, MySQL has not fully optimized the plan at this point. // noop here, and eventually MySQL will continue its optimization and get // to rnd_init again. if (ifp->functype() == Item_func::NOT_FUNC) return; buildSubselectFunc(ifp, gwip); return; } } } if (ifp->functype() == Item_func::TRIG_COND_FUNC && gwip->subQuery) { gwip->subQuery->handleFunc(gwip, ifp); break; } // having clause null function added by MySQL if (ifp->functype() == Item_func::ISNOTNULLTEST_FUNC) { // @bug 4215. remove the argument in rcWorkStack. if (!gwip->rcWorkStack.empty()) gwip->rcWorkStack.pop(); break; } } // try to evaluate const F&E vector tmpVec; uint16_t parseInfo = 0; parse_item(ifp, tmpVec, gwip->fatalParseError, parseInfo); // table mode takes only one table filter if (gwip->condPush) { set tableSet; for (uint32_t i = 0; i < tmpVec.size(); i++) { if (tmpVec[i]->table_name) tableSet.insert(tmpVec[i]->table_name); } if (tableSet.size() > 1) break; } if (!gwip->fatalParseError && !(parseInfo & AGG_BIT) && !(parseInfo & SUB_BIT) && !nonConstFunc(ifp) && !(parseInfo & AF_BIT) && tmpVec.size() == 0 && ifp->functype() != Item_func::MULT_EQUAL_FUNC) { String val, *str = ifp->val_str(&val); string valStr; if (str) valStr.assign(str->ptr(), str->length()); ConstantColumn* cc = NULL; if (!str) //@ bug 2844 check whether parameter is defined { cc = new ConstantColumn("", ConstantColumn::NULLDATA); } else if (ifp->result_type() == STRING_RESULT) { cc = new ConstantColumn(valStr, ConstantColumn::LITERAL); } else if (ifp->result_type() == DECIMAL_RESULT) { cc = buildDecimalColumn(ifp, *gwip); } else { cc = new ConstantColumn(valStr, ConstantColumn::NUM); cc->resultType(colType_MysqlToIDB(item)); } for (uint32_t i = 0; i < ifp->argument_count() && !gwip->rcWorkStack.empty(); i++) { gwip->rcWorkStack.pop(); } // bug 3137. If filter constant like 1=0, put it to ptWorkStack // MariaDB bug 750. Breaks if compare is an argument to a function. // if ((int32_t)gwip->rcWorkStack.size() <= (gwip->rcBookMarkStack.empty() ? 0 : gwip->rcBookMarkStack.top()) // && isPredicateFunction(ifp, gwip)) if (isPredicateFunction(ifp, gwip)) gwip->ptWorkStack.push(new ParseTree(cc)); else gwip->rcWorkStack.push(cc); if (str) IDEBUG( cerr << "Const F&E " << item->full_name() << " evaluate: " << valStr << endl ); break; } ReturnedColumn* rc = NULL; // @bug4488. Process function for table mode also, not just vtable mode. rc = buildFunctionColumn(ifp, *gwip, gwip->fatalParseError); if (gwip->fatalParseError) { if (gwip->clauseType == SELECT) return; // @bug 2585 if (gwip->parseErrorText.empty()) { Message::Args args; args.add(funcName); gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); } return; } // predicate operators fall in the old path if (rc) { // @bug 2383. For some reason func_name() for "in" gives " IN " always if (funcName == "between" || funcName == "in" || funcName == " IN ") gwip->ptWorkStack.push(new ParseTree(rc)); else gwip->rcWorkStack.push(rc); } else { // push to pt or rc stack is handled inside the function buildPredicateItem(ifp, gwip); } break; } case Item::SUM_FUNC_ITEM: { Item_sum* isp = (Item_sum*)item; ReturnedColumn* rc = buildAggregateColumn(isp, *gwip); if (rc) gwip->rcWorkStack.push(rc); break; } case Item::COND_ITEM: { // All logical functions are handled here, most of them are Item_cond, // but XOR (it is Item_func_boolean2) Item_func* func = (Item_func*)item; enum Item_func::Functype ftype = func->functype(); bool isOr = (ftype == Item_func::COND_OR_FUNC); bool isXor = (ftype == Item_func::XOR_FUNC); List* argumentList; List xorArgumentList; if (isXor) { for (unsigned i = 0; i < func->argument_count(); i++) { xorArgumentList.push_back(func->arguments()[i]); } argumentList = &xorArgumentList; } else { argumentList = ((Item_cond*)item)->argument_list(); } // @bug2932. if ptWorkStack contains less items than the condition's arguments, // the missing one should be in the rcWorkStack, unless the it's subselect. // @todo need to figure out a way to combine these two stacks while walking. //if (gwip->ptWorkStack.size() < icp->argument_list()->elements) { List_iterator_fast li(*argumentList); while (Item* it = li++) { //@bug3495, @bug5865 error out non-supported OR with correlated subquery if (isOr) { vector fieldVec; uint16_t parseInfo = 0; parse_item(it, fieldVec, gwip->fatalParseError, parseInfo); if (parseInfo & CORRELATED) { gwip->fatalParseError = true; gwip->parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_CORRELATED_SUB_OR); return; } } if ((it->type() == Item::FIELD_ITEM || it->type() == Item::INT_ITEM || it->type() == Item::DECIMAL_ITEM || it->type() == Item::STRING_ITEM || it->type() == Item::REAL_ITEM || it->type() == Item::NULL_ITEM || (it->type() == Item::FUNC_ITEM && !isPredicateFunction(it, gwip))) && !gwip->rcWorkStack.empty() ) { gwip->ptWorkStack.push(new ParseTree(gwip->rcWorkStack.top())); gwip->rcWorkStack.pop(); } } } // @bug1603. MySQL's filter tree is a multi-tree grouped by operator. So more than // two filters saved on the stack so far might belong to this operator. uint32_t leftInStack = gwip->ptWorkStack.size() - argumentList->elements + 1; while (true) { if (gwip->ptWorkStack.size() < 2) break; ParseTree* lhs = gwip->ptWorkStack.top(); gwip->ptWorkStack.pop(); SimpleFilter* lsf = dynamic_cast(lhs->data()); if (lsf && lsf->op()->data() == "noop") { if (isOr) { gwip->parseErrorText = "Unhandled item in WHERE or HAVING clause"; gwip->fatalParseError = true; break; } else continue; } ParseTree* rhs = gwip->ptWorkStack.top(); gwip->ptWorkStack.pop(); SimpleFilter* rsf = dynamic_cast(rhs->data()); if (rsf && rsf->op()->data() == "noop") { if (isOr) { gwip->parseErrorText = "Unhandled item in WHERE or HAVING clause"; gwip->fatalParseError = true; break; } else { gwip->ptWorkStack.push(lhs); continue; } } Operator* op = new LogicOperator(func->func_name()); ParseTree* ptp = new ParseTree(op); ptp->left(lhs); ptp->right(rhs); gwip->ptWorkStack.push(ptp); if (gwip->ptWorkStack.size() == leftInStack) break; } // special handling for subquery with aggregate. MySQL adds isnull function to the selected // column. InfiniDB will remove it and set nullmatch flag if it's NOT_IN sub. // @todo need more checking here to make sure it's not a user input OR operator if (isOr && gwip->subQuery) gwip->subQuery->handleFunc(gwip, func); break; } case Item::REF_ITEM: { Item* col = *(((Item_ref*)item)->ref); ReturnedColumn* rc = NULL; // ref item is not pre-walked. force clause type to SELECT ClauseType clauseType = gwip->clauseType; gwip->clauseType = SELECT; if (col->type() != Item::COND_ITEM) { rc = buildReturnedColumn(col, *gwip, gwip->fatalParseError); if ( col->type() == Item::FIELD_ITEM ) gwip->fatalParseError = false; } SimpleColumn* sc = dynamic_cast(rc); if (sc) { boost::shared_ptr scsp(sc->clone()); gwip->scsp = scsp; } bool cando = true; gwip->clauseType = clauseType; if (rc) { if (((Item_ref*)item)->depended_from) { rc->joinInfo(rc->joinInfo() | JOIN_CORRELATED); if (gwip->subQuery) gwip->subQuery->correlated(true); SimpleColumn* scp = dynamic_cast(rc); if (scp) gwip->correlatedTbNameVec.push_back(make_aliastable(scp->schemaName(), scp->tableName(), scp->tableAlias())); if (gwip->subSelectType == CalpontSelectExecutionPlan::SINGLEROW_SUBS) rc->joinInfo(rc->joinInfo() | JOIN_SCALAR | JOIN_SEMI); if (gwip->subSelectType == CalpontSelectExecutionPlan::SELECT_SUBS) rc->joinInfo(rc->joinInfo() | JOIN_SCALAR | JOIN_OUTER_SELECT); } gwip->rcWorkStack.push(rc); } else if (col->type() == Item::FUNC_ITEM) { // sometimes mysql treat having filter items inconsistently. In such cases, // which are always predicate operator, the function (gp_key>3) comes in as // one item. Item_func* ifp = (Item_func*)col; for (uint32_t i = 0; i < ifp->argument_count(); i++) { ReturnedColumn* operand = NULL; if (ifp->arguments()[i]->type() == Item::REF_ITEM) { Item* op = *(((Item_ref*)ifp->arguments()[i])->ref); operand = buildReturnedColumn(op, *gwip, gwip->fatalParseError); } else operand = buildReturnedColumn(ifp->arguments()[i], *gwip, gwip->fatalParseError); if (operand) gwip->rcWorkStack.push(operand); else { cando = false; break; } } if (cando) buildPredicateItem(ifp, gwip); } else if (col->type() == Item::COND_ITEM) { Item_func* ifp = (Item_func*)col; gwip->ptWorkStack.push(buildParseTree(ifp, *gwip, gwip->fatalParseError)); } else if (col->type() == Item::FIELD_ITEM && gwip->clauseType == HAVING) { ReturnedColumn* rc = buildAggFrmTempField(const_cast(item), *gwip); if (rc) gwip->rcWorkStack.push(rc); break; } else cando = false; if (!cando) { ostringstream oss; oss << "Unhandled Item type: " << item->type(); gwip->parseErrorText = oss.str(); gwip->fatalParseError = true; } break; } case Item::SUBSELECT_ITEM: { if (gwip->condPush) // table mode break; Item_subselect* sub = (Item_subselect*)item; if (sub->substype() == Item_subselect::EXISTS_SUBS) { SubQuery* orig = gwip->subQuery; ExistsSub* existsSub = new ExistsSub(*gwip, sub); gwip->hasSubSelect = true; gwip->subQuery = existsSub; gwip->ptWorkStack.push(existsSub->transform()); current_thd->infinidb_vtable.isUnion = true; // only temp. bypass the 2nd phase. // recover original gwip->subQuery = orig; gwip->lastSub = existsSub; } else if (sub->substype() == Item_subselect::IN_SUBS) { if (!((Item_in_subselect*)sub)->getOptimizer() && gwip->thd->derived_tables_processing) { ostringstream oss; oss << "Invalid In_optimizer: " << item->type(); gwip->parseErrorText = oss.str(); gwip->fatalParseError = true; break; } } // store a dummy subselect object. the transform is handled in item_func. SubSelect* subselect = new SubSelect(); gwip->rcWorkStack.push(subselect); break; } case Item::ROW_ITEM: { Item_row* row = (Item_row*)item; RowColumn* rowCol = new RowColumn(); vector cols; // temp change clause type because the elements of row column are not walked yet gwip->clauseType = SELECT; for (uint32_t i = 0; i < row->cols(); i++) cols.push_back(SRCP(buildReturnedColumn(row->element_index(i), *gwip, gwip->fatalParseError))); gwip->clauseType = WHERE; rowCol->columnVec(cols); gwip->rcWorkStack.push(rowCol); break; } case Item::EXPR_CACHE_ITEM: { ((Item_cache_wrapper*)item)->get_orig_item()->traverse_cond(gp_walk, arg, Item::POSTFIX); break; } case Item::DATE_ITEM: { Item_temporal_literal* itp = (Item_temporal_literal*)item; gwip->rcWorkStack.push(buildReturnedColumn(itp, *gwip, gwip->fatalParseError)); break; } case Item::WINDOW_FUNC_ITEM: { gwip->hasWindowFunc = true; Item_window_func* ifa = (Item_window_func*)item; ReturnedColumn* af = buildWindowFunctionColumn(ifa, *gwip, gwip->fatalParseError); if (af) gwip->rcWorkStack.push(af); break; } case Item::COPY_STR_ITEM: printf("********** received COPY_STR_ITEM *********\n"); break; case Item::FIELD_AVG_ITEM: printf("********** received FIELD_AVG_ITEM *********\n"); break; case Item::DEFAULT_VALUE_ITEM: printf("********** received DEFAULT_VALUE_ITEM *********\n"); break; case Item::PROC_ITEM: printf("********** received PROC_ITEM *********\n"); break; case Item::FIELD_STD_ITEM: printf("********** received FIELD_STD_ITEM *********\n"); break; case Item::FIELD_VARIANCE_ITEM: printf("********** received FIELD_VARIANCE_ITEM *********\n"); break; case Item::INSERT_VALUE_ITEM: printf("********** received INSERT_VALUE_ITEM *********\n"); break; case Item::Item::TYPE_HOLDER: printf("********** received TYPE_HOLDER *********\n"); break; case Item::PARAM_ITEM: printf("********** received PARAM_ITEM *********\n"); break; case Item::TRIGGER_FIELD_ITEM: printf("********** received TRIGGER_FIELD_ITEM *********\n"); break; case Item::XPATH_NODESET: printf("********** received XPATH_NODESET *********\n"); break; case Item::XPATH_NODESET_CMP: printf("********** received XPATH_NODESET_CMP *********\n"); break; case Item::VIEW_FIXER_ITEM: printf("********** received VIEW_FIXER_ITEM *********\n"); break; default: { if (gwip->condPush) { // push noop for unhandled item SimpleColumn* rc = new SimpleColumn("noop"); gwip->rcWorkStack.push(rc); break; } ostringstream oss; oss << "Unhandled Item type: " << item->type(); gwip->parseErrorText = oss.str(); gwip->fatalParseError = true; break; } } return; } /** @info this function recursivly walks an item's arguments and push all * the involved item_fields to the passed in vector. It's used in parsing * functions or arithmetic expressions for vtable post process. */ void parse_item (Item* item, vector& field_vec, bool& hasNonSupportItem, uint16_t& parseInfo, gp_walk_info* gwi) { Item::Type itype = item->type(); switch (itype) { case Item::FIELD_ITEM: { Item_field* ifp = reinterpret_cast(item); field_vec.push_back(ifp); return; } case Item::SUM_FUNC_ITEM: { //hasAggColumn = true; parseInfo |= AGG_BIT; Item_sum* isp = reinterpret_cast(item); Item** sfitempp = isp->arguments(); for (uint32_t i = 0; i < isp->argument_count(); i++) parse_item(sfitempp[i], field_vec, hasNonSupportItem, parseInfo); break; } case Item::FUNC_ITEM: { Item_func* isp = reinterpret_cast(item); if (string(isp->func_name()) == "") { parseInfo |= SUB_BIT; parseInfo |= CORRELATED; break; } for (uint32_t i = 0; i < isp->argument_count(); i++) parse_item(isp->arguments()[i], field_vec, hasNonSupportItem, parseInfo, gwi); // parse_item(sfitempp[i], field_vec, hasNonSupportItem, parseInfo); break; } case Item::COND_ITEM: { Item_cond* icp = reinterpret_cast(item); List_iterator_fast it(*(icp->argument_list())); Item* cond_item; while ((cond_item = it++)) parse_item(cond_item, field_vec, hasNonSupportItem, parseInfo); break; } case Item::REF_ITEM: { while (true) { Item_ref* ref = (Item_ref*)item; if ((*(ref->ref))->type() == Item::SUM_FUNC_ITEM) { parseInfo |= AGG_BIT; Item_sum* isp = reinterpret_cast(*(ref->ref)); Item** sfitempp = isp->arguments(); // special handling for count(*). This should not be treated as constant. if (isp->argument_count() == 1 && (sfitempp[0]->type() == Item::INT_ITEM || sfitempp[0]->type() == Item::STRING_ITEM || sfitempp[0]->type() == Item::REAL_ITEM || sfitempp[0]->type() == Item::DECIMAL_ITEM)) field_vec.push_back((Item_field*)item); //dummy for (uint32_t i = 0; i < isp->argument_count(); i++) parse_item(sfitempp[i], field_vec, hasNonSupportItem, parseInfo); break; } else if ((*(ref->ref))->type() == Item::FIELD_ITEM) { // MCOL-1510. This could be a non-supported function // argument in form of a temp_table_field, so check // and set hasNonSupportItem if it is so. ReturnedColumn* rc = NULL; if (gwi) rc = buildAggFrmTempField(ref, *gwi); if (!rc) { Item_field* ifp = reinterpret_cast(*(ref->ref)); field_vec.push_back(ifp); } else hasNonSupportItem = true; break; } else if ((*(ref->ref))->type() == Item::FUNC_ITEM) { Item_func* isp = reinterpret_cast(*(ref->ref)); Item** sfitempp = isp->arguments(); for (uint32_t i = 0; i < isp->argument_count(); i++) parse_item(sfitempp[i], field_vec, hasNonSupportItem, parseInfo); break; } else if ((*(ref->ref))->type() == Item::CACHE_ITEM) { Item_cache* isp = reinterpret_cast(*(ref->ref)); parse_item(isp->get_example(), field_vec, hasNonSupportItem, parseInfo); break; } else if ((*(ref->ref))->type() == Item::REF_ITEM) { item = (*(ref->ref)); continue; } else if ((*(ref->ref))->type() == Item::WINDOW_FUNC_ITEM) { parseInfo |= AF_BIT; break; } else { cerr << "UNKNOWN REF Item" << endl; break; } } break; } case Item::SUBSELECT_ITEM: { parseInfo |= SUB_BIT; Item_subselect* sub = (Item_subselect*)item; if (sub->is_correlated) parseInfo |= CORRELATED; break; } case Item::ROW_ITEM: { Item_row* row = (Item_row*)item; for (uint32_t i = 0; i < row->cols(); i++) parse_item(row->element_index(i), field_vec, hasNonSupportItem, parseInfo); break; } case Item::EXPR_CACHE_ITEM: { // item is a Item_cache_wrapper. Shouldn't get here. printf("EXPR_CACHE_ITEM in parse_item\n"); string parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_SUB_QUERY_TYPE); setError(gwi->thd, ER_CHECK_NOT_IMPLEMENTED, parseErrorText); break; } case Item::WINDOW_FUNC_ITEM: parseInfo |= AF_BIT; break; default: break; } } bool isInfiniDB(TABLE* table_ptr) { #if (defined(_MSC_VER) && defined(_DEBUG)) || defined(SAFE_MUTEX) if (!(table_ptr->s && (*table_ptr->s->db_plugin)->name.str)) #else if (!(table_ptr->s && (table_ptr->s->db_plugin)->name.str)) #endif return true; #if (defined(_MSC_VER) && defined(_DEBUG)) || defined(SAFE_MUTEX) string engineName = (*table_ptr->s->db_plugin)->name.str; #else string engineName = table_ptr->s->db_plugin->name.str; #endif if (engineName == "Columnstore" || engineName == "InfiniDB") return true; else return false; } int getSelectPlan(gp_walk_info& gwi, SELECT_LEX& select_lex, SCSEP& csep, bool isUnion) { #ifdef DEBUG_WALK_COND cerr << "getSelectPlan()" << endl; #endif // by pass the derived table resolve phase of mysql if (!(((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) && gwi.thd->derived_tables_processing) { gwi.thd->infinidb_vtable.isUnion = false; return -1; } // rollup is currently not supported if (select_lex.olap == ROLLUP_TYPE) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_ROLLUP_NOT_SUPPORT); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } gwi.internalDecimalScale = (gwi.thd->variables.infinidb_use_decimal_scale ? gwi.thd->variables.infinidb_decimal_scale : -1); gwi.subSelectType = csep->subType(); JOIN* join = select_lex.join; Item_cond* icp = 0; if (join != 0) icp = reinterpret_cast(join->conds); // if icp is null, try to find the where clause other where if (!join && gwi.thd->lex->derived_tables) { if (select_lex.prep_where) icp = (Item_cond*)(select_lex.prep_where); else if (select_lex.where) icp = (Item_cond*)(select_lex.where); } else if (!join && ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ))) { icp = reinterpret_cast(select_lex.where); } uint32_t sessionID = csep->sessionID(); gwi.sessionid = sessionID; boost::shared_ptr csc = CalpontSystemCatalog::makeCalpontSystemCatalog(sessionID); csc->identity(CalpontSystemCatalog::FE); gwi.csc = csc; // @bug 2123. Override large table estimate if infinidb_ordered hint was used. // @bug 2404. Always override if the infinidb_ordered_only variable is turned on. if (gwi.thd->infinidb_vtable.override_largeside_estimate || gwi.thd->variables.infinidb_ordered_only) csep->overrideLargeSideEstimate(true); // @bug 5741. Set a flag when in Local PM only query mode csep->localQuery(gwi.thd->variables.infinidb_local_query); // @bug 3321. Set max number of blocks in a dictionary file to be scanned for filtering csep->stringScanThreshold(gwi.thd->variables.infinidb_string_scan_threshold); csep->stringTableThreshold(gwi.thd->variables.infinidb_stringtable_threshold); csep->djsSmallSideLimit(gwi.thd->variables.infinidb_diskjoin_smallsidelimit * 1024ULL * 1024); csep->djsLargeSideLimit(gwi.thd->variables.infinidb_diskjoin_largesidelimit * 1024ULL * 1024); csep->djsPartitionSize(gwi.thd->variables.infinidb_diskjoin_bucketsize * 1024ULL * 1024); if (gwi.thd->variables.infinidb_um_mem_limit == 0) csep->umMemLimit(numeric_limits::max()); else csep->umMemLimit(gwi.thd->variables.infinidb_um_mem_limit * 1024ULL * 1024); // populate table map and trigger syscolumn cache for all the tables (@bug 1637). // all tables on FROM list must have at least one col in colmap TABLE_LIST* table_ptr = select_lex.get_table_list(); CalpontSelectExecutionPlan::SelectList derivedTbList; // DEBUG #ifdef DEBUG_WALK_COND List_iterator sj_list_it(select_lex.sj_nests); TABLE_LIST* sj_nest; while ((sj_nest = sj_list_it++)) { cerr << sj_nest->db.str << "." << sj_nest->table_name.str << endl; } #endif // @bug 1796. Remember table order on the FROM list. gwi.clauseType = FROM; try { for (; table_ptr; table_ptr = table_ptr->next_local) { // mysql put vtable here for from sub. we ignore it if (string(table_ptr->table_name.str).find("$vtable") != string::npos) continue; // Until we handle recursive cte: // Checking here ensures we catch all with clauses in the query. if (table_ptr->is_recursive_with_table()) { gwi.fatalParseError = true; gwi.parseErrorText = "Recursive CTE"; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } string viewName = getViewName(table_ptr); // @todo process from subquery if (table_ptr->derived) { String str; (table_ptr->derived->first_select())->print(gwi.thd, &str, QT_INFINIDB_DERIVED); SELECT_LEX* select_cursor = table_ptr->derived->first_select(); FromSubQuery fromSub(gwi, select_cursor); string alias(table_ptr->alias.str); fromSub.alias(lower(alias)); CalpontSystemCatalog::TableAliasName tn = make_aliasview("", "", alias, viewName); // @bug 3852. check return execplan SCSEP plan = fromSub.transform(); if (!plan) { setError(gwi.thd, ER_INTERNAL_ERROR, fromSub.gwip().parseErrorText, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } gwi.derivedTbList.push_back(plan); gwi.tbList.push_back(tn); CalpontSystemCatalog::TableAliasName tan = make_aliastable("", alias, alias); gwi.tableMap[tan] = make_pair(0, table_ptr); gwi.thd->infinidb_vtable.isUnion = true; //by-pass the 2nd pass of rnd_init } else if (table_ptr->view) { View* view = new View(table_ptr->view->select_lex, &gwi); CalpontSystemCatalog::TableAliasName tn = make_aliastable(table_ptr->db.str, table_ptr->table_name.str, table_ptr->alias.str); view->viewName(tn); gwi.viewList.push_back(view); view->transform(); } else { // check foreign engine tables bool infiniDB = (table_ptr->table ? isInfiniDB(table_ptr->table) : true); // trigger system catalog cache if (infiniDB) csc->columnRIDs(make_table(table_ptr->db.str, table_ptr->table_name.str), true); string table_name = table_ptr->table_name.str; // @bug5523 if (table_ptr->db.length && strcmp(table_ptr->db.str, "information_schema") == 0) table_name = (table_ptr->schema_table_name.length ? table_ptr->schema_table_name.str : table_ptr->alias.str); CalpontSystemCatalog::TableAliasName tn = make_aliasview(table_ptr->db.str, table_name, table_ptr->alias.str, viewName, infiniDB); gwi.tbList.push_back(tn); CalpontSystemCatalog::TableAliasName tan = make_aliastable(table_ptr->db.str, table_name, table_ptr->alias.str, infiniDB); gwi.tableMap[tan] = make_pair(0, table_ptr); #ifdef DEBUG_WALK_COND cerr << tn << endl; #endif } } if (gwi.fatalParseError) { setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_INTERNAL_ERROR; } } catch (IDBExcept& ie) { setError(gwi.thd, ER_INTERNAL_ERROR, ie.what(), gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); // @bug 3852. set error status for gwi. gwi.fatalParseError = true; gwi.parseErrorText = ie.what(); return ER_INTERNAL_ERROR; } catch (...) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_LOST_CONN_EXEMGR); // @bug3852 set error status for gwi. gwi.fatalParseError = true; gwi.parseErrorText = emsg; setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } csep->tableList(gwi.tbList); bool unionSel = false; if (!isUnion && select_lex.master_unit()->is_unit_op()) { gwi.thd->infinidb_vtable.isUnion = true; CalpontSelectExecutionPlan::SelectList unionVec; SELECT_LEX* select_cursor = select_lex.master_unit()->first_select(); unionSel = true; uint8_t distUnionNum = 0; for (SELECT_LEX* sl = select_cursor; sl; sl = sl->next_select()) { SCSEP plan(new CalpontSelectExecutionPlan()); plan->txnID(csep->txnID()); plan->verID(csep->verID()); plan->sessionID(csep->sessionID()); plan->traceFlags(csep->traceFlags()); plan->data(csep->data()); // @bug 3853. When one or more sides or union queries contain derived tables, // sl->join->zero_result_cause is not trustable. Since we've already handled // constant filter now (0/1), we can relax the following checking. // @bug 2547. ignore union unit of zero result set case // if (sl->join) // { // sl->join->optimize(); // @bug 3067. not clear MySQL's behavior. when in subquery, this variable // is not trustable. // if (sl->join->zero_result_cause && !gwi.subQuery) // continue; // } // gwi for the union unit gp_walk_info union_gwi; union_gwi.thd = gwi.thd; uint32_t err = 0; if ((err = getSelectPlan(union_gwi, *sl, plan, unionSel)) != 0) return err; unionVec.push_back(SCEP(plan)); // distinct union num if (sl == select_lex.master_unit()->union_distinct) distUnionNum = unionVec.size(); /*#ifdef DEBUG_WALK_COND IDEBUG( cerr << ">>>> UNION DEBUG" << endl ); JOIN* join = sl->join; Item_cond* icp = 0; if (join != 0) icp = reinterpret_cast(join->conds); if (icp) icp->traverse_cond(debug_walk, &gwi, Item::POSTFIX); IDEBUG ( cerr << *plan << endl ); IDEBUG ( cerr << "<<<unionVec(unionVec); csep->distinctUnionNum(distUnionNum); if (unionVec.empty()) gwi.thd->infinidb_vtable.impossibleWhereOnUnion = true; } gwi.clauseType = WHERE; if (icp) { // MariaDB bug 624 - without the fix_fields call, delete with join may error with "No query step". //#if MYSQL_VERSION_ID < 50172 //@bug 3039. fix fields for constants if (!icp->fixed) { icp->fix_fields(gwi.thd, (Item**)&icp); } //#endif gwi.fatalParseError = false; #ifdef DEBUG_WALK_COND cerr << "------------------ WHERE -----------------------" << endl; icp->traverse_cond(debug_walk, &gwi, Item::POSTFIX); cerr << "------------------------------------------------\n" << endl; #endif icp->traverse_cond(gp_walk, &gwi, Item::POSTFIX); if (gwi.fatalParseError) { // if this is dervied table process phase, mysql may have not developed the plan // completely. Do not error and eventually mysql will call JOIN::exec() again. // related to bug 2922. Need to find a way to skip calling rnd_init for derived table // processing. if (gwi.thd->derived_tables_processing) { gwi.thd->infinidb_vtable.isUnion = false; gwi.thd->infinidb_vtable.isUpdateWithDerive = true; return -1; } setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_INTERNAL_ERROR; } } else if (join && join->zero_result_cause) { gwi.rcWorkStack.push(new ConstantColumn((int64_t)0, ConstantColumn::NUM)); } // ZZ - the followinig debug shows the structure of nested outer join. should // use a recursive function. #ifdef OUTER_JOIN_DEBUG List* tables = &(select_lex.top_join_list); List_iterator_fast ti(*tables); //TABLE_LIST *inner; //TABLE_LIST **table= (TABLE_LIST **)gwi.thd->alloc(sizeof(TABLE_LIST*) * tables->elements); //for (TABLE_LIST **t= table + (tables->elements - 1); t >= table; t--) // *t= ti++; //DBUG_ASSERT(tables->elements >= 1); //TABLE_LIST **end= table + tables->elements; //for (TABLE_LIST **tbl= table; tbl < end; tbl++) TABLE_LIST* curr; while ((curr = ti++)) { TABLE_LIST* curr = *tbl; if (curr->table_name) cerr << curr->table_name << " "; else cerr << curr->alias << endl; if (curr->outer_join) cerr << " is inner table" << endl; else if (curr->straight) cerr << "straight_join" << endl; else cerr << "join" << endl; if (curr->nested_join) { List* inners = &(curr->nested_join->join_list); List_iterator_fast li(*inners); TABLE_LIST** inner = (TABLE_LIST**)gwi.thd->alloc(sizeof(TABLE_LIST*) * inners->elements); for (TABLE_LIST** t = inner + (inners->elements - 1); t >= inner; t--) *t = li++; TABLE_LIST** end1 = inner + inners->elements; for (TABLE_LIST** tb = inner; tb < end1; tb++) { TABLE_LIST* curr1 = *tb; cerr << curr1->alias << endl; if (curr1->sj_on_expr) { curr1->sj_on_expr->traverse_cond(debug_walk, &gwi, Item::POSTFIX); } } } if (curr->sj_on_expr) { curr->sj_on_expr->traverse_cond(debug_walk, &gwi, Item::POSTFIX); } } #endif uint32_t failed = buildOuterJoin(gwi, select_lex); if (failed) return failed; // @bug5764. build outer join for view, make sure outerjoin filter is appended // to the end of the filter list. for (uint i = 0; i < gwi.viewList.size(); i++) { failed = gwi.viewList[i]->processOuterJoin(gwi); if (failed) break; } if (failed != 0) return failed; ParseTree* filters = NULL; ParseTree* ptp = NULL; ParseTree* lhs = NULL; // @bug 2932. for "select * from region where r_name" case. if icp not null and // ptWorkStack empty, the item is in rcWorkStack. // MySQL 5.6 (MariaDB?). when icp is null and zero_result_cause is set, a constant 0 // is pushed to rcWorkStack. if (/*icp && */gwi.ptWorkStack.empty() && !gwi.rcWorkStack.empty()) { filters = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } while (!gwi.ptWorkStack.empty()) { filters = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); if (gwi.ptWorkStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); //ptp->left(filters); ptp->right(filters); lhs = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); //ptp->right(rhs); ptp->left(lhs); gwi.ptWorkStack.push(ptp); } if (filters) { csep->filters(filters); std::string aTmpDir(startup::StartUp::tmpDir()); aTmpDir = aTmpDir + "/filter1.dot"; filters->drawTree(aTmpDir); } gwi.clauseType = SELECT; #ifdef DEBUG_WALK_COND { cerr << "------------------- SELECT --------------------" << endl; List_iterator_fast it(select_lex.item_list); Item* item; while ((item = it++)) { debug_walk(item, 0); } cerr << "-----------------------------------------------\n" << endl; } #endif // populate returnedcolumnlist and columnmap List_iterator_fast it(select_lex.item_list); Item* item; vector funcFieldVec; string sel_cols_in_create; string sel_cols_in_select; bool redo = false; // empty rcWorkStack and ptWorkStack. They should all be empty by now. clearStacks(gwi); // indicate the starting pos of scalar returned column, because some join column // has been inserted to the returned column list. if (gwi.subQuery) { ScalarSub* scalar = dynamic_cast(gwi.subQuery); if (scalar) scalar->returnedColPos(gwi.additionalRetCols.size()); } CalpontSelectExecutionPlan::SelectList selectSubList; while ((item = it++)) { string itemAlias = (item->name.length ? item->name.str : ""); // @bug 5916. Need to keep checking until getting concret item in case // of nested view. while (item->type() == Item::REF_ITEM) { Item_ref* ref = (Item_ref*)item; item = (*(ref->ref)); } Item::Type itype = item->type(); switch (itype) { case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)item; SimpleColumn* sc = NULL; if (ifp->field_name.length && string(ifp->field_name.str) == "*") { collectAllCols(gwi, ifp); break; } sc = buildSimpleColumn(ifp, gwi); if (sc) { boost::shared_ptr spsc(sc); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; string fullname; String str; ifp->print(&str, QT_INFINIDB_NO_QUOTE); fullname = str.c_ptr(); //sel_cols_in_create += fullname; if (ifp->is_autogenerated_name) // no alias { sel_cols_in_create += fullname + " `" + escapeBackTick(str.c_ptr()) + "`"; sc->alias(fullname); } else // alias { if (!itemAlias.empty()) sc->alias(itemAlias); sel_cols_in_create += fullname + " `" + escapeBackTick(sc->alias().c_str()) + "`"; } if (ifp->is_autogenerated_name) gwi.selectCols.push_back("`" + escapeBackTick(fullname.c_str()) + "`" + " `" + escapeBackTick(itemAlias.empty() ? ifp->name.str : itemAlias.c_str()) + "`"); else gwi.selectCols.push_back("`" + escapeBackTick((itemAlias.empty() ? ifp->name.str : itemAlias.c_str())) + "`"); gwi.returnedCols.push_back(spsc); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), spsc)); TABLE_LIST* tmp = 0; if (ifp->cached_table) tmp = ifp->cached_table; gwi.tableMap[make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias(), sc->isInfiniDB())] = make_pair(1, tmp); } else { setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); delete sc; return ER_INTERNAL_ERROR; } break; } //aggregate column case Item::SUM_FUNC_ITEM: { ReturnedColumn* ac = buildAggregateColumn(item, gwi); if (gwi.fatalParseError) { // e.g., non-support ref column setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); delete ac; return ER_CHECK_NOT_IMPLEMENTED; } // add this agg col to returnedColumnList boost::shared_ptr spac(ac); gwi.returnedCols.push_back(spac); gwi.selectCols.push_back('`' + escapeBackTick(spac->alias().c_str()) + '`'); String str(256); item->print(&str, QT_INFINIDB_NO_QUOTE); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += string(str.c_ptr()) + " `" + escapeBackTick(spac->alias().c_str()) + "`"; break; } case Item::FUNC_ITEM: { Item_func* ifp = reinterpret_cast(item); // @bug4383. error out non-support stored function if (ifp->functype() == Item_func::FUNC_SP) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_SP_FUNCTION_NOT_SUPPORT); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } if (string(ifp->func_name()) == "xor") { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } uint16_t parseInfo = 0; vector tmpVec; bool hasNonSupportItem = false; parse_item(ifp, tmpVec, hasNonSupportItem, parseInfo); if (ifp->with_subquery() || string(ifp->func_name()) == string("") || ifp->functype() == Item_func::NOT_ALL_FUNC || parseInfo & SUB_BIT) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_SELECT_SUB); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } ReturnedColumn* rc = buildFunctionColumn(ifp, gwi, hasNonSupportItem); SRCP srcp(rc); if (rc) { if (!hasNonSupportItem && !nonConstFunc(ifp) && !(parseInfo & AF_BIT) && tmpVec.size() == 0) { if (isUnion || unionSel || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT || parseInfo & SUB_BIT || select_lex.group_list.elements != 0) { srcp.reset(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (ifp->name.length) srcp->alias(ifp->name.str); continue; } if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) { } else { redo = true; String str; ifp->print(&str, QT_INFINIDB_NO_QUOTE); gwi.selectCols.push_back(string(str.c_ptr()) + " " + "`" + escapeBackTick(item->name.str) + "`"); } break; } //SRCP srcp(rc); gwi.returnedCols.push_back(srcp); if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { String str(256); ifp->print(&str, QT_INFINIDB_NO_QUOTE); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += string(str.c_ptr()) + " `" + ifp->name.str + "`"; gwi.selectCols.push_back("`" + escapeBackTick(ifp->name.str) + "`"); } } else // InfiniDB Non support functions still go through post process for now { hasNonSupportItem = false; uint32_t before_size = funcFieldVec.size(); parse_item(ifp, funcFieldVec, hasNonSupportItem, parseInfo); uint32_t after_size = funcFieldVec.size(); // group by func and func in subquery can not be post processed // @bug3881. set_user_var can not be treated as constant function // @bug5716. Try to avoid post process function for union query. if ((gwi.subQuery || select_lex.group_list.elements != 0 || !csep->unionVec().empty() || isUnion) && !hasNonSupportItem && (after_size - before_size) == 0 && !(parseInfo & AGG_BIT) && !(parseInfo & SUB_BIT) && string(ifp->func_name()) != "set_user_var") { String val, *str = ifp->val_str(&val); string valStr; if (str) valStr.assign(str->ptr(), str->length()); ConstantColumn* cc = NULL; if (!str) { cc = new ConstantColumn("", ConstantColumn::NULLDATA); } else if (ifp->result_type() == STRING_RESULT) { cc = new ConstantColumn(valStr, ConstantColumn::LITERAL); } else if (ifp->result_type() == DECIMAL_RESULT) { cc = buildDecimalColumn(ifp, gwi); } else { cc = new ConstantColumn(valStr, ConstantColumn::NUM); cc->resultType(colType_MysqlToIDB(item)); } SRCP srcp(cc); if (ifp->name.length) cc->alias(ifp->name.str); gwi.returnedCols.push_back(srcp); // clear the error set by buildFunctionColumn gwi.fatalParseError = false; gwi.parseErrorText = ""; break; } else if (hasNonSupportItem || parseInfo & AGG_BIT || parseInfo & SUB_BIT || (gwi.fatalParseError && gwi.subQuery)) { if (gwi.parseErrorText.empty()) { Message::Args args; args.add(ifp->func_name()); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); } setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } else if ( gwi.subQuery && (isPredicateFunction(ifp, &gwi) || ifp->type() == Item::COND_ITEM )) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } //@Bug 3030 Add error check for dml statement if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) { if ( after_size - before_size != 0 ) { gwi.parseErrorText = ifp->func_name(); return -1; } } //@Bug 3021. Bypass postprocess for update and delete. //if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) //{} else { // @bug 3881. Here is the real redo part. redo = true; // @bug 1706 String funcStr; ifp->print(&funcStr, QT_INFINIDB); string valStr; valStr.assign(funcStr.ptr(), funcStr.length()); gwi.selectCols.push_back(valStr + " `" + escapeBackTick(ifp->name.str) + "`"); // clear the error set by buildFunctionColumn gwi.fatalParseError = false; gwi.parseErrorText = ""; } } break; } case Item::INT_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { // do not push the dummy column (mysql added) to returnedCol if (item->name.length && string(item->name.str) == "Not_used") continue; // @bug3509. Constant column is sent to ExeMgr now. SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); if (item->name.length) srcp->alias(item->name.str); gwi.returnedCols.push_back(srcp); Item_int* isp = reinterpret_cast(item); ostringstream oss; oss << isp->value << " `" << escapeBackTick(srcp->alias().c_str()) << "`"; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += oss.str(); gwi.selectCols.push_back(oss.str()); } break; } case Item::STRING_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (item->name.length) srcp->alias(item->name.str); Item_string* isp = reinterpret_cast(item); String val, *str = isp->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); string name = "'" + valStr + "'" + " " + "`" + escapeBackTick(srcp->alias().c_str()) + "`"; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += name; gwi.selectCols.push_back(name); } break; } case Item::DECIMAL_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (item->name.length) srcp->alias(item->name.str); Item_decimal* isp = reinterpret_cast(item); String val, *str = isp->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); ostringstream oss; oss << valStr.c_str() << " `" << escapeBackTick(srcp->alias().c_str()) << "`"; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += oss.str(); gwi.selectCols.push_back(oss.str()); } break; } case Item::NULL_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) { } else { SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (item->name.length) srcp->alias(item->name.str); string name = string("null `") + escapeBackTick(srcp->alias().c_str()) + string("`") ; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += name; gwi.selectCols.push_back("null"); } break; } case Item::SUBSELECT_ITEM: { Item_subselect* sub = (Item_subselect*)item; if (sub->substype() != Item_subselect::SINGLEROW_SUBS) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_SELECT_SUB); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } #ifdef DEBUG_WALK_COND cerr << "SELECT clause SUBSELECT Item: " << sub->substype() << endl; JOIN* join = sub->get_select_lex()->join; if (join) { Item_cond* cond = reinterpret_cast(join->conds); if (cond) cond->traverse_cond(debug_walk, &gwi, Item::POSTFIX); } cerr << "Finish SELECT clause subselect item traversing" << endl; #endif SelectSubQuery* selectSub = new SelectSubQuery(gwi, sub); //selectSub->gwip(&gwi); SCSEP ssub = selectSub->transform(); if (!ssub || gwi.fatalParseError) { if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Unsupported Item in SELECT subquery."; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } selectSubList.push_back(ssub); SimpleColumn* rc = new SimpleColumn(); rc->colSource(rc->colSource() | SELECT_SUB); if (sub->get_select_lex()->get_table_list()) rc->viewName(lower(getViewName(sub->get_select_lex()->get_table_list()))); if (sub->name.length) rc->alias(sub->name.str); gwi.returnedCols.push_back(SRCP(rc)); String str; sub->get_select_lex()->print(gwi.thd, &str, QT_INFINIDB_NO_QUOTE); sel_cols_in_create += "(" + string(str.c_ptr()) + ")"; if (sub->name.length) { sel_cols_in_create += "`" + escapeBackTick(sub->name.str) + "`"; gwi.selectCols.push_back(sub->name.str); } else { sel_cols_in_create += "`" + escapeBackTick(str.c_ptr()) + "`"; gwi.selectCols.push_back("`" + escapeBackTick(str.c_ptr()) + "`"); } break; } case Item::COND_ITEM: { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } case Item::EXPR_CACHE_ITEM: { printf("EXPR_CACHE_ITEM in getSelectPlan\n"); gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_UNKNOWN_COL); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } case Item::WINDOW_FUNC_ITEM: { SRCP srcp(buildWindowFunctionColumn(item, gwi, gwi.fatalParseError)); if (!srcp || gwi.fatalParseError) { if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Unsupported Item in SELECT subquery."; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } gwi.returnedCols.push_back(srcp); break; } default: { break; } } } // @bug4388 normalize the project coltypes for union main select list if (!csep->unionVec().empty()) { for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { vector coltypes; for (uint32_t j = 0; j < csep->unionVec().size(); j++) { coltypes.push_back( dynamic_cast(csep->unionVec()[j].get())->returnedCols()[i]->resultType()); // @bug5976. set hasAggregate true for the main column if // one corresponding union column has aggregate if (dynamic_cast(csep->unionVec()[j].get())->returnedCols()[i]->hasAggregate()) gwi.returnedCols[i]->hasAggregate(true); } gwi.returnedCols[i]->resultType(dataconvert::DataConvert::convertUnionColType(coltypes)); } } // Having clause handling gwi.clauseType = HAVING; clearStacks(gwi); ParseTree* havingFilter = 0; // clear fatalParseError that may be left from post process functions gwi.fatalParseError = false; gwi.parseErrorText = ""; if (select_lex.having != 0) { Item_cond* having = reinterpret_cast(select_lex.having); #ifdef DEBUG_WALK_COND cerr << "------------------- HAVING ---------------------" << endl; having->traverse_cond(debug_walk, &gwi, Item::POSTFIX); cerr << "------------------------------------------------\n" << endl; #endif having->traverse_cond(gp_walk, &gwi, Item::POSTFIX); if (gwi.fatalParseError) { setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_INTERNAL_ERROR; } ParseTree* ptp = 0; ParseTree* rhs = 0; // @bug 4215. some function filter will be in the rcWorkStack. if (gwi.ptWorkStack.empty() && !gwi.rcWorkStack.empty()) { havingFilter = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } while (!gwi.ptWorkStack.empty()) { havingFilter = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); if (gwi.ptWorkStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); ptp->left(havingFilter); rhs = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); ptp->right(rhs); gwi.ptWorkStack.push(ptp); } } // for post process expressions on the select list // error out post process for union and sub select unit if (isUnion || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT) { if (funcFieldVec.size() != 0 && !gwi.fatalParseError) { string emsg("Fatal parse error in vtable mode: Unsupported Items in union or sub select unit"); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg); return ER_CHECK_NOT_IMPLEMENTED; } } for (uint32_t i = 0; i < funcFieldVec.size(); i++) { //SimpleColumn *sc = new SimpleColumn(funcFieldVec[i]->db_name, bestTableName(funcFieldVec[i])/*funcFieldVec[i]->table_name*/, funcFieldVec[i]->field_name, sessionID); SimpleColumn* sc = buildSimpleColumn(funcFieldVec[i], gwi); if (!sc || gwi.fatalParseError) { string emsg; if (gwi.parseErrorText.empty()) { emsg = "un-recognized column"; if (funcFieldVec[i]->name.length) emsg += string(funcFieldVec[i]->name.str); } else { emsg = gwi.parseErrorText; } setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); return ER_INTERNAL_ERROR; } String str; funcFieldVec[i]->print(&str, QT_INFINIDB_NO_QUOTE); sc->alias(string(str.c_ptr())); //sc->tableAlias(funcFieldVec[i]->table_name); sc->tableAlias(sc->tableAlias()); SRCP srcp(sc); uint32_t j = 0; for (; j < gwi.returnedCols.size(); j++) { if (sc->sameColumn(gwi.returnedCols[j].get())) { SimpleColumn* field = dynamic_cast(gwi.returnedCols[j].get()); if (field && field->alias() == sc->alias()) break; } } if (j == gwi.returnedCols.size()) { gwi.returnedCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(funcFieldVec[i]->field_name.str), srcp)); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; string fullname; fullname = str.c_ptr(); sel_cols_in_create += fullname + " `" + escapeBackTick(fullname.c_str()) + "`"; TABLE_LIST* tmp = (funcFieldVec[i]->cached_table ? funcFieldVec[i]->cached_table : 0); gwi.tableMap[make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias(), sc->isInfiniDB())] = make_pair(1, tmp); } } // post-process Order by list and expressions on select by redo phase1. only for vtable // ignore ORDER BY clause for union select unit string ord_cols = ""; // for normal select phase SRCP minSc; // min width projected column. for count(*) use // Group by list. not valid for union main query if (gwi.thd->infinidb_vtable.vtable_state == THD::INFINIDB_CREATE_VTABLE && !unionSel) { gwi.clauseType = GROUP_BY; Item* nonSupportItem = NULL; ORDER* groupcol = reinterpret_cast(select_lex.group_list.first); // check if window functions are in order by. InfiniDB process order by list if // window functions are involved, either in order by or projection. bool hasWindowFunc = gwi.hasWindowFunc; gwi.hasWindowFunc = false; for (; groupcol; groupcol = groupcol->next) { if ((*(groupcol->item))->type() == Item::WINDOW_FUNC_ITEM) gwi.hasWindowFunc = true; } if (gwi.hasWindowFunc) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_WF_NOT_ALLOWED, "GROUP BY clause"); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } gwi.hasWindowFunc = hasWindowFunc; groupcol = reinterpret_cast(select_lex.group_list.first); for (; groupcol; groupcol = groupcol->next) { Item* groupItem = *(groupcol->item); // @bug5993. Could be nested ref. while (groupItem->type() == Item::REF_ITEM) groupItem = (*((Item_ref*)groupItem)->ref); if (groupItem->type() == Item::FUNC_ITEM) { Item_func* ifp = (Item_func*)groupItem; // call buildFunctionColumn here mostly for finding out // non-support column on GB list. Should be simplified. ReturnedColumn* fc = buildFunctionColumn(ifp, gwi, gwi.fatalParseError); if (!fc || gwi.fatalParseError) { nonSupportItem = ifp; break; } if (groupcol->in_field_list && groupcol->counter_used) { delete fc; fc = gwi.returnedCols[groupcol->counter - 1].get(); SRCP srcp(fc->clone()); // check if no column parm for (uint32_t i = 0; i < gwi.no_parm_func_list.size(); i++) { if (gwi.no_parm_func_list[i]->expressionId() == fc->expressionId()) { gwi.no_parm_func_list.push_back(dynamic_cast(srcp.get())); break; } } srcp->orderPos(groupcol->counter - 1); gwi.groupByCols.push_back(srcp); continue; } else if (!groupItem->is_autogenerated_name) // alias { uint32_t i = 0; for (; i < gwi.returnedCols.size(); i++) { if (string(groupItem->name.str) == gwi.returnedCols[i]->alias()) { ReturnedColumn* rc = gwi.returnedCols[i]->clone(); rc->orderPos(i); gwi.groupByCols.push_back(SRCP(rc)); delete fc; break; } } if (i == gwi.returnedCols.size()) { nonSupportItem = groupItem; break; } } else { uint32_t i = 0; for (; i < gwi.returnedCols.size(); i++) { if (fc->operator==(gwi.returnedCols[i].get())) { ReturnedColumn* rc = gwi.returnedCols[i]->clone(); rc->orderPos(i); gwi.groupByCols.push_back(SRCP(rc)); delete fc; break; } } if (i == gwi.returnedCols.size()) { gwi.groupByCols.push_back(SRCP(fc)); break; } } } else if (groupItem->type() == Item::FIELD_ITEM) { Item_field* ifp = (Item_field*)groupItem; // this GB col could be an alias of F&E on the SELECT clause, not necessarily a field. ReturnedColumn* rc = buildSimpleColumn(ifp, gwi); SimpleColumn* sc = dynamic_cast(rc); for (uint32_t j = 0; j < gwi.returnedCols.size(); j++) { if (sc) { if (sc->sameColumn(gwi.returnedCols[j].get())) { sc->orderPos(j); break; } else if (strcasecmp(sc->alias().c_str(), gwi.returnedCols[j]->alias().c_str()) == 0) { rc = gwi.returnedCols[j].get()->clone(); rc->orderPos(j); break; } } else { if (ifp->name.length && string(ifp->name.str) == gwi.returnedCols[j].get()->alias()) { rc = gwi.returnedCols[j].get()->clone(); rc->orderPos(j); break; } } } if (!rc) { nonSupportItem = ifp; break; } SRCP srcp(rc); // bug 3151 AggregateColumn* ac = dynamic_cast(rc); if (ac) { nonSupportItem = ifp; break; } gwi.groupByCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), srcp)); } // @bug5638. The group by column is constant but not counter, alias has to match a column // on the select list else if (!groupcol->counter_used && (groupItem->type() == Item::INT_ITEM || groupItem->type() == Item::STRING_ITEM || groupItem->type() == Item::REAL_ITEM || groupItem->type() == Item::DECIMAL_ITEM)) { ReturnedColumn* rc = 0; for (uint32_t j = 0; j < gwi.returnedCols.size(); j++) { if (groupItem->name.length && string(groupItem->name.str) == gwi.returnedCols[j].get()->alias()) { rc = gwi.returnedCols[j].get()->clone(); rc->orderPos(j); break; } } if (!rc) { nonSupportItem = groupItem; break; } gwi.groupByCols.push_back(SRCP(rc)); } else if ((*(groupcol->item))->type() == Item::SUBSELECT_ITEM) { if (!groupcol->in_field_list || !groupItem->name.length) { nonSupportItem = groupItem; } else { uint32_t i = 0; for (; i < gwi.returnedCols.size(); i++) { if (string(groupItem->name.str) == gwi.returnedCols[i]->alias()) { ReturnedColumn* rc = gwi.returnedCols[i]->clone(); rc->orderPos(i); gwi.groupByCols.push_back(SRCP(rc)); break; } } if (i == gwi.returnedCols.size()) { nonSupportItem = groupItem; } } } // @bug 3761. else if (groupcol->counter_used) { if (gwi.returnedCols.size() <= (uint32_t)(groupcol->counter - 1)) { nonSupportItem = groupItem; } else { gwi.groupByCols.push_back(SRCP(gwi.returnedCols[groupcol->counter - 1]->clone())); } } else { nonSupportItem = groupItem; } } // @bug 4756. Add internal groupby column for correlated join to the groupby list if (gwi.aggOnSelect && !gwi.subGroupByCols.empty()) gwi.groupByCols.insert(gwi.groupByCols.end(), gwi.subGroupByCols.begin(), gwi.subGroupByCols.end()); // this is window func on SELECT becuase ORDER BY has not been processed if (!gwi.windowFuncList.empty() && !gwi.subGroupByCols.empty()) { for (uint32_t i = 0; i < gwi.windowFuncList.size(); i++) { if (gwi.windowFuncList[i]->hasWindowFunc()) { vector windowFunctions = gwi.windowFuncList[i]->windowfunctionColumnList(); for (uint32_t j = 0; j < windowFunctions.size(); j++) windowFunctions[j]->addToPartition(gwi.subGroupByCols); } } } if (nonSupportItem) { Message::Args args; if (nonSupportItem->name.length) args.add("'" + string(nonSupportItem->name.str) + "'"); else args.add(""); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_GROUP_BY, args); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } } if (gwi.thd->infinidb_vtable.vtable_state == THD::INFINIDB_CREATE_VTABLE) { SQL_I_List order_list = select_lex.order_list; ORDER* ordercol = reinterpret_cast(order_list.first); string create_query(gwi.thd->infinidb_vtable.create_vtable_query.c_ptr()); string select_query(gwi.thd->infinidb_vtable.select_vtable_query.c_ptr()); string lower_create_query(gwi.thd->infinidb_vtable.create_vtable_query.c_ptr()); string lower_select_query(gwi.thd->infinidb_vtable.select_vtable_query.c_ptr()); boost::algorithm::to_lower(lower_create_query); boost::algorithm::to_lower(lower_select_query); // check if window functions are in order by. InfiniDB process order by list if // window functions are involved, either in order by or projection. for (; ordercol; ordercol = ordercol->next) { if ((*(ordercol->item))->type() == Item::WINDOW_FUNC_ITEM) gwi.hasWindowFunc = true; } // re-visit the first of ordercol list ordercol = reinterpret_cast(order_list.first); // for subquery, order+limit by will be supported in infinidb. build order by columns // @todo union order by and limit support if (gwi.hasWindowFunc || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT) { for (; ordercol; ordercol = ordercol->next) { ReturnedColumn* rc = NULL; if (ordercol->in_field_list && ordercol->counter_used) { rc = gwi.returnedCols[ordercol->counter - 1]->clone(); rc->orderPos(ordercol->counter - 1); // can not be optimized off if used in order by with counter. // set with self derived table alias if it's derived table gwi.returnedCols[ordercol->counter - 1]->incRefCount(); } else { Item* ord_item = *(ordercol->item); // ignore not_used column on order by. if (ord_item->type() == Item::INT_ITEM && ord_item->full_name() && string(ord_item->full_name()) == "Not_used") continue; else if (ord_item->type() == Item::INT_ITEM) rc = gwi.returnedCols[((Item_int*)ord_item)->val_int() - 1]->clone(); else if (ord_item->type() == Item::SUBSELECT_ITEM) gwi.fatalParseError = true; else rc = buildReturnedColumn(ord_item, gwi, gwi.fatalParseError); // @bug5501 try item_ptr if item can not be fixed. For some // weird dml statement state, item can not be fixed but the // infomation is available in item_ptr. if (!rc || gwi.fatalParseError) { Item* item_ptr = ordercol->item_ptr; while (item_ptr->type() == Item::REF_ITEM) item_ptr = *(((Item_ref*)item_ptr)->ref); rc = buildReturnedColumn(item_ptr, gwi, gwi.fatalParseError); } if (!rc) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); gwi.parseErrorText = emsg; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } } if (ordercol->direction == ORDER::ORDER_ASC) rc->asc(true); else rc->asc(false); gwi.orderByCols.push_back(SRCP(rc)); } } else if (!isUnion) { vector fieldVec; bool addToSel; // the following order by is just for redo phase if (!unionSel) { for (; ordercol; ordercol = ordercol->next) { Item* ord_item = *(ordercol->item); // @bug5993. Could be nested ref. while (ord_item->type() == Item::REF_ITEM) ord_item = (*((Item_ref*)ord_item)->ref); // @bug 1706. re-construct the order by item one by one //Item* ord_item = *(ordercol->item); if (ord_cols.length() != 0) ord_cols += ", "; addToSel = true; string fullname; if (ordercol->in_field_list && ordercol->counter_used) { ostringstream oss; oss << ordercol->counter; ord_cols += oss.str(); if (ordercol->direction != ORDER::ORDER_ASC) ord_cols += " desc"; continue; } else if (ord_item->type() == Item::FUNC_ITEM) { // @bug 2621. order by alias if (!ord_item->is_autogenerated_name && ord_item->name.length) { ord_cols += ord_item->name.str; continue; } // if there's group by clause or aggregate column, check to see // if this item or the arguments is on the GB list. ReturnedColumn* rc = 0; // check if this order by column is on the select list Item_func* ifp = (Item_func*)(*(ordercol->item)); rc = buildFunctionColumn(ifp, gwi, gwi.fatalParseError); if (rc) { for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if (rc && rc->operator==(gwi.returnedCols[i].get())) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); addToSel = false; break; } } } if (addToSel) { FunctionColumn* fc = dynamic_cast(rc); if (fc) { addToSel = false; redo = true; string ord_func = string(ifp->func_name()) + "("; for (uint32_t i = 0; i < fc->functionParms().size(); i++) { if (i != 0) ord_func += ","; for (uint32_t j = 0; j < gwi.returnedCols.size(); j++) { if (fc->functionParms()[i]->data()->operator==(gwi.returnedCols[j].get())) { ord_func += "`" + escapeBackTick(gwi.returnedCols[j]->alias().c_str()) + "`"; continue; } AggregateColumn* ac = dynamic_cast(fc->functionParms()[i]->data()); if (ac) { gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } addToSel = true; //continue; } } ord_func += ")"; if (!addToSel) ord_cols += ord_func; } } } else if (ord_item->type() == Item::SUBSELECT_ITEM) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } else if (ord_item->type() == Item::SUM_FUNC_ITEM) { ReturnedColumn* ac = 0; Item_sum* ifp = (Item_sum*)(*(ordercol->item)); // @bug3477. add aggregate column to the select list of the create phase. ac = buildAggregateColumn(ifp, gwi); if (!ac) { setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY), gwi); return ER_CHECK_NOT_IMPLEMENTED; } // check if this order by column is on the select list for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { AggregateColumn* ret = dynamic_cast(gwi.returnedCols[i].get()); if (!ret) continue; if (ac->operator==(gwi.returnedCols[i].get())) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); addToSel = false; break; } } if (ac || !gwi.groupByCols.empty()) { if (addToSel) { redo = true; // @bug 3076. do not add the argument of aggregate function to the SELECT list, // instead, add the whole column String str; ord_item->print(&str, QT_INFINIDB_NO_QUOTE); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += str.c_ptr(); //gwi.selectCols.push_back(" `" + string(str.c_ptr()) + "`"); SRCP srcp(ac); gwi.returnedCols.push_back(srcp); ord_cols += " `" + escapeBackTick(str.c_ptr()) + "`"; } if (ordercol->direction != ORDER::ORDER_ASC) ord_cols += " desc"; continue; } } else if (ord_item->name.length && ord_item->type() == Item::FIELD_ITEM) { Item_field* field = reinterpret_cast(ord_item); ReturnedColumn* rc = buildSimpleColumn(field, gwi); fullname = field->full_name(); // if (field->db_name) // fullname += string(field->db_name) + "."; // if (field->table_name) // fullname += string(field->table_name) + "."; // if (field->field_name) // fullname += string(field->field_name); for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { SimpleColumn* sc = dynamic_cast(gwi.returnedCols[i].get()); if (sc && ((Item_field*)ord_item)->cached_table && (strcasecmp(getViewName(((Item_field*)ord_item)->cached_table).c_str(), sc->viewName().c_str()) != 0)) { continue; } if (strcasecmp(fullname.c_str(), gwi.returnedCols[i]->alias().c_str()) == 0 || strcasecmp(ord_item->name.str, gwi.returnedCols[i]->alias().c_str()) == 0) { ord_cols += string(" `") + escapeBackTick(gwi.returnedCols[i]->alias().c_str()) + '`'; addToSel = false; break; } if (sc && sc->sameColumn(rc)) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); addToSel = false; break; } } } if (addToSel) { // @bug 2719. Error out order by not on the distinct select list. if (select_lex.options & SELECT_DISTINCT) { gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_ORDERBY_NOT_IN_DISTINCT); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } bool hasNonSupportItem = false; uint16_t parseInfo = 0; parse_item(ord_item, fieldVec, hasNonSupportItem, parseInfo); if (hasNonSupportItem) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } String str; ord_item->print(&str, QT_INFINIDB); ord_cols += str.c_ptr(); } if (ordercol->direction != ORDER::ORDER_ASC) ord_cols += " desc"; } } redo = (redo || fieldVec.size() != 0); // populate string to be added to the select list for order by for (uint32_t i = 0; i < fieldVec.size(); i++) { SimpleColumn* sc = buildSimpleColumn(fieldVec[i], gwi); if (!sc) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } String str; fieldVec[i]->print(&str, QT_INFINIDB_NO_QUOTE); sc->alias(string(str.c_ptr())); SRCP srcp(sc); uint32_t j = 0; for (; j < gwi.returnedCols.size(); j++) { if (sc->sameColumn(gwi.returnedCols[j].get())) { SimpleColumn* field = dynamic_cast(gwi.returnedCols[j].get()); if (field && field->alias() == sc->alias()) break; } } if (j == gwi.returnedCols.size()) { string fullname; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; fullname = str.c_ptr(); sel_cols_in_create += fullname + " `" + escapeBackTick(fullname.c_str()) + "`"; gwi.returnedCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(fieldVec[i]->field_name.str), srcp)); TABLE_LIST* tmp = (fieldVec[i]->cached_table ? fieldVec[i]->cached_table : 0); gwi.tableMap[make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias(), sc->isInfiniDB())] = make_pair(1, tmp); } } } // make sure columnmap, returnedcols and count(*) arg_list are not empty TableMap::iterator tb_iter = gwi.tableMap.begin(); try { for (; tb_iter != gwi.tableMap.end(); tb_iter++) { if ((*tb_iter).second.first == 1) continue; CalpontSystemCatalog::TableAliasName tan = (*tb_iter).first; CalpontSystemCatalog::TableName tn = make_table((*tb_iter).first.schema, (*tb_iter).first.table); SimpleColumn* sc = getSmallestColumn(csc, tn, tan, (*tb_iter).second.second->table, gwi); SRCP srcp(sc); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(sc->columnName(), srcp)); (*tb_iter).second.first = 1; } } catch (runtime_error& e) { setError(gwi.thd, ER_INTERNAL_ERROR, e.what(), gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } catch (...) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_LOST_CONN_EXEMGR); setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } if (!gwi.count_asterisk_list.empty() || !gwi.no_parm_func_list.empty() || gwi.returnedCols.empty()) { // get the smallest column from colmap CalpontSelectExecutionPlan::ColumnMap::const_iterator iter; int minColWidth = 0; CalpontSystemCatalog::ColType ct; try { for (iter = gwi.columnMap.begin(); iter != gwi.columnMap.end(); ++iter) { // should always not null SimpleColumn* sc = dynamic_cast(iter->second.get()); if (sc && !(sc->joinInfo() & JOIN_CORRELATED)) { ct = csc->colType(sc->oid()); if (minColWidth == 0) { minColWidth = ct.colWidth; minSc = iter->second; } else if (ct.colWidth < minColWidth) { minColWidth = ct.colWidth; minSc = iter->second; } } } } catch (...) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_LOST_CONN_EXEMGR); setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } if (gwi.returnedCols.empty() && gwi.additionalRetCols.empty() && minSc) gwi.returnedCols.push_back(minSc); } if (!isUnion && !gwi.hasWindowFunc && gwi.subSelectType == CalpontSelectExecutionPlan::MAIN_SELECT) { std::ostringstream vtb; vtb << "infinidb_vtable.$vtable_" << gwi.thd->thread_id; //vtb << "$vtable_" << gwi.thd->thread_id; // re-construct the select query and redo phase 1 if (redo) { // select now() from region case. returnedCols should have minSc. if (sel_cols_in_create.length() == 0) { SimpleColumn* sc = dynamic_cast(gwi.returnedCols[0].get()); if (sc) sel_cols_in_create = dynamic_cast(gwi.returnedCols[0].get())->columnName(); else sel_cols_in_create = gwi.returnedCols[0]->alias(); } // select * from derived table case if (gwi.selectCols.empty()) sel_cols_in_create = " * "; create_query = "create temporary table " + vtb.str() + " engine = aria as select " + sel_cols_in_create + " from "; TABLE_LIST* table_ptr = select_lex.get_table_list(); bool firstTb = true; // put all tables, derived tables and views on the list //TABLE_LIST* table_ptr = select_lex.get_table_list(); set aliasSet; // to avoid duplicate table alias for (; table_ptr; table_ptr = table_ptr->next_global) { if (string(table_ptr->table_name.str).find("$vtable") != string::npos) continue; if (table_ptr->derived) { if (aliasSet.find(table_ptr->alias.str) != aliasSet.end()) continue; String str; (table_ptr->derived->first_select())->print(gwi.thd, &str, QT_INFINIDB_DERIVED); if (!firstTb) create_query += ", "; create_query += "(" + string(str.c_ptr()) + ") " + string(table_ptr->alias.str); firstTb = false; aliasSet.insert(table_ptr->alias.str); } else if (table_ptr->view) { if (aliasSet.find(table_ptr->alias.str) != aliasSet.end()) continue; if (!firstTb) create_query += ", "; create_query += string(table_ptr->db.str) + "." + string(table_ptr->table_name.str) + string(" `") + escapeBackTick(table_ptr->alias.str) + string("`"); aliasSet.insert(table_ptr->alias.str); firstTb = false; } else { // table referenced by view is represented by viewAlias_tableAlias. // consistent with item.cc field print. if (table_ptr->referencing_view) { if (aliasSet.find(string(table_ptr->referencing_view->alias.str) + "_" + string(table_ptr->alias.str)) != aliasSet.end()) continue; if (!firstTb) create_query += ", "; create_query += string(table_ptr->db.str) + "." + string(table_ptr->table_name.str) + string(" "); create_query += string(" `") + escapeBackTick(table_ptr->referencing_view->alias.str) + "_" + escapeBackTick(table_ptr->alias.str) + string("`"); aliasSet.insert(string(table_ptr->referencing_view->alias.str) + "_" + string(table_ptr->alias.str)); } else { if (aliasSet.find(table_ptr->alias.str) != aliasSet.end()) continue; if (!firstTb) create_query += ", "; create_query += string(table_ptr->db.str) + "." + string(table_ptr->table_name.str) + string(" "); create_query += string("`") + escapeBackTick(table_ptr->alias.str) + string("`"); aliasSet.insert(table_ptr->alias.str); } firstTb = false; } } gwi.thd->infinidb_vtable.create_vtable_query.free(); gwi.thd->infinidb_vtable.create_vtable_query.append(create_query.c_str(), create_query.length()); gwi.thd->infinidb_vtable.vtable_state = THD::INFINIDB_REDO_PHASE1; // redo phase 1 // turn off select distinct from post process unless there're post process functions // on the select list. string sel_query = "select "; if (/*join->select_options*/select_lex.options & SELECT_DISTINCT && redo) sel_query = "select distinct "; else sel_query = "select "; // select * from derived table... if (gwi.selectCols.size() == 0) sel_query += " * "; for (uint32_t i = 0; i < gwi.selectCols.size(); i++) { sel_query += gwi.selectCols[i]; if ( i + 1 != gwi.selectCols.size()) sel_query += ", "; } select_query.replace(lower_select_query.find("select *"), string("select *").length(), sel_query); } else { // remove order by clause in case this phase has been executed before. // need a better fix later, like skip all the other non-optimized phase. size_t pos = lower_select_query.find("order by"); if (pos != string::npos) select_query.replace(pos, lower_select_query.length() - pos, ""); //select_query = "select * from " + vtb.str(); if (unionSel) order_list = select_lex.master_unit()->global_parameters()->order_list; ordercol = reinterpret_cast(order_list.first); ord_cols = ""; for (; ordercol; ordercol = ordercol->next) { Item* ord_item = *(ordercol->item); // @bug 1706. re-construct the order by item one by one, because the ord_cols constucted so far // is for REDO phase. if (ord_cols.length() != 0) ord_cols += ", "; if (ordercol->in_field_list && ordercol->counter_used) { ostringstream oss; oss << ordercol->counter; ord_cols += oss.str(); } else if (ord_item->type() == Item::NULL_ITEM) { // MCOL-793 Do nothing for an ORDER BY NULL } else if (ord_item->type() == Item::SUM_FUNC_ITEM) { Item_sum* ifp = (Item_sum*)(*(ordercol->item)); ReturnedColumn* fc = buildAggregateColumn(ifp, gwi); for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if (fc->operator==(gwi.returnedCols[i].get())) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); break; } } //continue; } // @bug 3518. if order by clause = selected column, use position. else if (ord_item->name.length && ord_item->type() == Item::FIELD_ITEM) { Item_field* field = reinterpret_cast(ord_item); string fullname; if (field->db_name) fullname += string(field->db_name) + "."; if (field->table_name) fullname += string(field->table_name) + "."; if (field->field_name.length) fullname += string(field->field_name.str); uint32_t i = 0; for (i = 0; i < gwi.returnedCols.size(); i++) { SimpleColumn* sc = dynamic_cast(gwi.returnedCols[i].get()); if (sc && ((Item_field*)ord_item)->cached_table && (strcasecmp(getViewName(((Item_field*)ord_item)->cached_table).c_str(), sc->viewName().c_str()) != 0)) continue; if (strcasecmp(fullname.c_str(), gwi.returnedCols[i]->alias().c_str()) == 0 || strcasecmp(ord_item->name.str, gwi.returnedCols[i]->alias().c_str()) == 0) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); break; } } if (i == gwi.returnedCols.size()) ord_cols += string(" `") + escapeBackTick(ord_item->name.str) + '`'; } else if (ord_item->name.length) { // for union order by 1 case. For unknown reason, it doesn't show in_field_list if (ord_item->type() == Item::INT_ITEM) { ord_cols += ord_item->name.str; } else if (ord_item->type() == Item::SUBSELECT_ITEM) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } else { ord_cols += string(" `") + escapeBackTick(ord_item->name.str) + '`'; } } else if (ord_item->type() == Item::FUNC_ITEM) { // @bug5636. check if this order by column is on the select list ReturnedColumn* rc = buildFunctionColumn((Item_func*)(ord_item), gwi, gwi.fatalParseError); for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if (rc && rc->operator==(gwi.returnedCols[i].get())) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); break; } } } else { String str; ord_item->print(&str, QT_INFINIDB_NO_QUOTE); ord_cols += string(str.c_ptr()); } if (ordercol->direction != ORDER::ORDER_ASC) ord_cols += " desc"; } } if (ord_cols.length() > 0) // has order by { gwi.thd->infinidb_vtable.has_order_by = true; csep->hasOrderBy(true); ord_cols = " order by " + ord_cols; select_query += ord_cols; } } if (unionSel || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT) { if (select_lex.master_unit()->global_parameters()->explicit_limit) { if (select_lex.master_unit()->global_parameters()->offset_limit) { Item_int* offset = (Item_int*)select_lex.master_unit()->global_parameters()->offset_limit; csep->limitStart(offset->val_int()); } if (select_lex.master_unit()->global_parameters()->select_limit) { Item_int* select = (Item_int*)select_lex.master_unit()->global_parameters()->select_limit; csep->limitNum(select->val_int()); } if (unionSel && gwi.subSelectType == CalpontSelectExecutionPlan::MAIN_SELECT) { ostringstream limit; limit << " limit "; limit << csep->limitStart() << ", "; limit << csep->limitNum(); select_query += limit.str(); } } } else if (isUnion && select_lex.explicit_limit) { if (select_lex.braces) { if (select_lex.offset_limit) csep->limitStart(((Item_int*)select_lex.offset_limit)->val_int()); if (select_lex.select_limit) csep->limitNum(((Item_int*)select_lex.select_limit)->val_int()); } } else if (select_lex.explicit_limit) { uint32_t limitOffset = 0; uint32_t limitNum = std::numeric_limits::max(); if (join) { #if MYSQL_VERSION_ID >= 50172 // @bug5729. After upgrade, join->unit sometimes is uninitialized pointer // (not null though) and will cause seg fault. Prefer checking // select_lex->offset_limit if not null. if (join->select_lex && join->select_lex->offset_limit && join->select_lex->offset_limit->fixed && join->select_lex->select_limit && join->select_lex->select_limit->fixed) { limitOffset = join->select_lex->offset_limit->val_int(); limitNum = join->select_lex->select_limit->val_int(); } else if (join->unit) { limitOffset = join->unit->offset_limit_cnt; limitNum = join->unit->select_limit_cnt - limitOffset; } #else limitOffset = (join->unit)->offset_limit_cnt; limitNum = (join->unit)->select_limit_cnt - (join->unit)->offset_limit_cnt; #endif } else { if (select_lex.master_unit()->global_parameters()->offset_limit) { Item_int* offset = (Item_int*)select_lex.master_unit()->global_parameters()->offset_limit; limitOffset = offset->val_int(); } if (select_lex.master_unit()->global_parameters()->select_limit) { Item_int* select = (Item_int*)select_lex.master_unit()->global_parameters()->select_limit; limitNum = select->val_int(); } } // relate to bug4848. let mysql drive limit when limit session variable set. // do not set in csep. @bug5096. ignore session limit setting for dml if ((gwi.thd->variables.select_limit == (uint64_t) - 1 || (gwi.thd->variables.select_limit != (uint64_t) - 1 && gwi.thd->infinidb_vtable.vtable_state != THD::INFINIDB_CREATE_VTABLE)) && !csep->hasOrderBy()) { csep->limitStart(limitOffset); csep->limitNum(limitNum); } else { ostringstream limit; limit << " limit " << limitOffset << ", " << limitNum; select_query += limit.str(); } } gwi.thd->infinidb_vtable.select_vtable_query.free(); gwi.thd->infinidb_vtable.select_vtable_query.append(select_query.c_str(), select_query.length()); // We don't currently support limit with correlated subquery if (csep->limitNum() != (uint64_t) - 1 && gwi.subQuery && !gwi.correlatedTbNameVec.empty()) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_LIMIT_SUB); setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } } if (/*join->select_options*/select_lex.options & SELECT_DISTINCT) csep->distinct(true); // add the smallest column to count(*) parm. // select constant in subquery case std::vector::iterator coliter; if (!minSc) { if (!gwi.returnedCols.empty()) minSc = gwi.returnedCols[0]; else if (!gwi.additionalRetCols.empty()) minSc = gwi.additionalRetCols[0]; } // @bug3523, count(*) on subquery always pick column[0]. SimpleColumn* sc = dynamic_cast(minSc.get()); if (sc && sc->schemaName().empty()) { if (gwi.derivedTbList.size() >= 1) { SimpleColumn* sc1 = new SimpleColumn(); sc1->columnName(sc->columnName()); sc1->tableName(sc->tableName()); sc1->tableAlias(sc->tableAlias()); sc1->viewName(lower(sc->viewName())); sc1->colPosition(0); minSc.reset(sc1); } } for (coliter = gwi.count_asterisk_list.begin(); coliter != gwi.count_asterisk_list.end(); ++coliter) { // @bug5977 @note should never throw this, but checking just in case. // When ExeMgr fix is ready, this should not error out... if (dynamic_cast(minSc.get())) { gwi.fatalParseError = true; gwi.parseErrorText = "No project column found for aggregate function"; setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } // Replace the last (presumably constant) object with minSc if ((*coliter)->aggParms().empty()) { (*coliter)->aggParms().push_back(minSc); } else { (*coliter)->aggParms()[0] = minSc; } } std::vector::iterator funciter; SPTP sptp(new ParseTree(minSc.get()->clone())); for (funciter = gwi.no_parm_func_list.begin(); funciter != gwi.no_parm_func_list.end(); ++funciter) { FunctionParm funcParms = (*funciter)->functionParms(); funcParms.push_back(sptp); (*funciter)->functionParms(funcParms); } // set sequence# for subquery localCols for (uint32_t i = 0; i < gwi.localCols.size(); i++) gwi.localCols[i]->sequence(i); // append additionalRetCols to returnedCols gwi.returnedCols.insert(gwi.returnedCols.begin(), gwi.additionalRetCols.begin(), gwi.additionalRetCols.end()); csep->groupByCols(gwi.groupByCols); csep->orderByCols(gwi.orderByCols); csep->returnedCols(gwi.returnedCols); csep->columnMap(gwi.columnMap); csep->having(havingFilter); csep->derivedTableList(gwi.derivedTbList); csep->selectSubList(selectSubList); csep->subSelectList(gwi.subselectList); gwi.thd->infinidb_vtable.duplicate_field_name = false; clearStacks(gwi); return 0; } int cp_get_plan(THD* thd, SCSEP& csep) { LEX* lex = thd->lex; idbassert(lex != 0); SELECT_LEX select_lex = lex->select_lex; gp_walk_info gwi; gwi.thd = thd; int status = getSelectPlan(gwi, select_lex, csep); if (status > 0) return ER_INTERNAL_ERROR; else if (status < 0) return status; #ifdef DEBUG_WALK_COND cerr << "---------------- cp_get_plan EXECUTION PLAN ----------------" << endl; cerr << *csep << endl ; cerr << "-------------- EXECUTION PLAN END --------------\n" << endl; #endif // Derived table projection and filter optimization. derivedTableOptimization(csep); return 0; } int cp_get_table_plan(THD* thd, SCSEP& csep, cal_table_info& ti) { gp_walk_info* gwi = ti.condInfo; if (!gwi) gwi = new gp_walk_info(); gwi->thd = thd; LEX* lex = thd->lex; idbassert(lex != 0); uint32_t sessionID = csep->sessionID(); gwi->sessionid = sessionID; TABLE* table = ti.msTablePtr; boost::shared_ptr csc = CalpontSystemCatalog::makeCalpontSystemCatalog(sessionID); csc->identity(CalpontSystemCatalog::FE); // get all columns that mysql needs to fetch MY_BITMAP* read_set = table->read_set; Field** f_ptr, *field; gwi->columnMap.clear(); for (f_ptr = table->field ; (field = *f_ptr) ; f_ptr++) { if (bitmap_is_set(read_set, field->field_index)) { SimpleColumn* sc = new SimpleColumn(table->s->db.str, table->s->table_name.str, field->field_name.str, sessionID); string alias(table->alias.c_ptr()); sc->tableAlias(lower(alias)); assert (sc); boost::shared_ptr spsc(sc); gwi->returnedCols.push_back(spsc); gwi->columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(field->field_name.str), spsc)); } } if (gwi->columnMap.empty()) { CalpontSystemCatalog::TableName tn = make_table(table->s->db.str, table->s->table_name.str); CalpontSystemCatalog::TableAliasName tan = make_aliastable(table->s->db.str, table->s->table_name.str, table->alias.c_ptr()); SimpleColumn* sc = getSmallestColumn(csc, tn, tan, table, *gwi); SRCP srcp(sc); gwi->columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(sc->columnName(), srcp)); gwi->returnedCols.push_back(srcp); } // get filter if (ti.condInfo) { gp_walk_info* gwi = ti.condInfo; ParseTree* filters = 0; ParseTree* ptp = 0; ParseTree* rhs = 0; while (!gwi->ptWorkStack.empty()) { filters = gwi->ptWorkStack.top(); gwi->ptWorkStack.pop(); SimpleFilter* sf = dynamic_cast(filters->data()); if (sf && sf->op()->data() == "noop") { delete filters; filters = 0; if (gwi->ptWorkStack.empty()) break; continue; } if (gwi->ptWorkStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); ptp->left(filters); rhs = gwi->ptWorkStack.top(); gwi->ptWorkStack.pop(); ptp->right(rhs); gwi->ptWorkStack.push(ptp); } csep->filters(filters); } csep->returnedCols(gwi->returnedCols); csep->columnMap(gwi->columnMap); CalpontSelectExecutionPlan::TableList tblist; tblist.push_back(make_aliastable(table->s->db.str, table->s->table_name.str, table->alias.c_ptr())); csep->tableList(tblist); // @bug 3321. Set max number of blocks in a dictionary file to be scanned for filtering csep->stringScanThreshold(gwi->thd->variables.infinidb_string_scan_threshold); return 0; } int cp_get_group_plan(THD* thd, SCSEP& csep, cal_impl_if::cal_group_info& gi) { SELECT_LEX *select_lex = gi.groupByTables->select_lex; gp_walk_info gwi; gwi.thd = thd; int status = getGroupPlan(gwi, *select_lex, csep, gi); #ifdef DEBUG_WALK_COND cerr << "---------------- cp_get_group_plan EXECUTION PLAN ----------------" << endl; cerr << *csep << endl ; cerr << "-------------- EXECUTION PLAN END --------------\n" << endl; #endif if (status > 0) return ER_INTERNAL_ERROR; else if (status < 0) return status; return 0; } /*@brief buildConstColFromFilter- change SimpleColumn into ConstColumn*/ /*********************************************************** * DESCRIPTION: * Server could optimize out fields from GROUP BY list, when certain * filter predicate is used, e.g. * field = 'AIR', field IN ('AIR'). This utility function tries to * replace such fields with ConstantColumns using cond_pushed filters. * PARAMETERS: * originalSC SimpleColumn* removed field * gwi main strucutre * gi auxilary group_by handler structure * RETURNS * ConstantColumn* if originalSC equals with cond_pushed columns. * NULL otherwise ***********************************************************/ ConstantColumn* buildConstColFromFilter(SimpleColumn* originalSC, gp_walk_info& gwi, cal_group_info& gi) { execplan::SimpleColumn* simpleCol; execplan::ConstantColumn* constCol; execplan::SOP op; execplan::SimpleFilter* simpFilter; execplan::ConstantColumn* result = NULL; std::vector::iterator ptIt = gi.pushedPts.begin(); for (; ptIt != gi.pushedPts.end(); ptIt++) { simpFilter = dynamic_cast((*ptIt)->data()); if (simpFilter == NULL) continue; simpleCol = dynamic_cast(simpFilter->lhs()); constCol = dynamic_cast(simpFilter->rhs()); if (simpleCol == NULL || constCol == NULL) continue; op = simpFilter->op(); if ( originalSC->sameColumn(dynamic_cast(simpleCol)) && op.get()->op() == OP_EQ && constCol) { #ifdef DEBUG_WALK_COND cerr << "buildConstColFromFilter() replaced " << endl; cerr << simpleCol->toString() << endl; cerr << " with " << endl; cerr << constCol << endl; #endif result = constCol; } } return result; } int getGroupPlan(gp_walk_info& gwi, SELECT_LEX& select_lex, SCSEP& csep, cal_group_info& gi, bool isUnion) { #ifdef DEBUG_WALK_COND cerr << "getGroupPlan()" << endl; #endif // rollup is currently not supported if (select_lex.olap == ROLLUP_TYPE) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_ROLLUP_NOT_SUPPORT); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } gwi.internalDecimalScale = (gwi.thd->variables.infinidb_use_decimal_scale ? gwi.thd->variables.infinidb_decimal_scale : -1); gwi.subSelectType = csep->subType(); JOIN* join = select_lex.join; Item_cond* icp = 0; if ( gi.groupByWhere ) icp = reinterpret_cast(gi.groupByWhere); uint32_t sessionID = csep->sessionID(); gwi.sessionid = sessionID; boost::shared_ptr csc = CalpontSystemCatalog::makeCalpontSystemCatalog(sessionID); csc->identity(CalpontSystemCatalog::FE); gwi.csc = csc; // @bug 2123. Override large table estimate if infinidb_ordered hint was used. // @bug 2404. Always override if the infinidb_ordered_only variable is turned on. if (gwi.thd->infinidb_vtable.override_largeside_estimate || gwi.thd->variables.infinidb_ordered_only) csep->overrideLargeSideEstimate(true); // @bug 5741. Set a flag when in Local PM only query mode csep->localQuery(gwi.thd->variables.infinidb_local_query); // @bug 3321. Set max number of blocks in a dictionary file to be scanned for filtering csep->stringScanThreshold(gwi.thd->variables.infinidb_string_scan_threshold); csep->stringTableThreshold(gwi.thd->variables.infinidb_stringtable_threshold); csep->djsSmallSideLimit(gwi.thd->variables.infinidb_diskjoin_smallsidelimit * 1024ULL * 1024); csep->djsLargeSideLimit(gwi.thd->variables.infinidb_diskjoin_largesidelimit * 1024ULL * 1024); csep->djsPartitionSize(gwi.thd->variables.infinidb_diskjoin_bucketsize * 1024ULL * 1024); if (gwi.thd->variables.infinidb_um_mem_limit == 0) csep->umMemLimit(numeric_limits::max()); else csep->umMemLimit(gwi.thd->variables.infinidb_um_mem_limit * 1024ULL * 1024); // populate table map and trigger syscolumn cache for all the tables (@bug 1637). // all tables on FROM list must have at least one col in colmap TABLE_LIST* table_ptr = gi.groupByTables; CalpontSelectExecutionPlan::SelectList derivedTbList; // DEBUG #ifdef DEBUG_WALK_COND List_iterator sj_list_it(select_lex.sj_nests); TABLE_LIST* sj_nest; while ((sj_nest = sj_list_it++)) { cerr << sj_nest->db.str << "." << sj_nest->table_name.str << endl; } #endif // @bug 1796. Remember table order on the FROM list. gwi.clauseType = FROM; try { for (; table_ptr; table_ptr = table_ptr->next_local) { // mysql put vtable here for from sub. we ignore it //if (string(table_ptr->table_name).find("$vtable") != string::npos) // continue; // Until we handle recursive cte: // Checking here ensures we catch all with clauses in the query. if (table_ptr->is_recursive_with_table()) { gwi.fatalParseError = true; gwi.parseErrorText = "Recursive CTE"; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } string viewName = getViewName(table_ptr); // @todo process from subquery if (table_ptr->derived) { String str; (table_ptr->derived->first_select())->print(gwi.thd, &str, QT_INFINIDB_DERIVED); SELECT_LEX* select_cursor = table_ptr->derived->first_select(); FromSubQuery fromSub(gwi, select_cursor); string alias(table_ptr->alias.str); fromSub.alias(lower(alias)); CalpontSystemCatalog::TableAliasName tn = make_aliasview("", "", alias, viewName); // @bug 3852. check return execplan SCSEP plan = fromSub.transform(); if (!plan) { setError(gwi.thd, ER_INTERNAL_ERROR, fromSub.gwip().parseErrorText, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } gwi.derivedTbList.push_back(plan); gwi.tbList.push_back(tn); CalpontSystemCatalog::TableAliasName tan = make_aliastable("", alias, alias); gwi.tableMap[tan] = make_pair(0, table_ptr); // gwi.thd->infinidb_vtable.isUnion = true; //by-pass the 2nd pass of rnd_init } else if (table_ptr->view) { View* view = new View(table_ptr->view->select_lex, &gwi); CalpontSystemCatalog::TableAliasName tn = make_aliastable(table_ptr->db.str, table_ptr->table_name.str, table_ptr->alias.str); view->viewName(tn); gwi.viewList.push_back(view); view->transform(); } else { // check foreign engine tables bool infiniDB = (table_ptr->table ? isInfiniDB(table_ptr->table) : true); // trigger system catalog cache if (infiniDB) csc->columnRIDs(make_table(table_ptr->db.str, table_ptr->table_name.str), true); string table_name = table_ptr->table_name.str; // @bug5523 if (table_ptr->db.length && strcmp(table_ptr->db.str, "information_schema") == 0) table_name = (table_ptr->schema_table_name.length ? table_ptr->schema_table_name.str : table_ptr->alias.str); CalpontSystemCatalog::TableAliasName tn = make_aliasview(table_ptr->db.str, table_name, table_ptr->alias.str, viewName, infiniDB); gwi.tbList.push_back(tn); CalpontSystemCatalog::TableAliasName tan = make_aliastable(table_ptr->db.str, table_name, table_ptr->alias.str, infiniDB); gwi.tableMap[tan] = make_pair(0, table_ptr); #ifdef DEBUG_WALK_COND cerr << tn << endl; #endif } } if (gwi.fatalParseError) { setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_INTERNAL_ERROR; } } catch (IDBExcept& ie) { setError(gwi.thd, ER_INTERNAL_ERROR, ie.what(), gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); // @bug 3852. set error status for gwi. gwi.fatalParseError = true; gwi.parseErrorText = ie.what(); return ER_INTERNAL_ERROR; } catch (...) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_LOST_CONN_EXEMGR); // @bug3852 set error status for gwi. gwi.fatalParseError = true; gwi.parseErrorText = emsg; setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } csep->tableList(gwi.tbList); bool unionSel = false; gwi.clauseType = WHERE; if (icp) { // MCOL-1052 The condition could be useless. // MariaDB bug 624 - without the fix_fields call, delete with join may error with "No query step". //#if MYSQL_VERSION_ID < 50172 //@bug 3039. fix fields for constants if (!icp->fixed) { icp->fix_fields(gwi.thd, (Item**)&icp); } //#endif gwi.fatalParseError = false; #ifdef DEBUG_WALK_COND cerr << "------------------ WHERE -----------------------" << endl; icp->traverse_cond(debug_walk, &gwi, Item::POSTFIX); cerr << "------------------------------------------------\n" << endl; #endif icp->traverse_cond(gp_walk, &gwi, Item::POSTFIX); if (gwi.fatalParseError) { // if this is dervied table process phase, mysql may have not developed the plan // completely. Do not error and eventually mysql will call JOIN::exec() again. // related to bug 2922. Need to find a way to skip calling rnd_init for derived table // processing. if (gwi.thd->derived_tables_processing) { gwi.thd->infinidb_vtable.isUnion = false; gwi.thd->infinidb_vtable.isUpdateWithDerive = true; return -1; } setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_INTERNAL_ERROR; } } else if (join && join->zero_result_cause) { gwi.rcWorkStack.push(new ConstantColumn((int64_t)0, ConstantColumn::NUM)); } SELECT_LEX tmp_select_lex; tmp_select_lex.table_list.first = gi.groupByTables; uint32_t failed = buildOuterJoin(gwi, tmp_select_lex); if (failed) return failed; // @bug5764. build outer join for view, make sure outerjoin filter is appended // to the end of the filter list. for (uint i = 0; i < gwi.viewList.size(); i++) { failed = gwi.viewList[i]->processOuterJoin(gwi); if (failed) break; } if (failed != 0) return failed; ParseTree* filters = NULL; ParseTree* ptp = NULL; ParseTree* lhs = NULL; // @bug 2932. for "select * from region where r_name" case. if icp not null and // ptWorkStack empty, the item is in rcWorkStack. // MySQL 5.6 (MariaDB?). when icp is null and zero_result_cause is set, a constant 0 // is pushed to rcWorkStack. if (/*icp && */gwi.ptWorkStack.empty() && !gwi.rcWorkStack.empty()) { filters = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } while (!gwi.ptWorkStack.empty()) { filters = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); if (gwi.ptWorkStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); //ptp->left(filters); ptp->right(filters); lhs = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); //ptp->right(rhs); ptp->left(lhs); gwi.ptWorkStack.push(ptp); } if (filters) { csep->filters(filters); #ifdef DEBUG_WALK_COND std::string aTmpDir(startup::StartUp::tmpDir()); aTmpDir = aTmpDir + "/filter1.dot"; filters->drawTree(aTmpDir); #endif } gwi.clauseType = SELECT; #ifdef DEBUG_WALK_COND { cerr << "------------------- SELECT --------------------" << endl; List_iterator_fast it(*gi.groupByFields); Item* item; while ((item = it++)) { debug_walk(item, 0); } cerr << "-----------------------------------------------\n" << endl; } #endif // populate returnedcolumnlist and columnmap List_iterator_fast it(*gi.groupByFields); Item* item; vector funcFieldVec; string sel_cols_in_create; string sel_cols_in_select; bool redo = false; // empty rcWorkStack and ptWorkStack. They should all be empty by now. clearStacks(gwi); // indicate the starting pos of scalar returned column, because some join column // has been inserted to the returned column list. if (gwi.subQuery) { ScalarSub* scalar = dynamic_cast(gwi.subQuery); if (scalar) scalar->returnedColPos(gwi.additionalRetCols.size()); } CalpontSelectExecutionPlan::SelectList selectSubList; while ((item = it++)) { string itemAlias; if(item->name.length) itemAlias = (item->name.str); else { itemAlias = ""; } // @bug 5916. Need to keep checking until getting concret item in case // of nested view. while (item->type() == Item::REF_ITEM) { Item_ref* ref = (Item_ref*)item; item = (*(ref->ref)); } Item::Type itype = item->type(); switch (itype) { case Item::FIELD_ITEM: { Item_field* ifp = (Item_field*)item; SimpleColumn* sc = NULL; ConstantColumn* constCol = NULL; if (ifp->field_name.length && string(ifp->field_name.str) == "*") { collectAllCols(gwi, ifp); break; } sc = buildSimpleColumn(ifp, gwi); if (sc) { constCol = buildConstColFromFilter(sc, gwi, gi); boost::shared_ptr spcc(constCol); boost::shared_ptr spsc(sc); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; string fullname; String str; ifp->print(&str, QT_INFINIDB_NO_QUOTE); fullname = str.c_ptr(); //sel_cols_in_create += fullname; if (ifp->is_autogenerated_name) // no alias { sel_cols_in_create += fullname + " `" + escapeBackTick(str.c_ptr()) + "`"; sc->alias(fullname); } else // alias { if (!itemAlias.empty()) sc->alias(itemAlias); sel_cols_in_create += fullname + " `" + escapeBackTick(sc->alias().c_str()) + "`"; } if (ifp->is_autogenerated_name) gwi.selectCols.push_back("`" + escapeBackTick(fullname.c_str()) + "`" + " `" + escapeBackTick(itemAlias.empty() ? ifp->name.str : itemAlias.c_str()) + "`"); else gwi.selectCols.push_back("`" + escapeBackTick((itemAlias.empty() ? ifp->name.str : itemAlias.c_str())) + "`"); // MCOL-1052 Replace SimpleColumn with ConstantColumn, // since it must have a single value only. if (constCol) { gwi.returnedCols.push_back(spcc); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), spcc)); } else { gwi.returnedCols.push_back(spsc); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), spsc)); } TABLE_LIST* tmp = 0; if (ifp->cached_table) tmp = ifp->cached_table; gwi.tableMap[make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias(), sc->isInfiniDB())] = make_pair(1, tmp); } else { setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); delete sc; return ER_INTERNAL_ERROR; } break; } //aggregate column case Item::SUM_FUNC_ITEM: { ReturnedColumn* ac = buildAggregateColumn(item, gwi); if (gwi.fatalParseError) { // e.g., non-support ref column setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); delete ac; return ER_CHECK_NOT_IMPLEMENTED; } // add this agg col to returnedColumnList boost::shared_ptr spac(ac); gwi.returnedCols.push_back(spac); // This item could be used in projection or HAVING later. gwi.extSelAggColsItems.push_back(item); gwi.selectCols.push_back('`' + escapeBackTick(spac->alias().c_str()) + '`'); String str(256); item->print(&str, QT_INFINIDB_NO_QUOTE); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += string(str.c_ptr()) + " `" + escapeBackTick(spac->alias().c_str()) + "`"; break; } case Item::FUNC_ITEM: { Item_func* ifp = reinterpret_cast(item); // @bug4383. error out non-support stored function if (ifp->functype() == Item_func::FUNC_SP) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_SP_FUNCTION_NOT_SUPPORT); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } if (string(ifp->func_name()) == "xor") { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } uint16_t parseInfo = 0; vector tmpVec; bool hasNonSupportItem = false; parse_item(ifp, tmpVec, hasNonSupportItem, parseInfo); if (ifp->with_subquery() || string(ifp->func_name()) == string("") || ifp->functype() == Item_func::NOT_ALL_FUNC || parseInfo & SUB_BIT) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_SELECT_SUB); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } ReturnedColumn* rc = buildFunctionColumn(ifp, gwi, hasNonSupportItem, true); SRCP srcp(rc); if (rc) { if (!hasNonSupportItem && !nonConstFunc(ifp) && !(parseInfo & AF_BIT) && tmpVec.size() == 0) { if (isUnion || unionSel || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT || parseInfo & SUB_BIT ) //|| select_lex.group_list.elements != 0) { srcp.reset(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (ifp->name.length) srcp->alias(ifp->name.str); continue; } if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) { } else { redo = true; String str; ifp->print(&str, QT_INFINIDB_NO_QUOTE); gwi.selectCols.push_back(string(str.c_ptr()) + " " + "`" + escapeBackTick(item->name.str) + "`"); } break; } //SRCP srcp(rc); gwi.returnedCols.push_back(srcp); if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { String str(256); ifp->print(&str, QT_INFINIDB_NO_QUOTE); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += string(str.c_ptr()) + " `" + ifp->name.str + "`"; gwi.selectCols.push_back("`" + escapeBackTick(ifp->name.str) + "`"); } } else // InfiniDB Non support functions still go through post process for now { hasNonSupportItem = false; uint32_t before_size = funcFieldVec.size(); // MCOL-1510 Use gwi pointer here to catch funcs with // not supported aggregate args in projections, // e.g. NOT(SUM(i)). parse_item(ifp, funcFieldVec, hasNonSupportItem, parseInfo, &gwi); uint32_t after_size = funcFieldVec.size(); // group by func and func in subquery can not be post processed // @bug3881. set_user_var can not be treated as constant function // @bug5716. Try to avoid post process function for union query. if ((gwi.subQuery /*|| select_lex.group_list.elements != 0 */ || !csep->unionVec().empty() || isUnion) && !hasNonSupportItem && (after_size - before_size) == 0 && !(parseInfo & AGG_BIT) && !(parseInfo & SUB_BIT) && string(ifp->func_name()) != "set_user_var") { String val, *str = ifp->val_str(&val); string valStr; if (str) valStr.assign(str->ptr(), str->length()); ConstantColumn* cc = NULL; if (!str) { cc = new ConstantColumn("", ConstantColumn::NULLDATA); } else if (ifp->result_type() == STRING_RESULT) { cc = new ConstantColumn(valStr, ConstantColumn::LITERAL); } else if (ifp->result_type() == DECIMAL_RESULT) { cc = buildDecimalColumn(ifp, gwi); } else { cc = new ConstantColumn(valStr, ConstantColumn::NUM); cc->resultType(colType_MysqlToIDB(item)); } SRCP srcp(cc); if (ifp->name.length) cc->alias(ifp->name.str); gwi.returnedCols.push_back(srcp); // clear the error set by buildFunctionColumn gwi.fatalParseError = false; gwi.parseErrorText = ""; break; } else if (hasNonSupportItem || parseInfo & AGG_BIT || parseInfo & SUB_BIT || (gwi.fatalParseError && gwi.subQuery)) { if (gwi.parseErrorText.empty()) { Message::Args args; args.add(ifp->func_name()); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORTED_FUNCTION, args); } setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } else if ( gwi.subQuery && (isPredicateFunction(ifp, &gwi) || ifp->type() == Item::COND_ITEM )) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } //@Bug 3030 Add error check for dml statement if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) { if ( after_size - before_size != 0 ) { gwi.parseErrorText = ifp->func_name(); return -1; } } //@Bug 3021. Bypass postprocess for update and delete. //if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) //{} else { // @bug 3881. Here is the real redo part. redo = true; // @bug 1706 String funcStr; ifp->print(&funcStr, QT_INFINIDB); gwi.selectCols.push_back(string(funcStr.c_ptr()) + " `" + escapeBackTick(ifp->name.str) + "`"); // clear the error set by buildFunctionColumn gwi.fatalParseError = false; gwi.parseErrorText = ""; } } break; } case Item::INT_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { // do not push the dummy column (mysql added) to returnedCol if (item->name.length && string(item->name.str) == "Not_used") continue; // @bug3509. Constant column is sent to ExeMgr now. SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); if (item->name.length) srcp->alias(item->name.str); gwi.returnedCols.push_back(srcp); Item_int* isp = reinterpret_cast(item); ostringstream oss; oss << isp->value << " `" << escapeBackTick(srcp->alias().c_str()) << "`"; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += oss.str(); gwi.selectCols.push_back(oss.str()); } break; } case Item::STRING_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (item->name.length) srcp->alias(item->name.str); Item_string* isp = reinterpret_cast(item); String val, *str = isp->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); string name = "'" + valStr + "'" + " " + "`" + escapeBackTick(srcp->alias().c_str()) + "`"; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += name; gwi.selectCols.push_back(name); } break; } case Item::DECIMAL_ITEM: { if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI )) { } else { SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (item->name.length) srcp->alias(item->name.str); Item_decimal* isp = reinterpret_cast(item); String val, *str = isp->val_str(&val); string valStr; valStr.assign(str->ptr(), str->length()); ostringstream oss; oss << valStr.c_str() << " `" << escapeBackTick(srcp->alias().c_str()) << "`"; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += oss.str(); gwi.selectCols.push_back(oss.str()); } break; } case Item::NULL_ITEM: { /*if ( ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE ) || ((gwi.thd->lex)->sql_command == SQLCOM_UPDATE_MULTI ) || ((gwi.thd->lex)->sql_command == SQLCOM_DELETE_MULTI ) ) { } else { SRCP srcp(buildReturnedColumn(item, gwi, gwi.fatalParseError)); gwi.returnedCols.push_back(srcp); if (item->name) srcp->alias(item->name); string name = string("null `") + escapeBackTick(srcp->alias().c_str()) + string("`") ; if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; sel_cols_in_create += name; gwi.selectCols.push_back("null"); }*/ break; } case Item::SUBSELECT_ITEM: { Item_subselect* sub = (Item_subselect*)item; if (sub->substype() != Item_subselect::SINGLEROW_SUBS) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_SELECT_SUB); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } #ifdef DEBUG_WALK_COND cerr << "SELECT clause SUBSELECT Item: " << sub->substype() << endl; JOIN* join = sub->get_select_lex()->join; if (join) { Item_cond* cond = reinterpret_cast(join->conds); if (cond) cond->traverse_cond(debug_walk, &gwi, Item::POSTFIX); } cerr << "Finish SELECT clause subselect item traversing" << endl; #endif SelectSubQuery* selectSub = new SelectSubQuery(gwi, sub); //selectSub->gwip(&gwi); SCSEP ssub = selectSub->transform(); if (!ssub || gwi.fatalParseError) { if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Unsupported Item in SELECT subquery."; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } selectSubList.push_back(ssub); SimpleColumn* rc = new SimpleColumn(); rc->colSource(rc->colSource() | SELECT_SUB); if (sub->get_select_lex()->get_table_list()) rc->viewName(lower(getViewName(sub->get_select_lex()->get_table_list()))); if (sub->name.length) rc->alias(sub->name.str); gwi.returnedCols.push_back(SRCP(rc)); String str; sub->get_select_lex()->print(gwi.thd, &str, QT_INFINIDB_NO_QUOTE); sel_cols_in_create += "(" + string(str.c_ptr()) + ")"; if (sub->name.length) { sel_cols_in_create += "`" + escapeBackTick(sub->name.str) + "`"; gwi.selectCols.push_back(sub->name.str); } else { sel_cols_in_create += "`" + escapeBackTick(str.c_ptr()) + "`"; gwi.selectCols.push_back("`" + escapeBackTick(str.c_ptr()) + "`"); } break; } case Item::COND_ITEM: { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_FILTER_COND_EXP); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } case Item::EXPR_CACHE_ITEM: { printf("EXPR_CACHE_ITEM in getSelectPlan\n"); gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_UNKNOWN_COL); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } case Item::WINDOW_FUNC_ITEM: { SRCP srcp(buildWindowFunctionColumn(item, gwi, gwi.fatalParseError)); if (!srcp || gwi.fatalParseError) { if (gwi.parseErrorText.empty()) gwi.parseErrorText = "Unsupported Item in SELECT subquery."; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } gwi.returnedCols.push_back(srcp); break; } default: { break; } } } // @bug4388 normalize the project coltypes for union main select list if (!csep->unionVec().empty()) { for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { vector coltypes; for (uint32_t j = 0; j < csep->unionVec().size(); j++) { coltypes.push_back( dynamic_cast(csep->unionVec()[j].get())->returnedCols()[i]->resultType()); // @bug5976. set hasAggregate true for the main column if // one corresponding union column has aggregate if (dynamic_cast(csep->unionVec()[j].get())->returnedCols()[i]->hasAggregate()) gwi.returnedCols[i]->hasAggregate(true); } gwi.returnedCols[i]->resultType(dataconvert::DataConvert::convertUnionColType(coltypes)); } } // Having clause handling gwi.clauseType = HAVING; clearStacks(gwi); ParseTree* havingFilter = 0; // clear fatalParseError that may be left from post process functions gwi.fatalParseError = false; gwi.parseErrorText = ""; if (gi.groupByHaving != 0) { Item_cond* having = reinterpret_cast(gi.groupByHaving); #ifdef DEBUG_WALK_COND cerr << "------------------- HAVING ---------------------" << endl; having->traverse_cond(debug_walk, &gwi, Item::POSTFIX); cerr << "------------------------------------------------\n" << endl; #endif having->traverse_cond(gp_walk, &gwi, Item::POSTFIX); if (gwi.fatalParseError) { setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_INTERNAL_ERROR; } ParseTree* ptp = 0; ParseTree* rhs = 0; // @bug 4215. some function filter will be in the rcWorkStack. if (gwi.ptWorkStack.empty() && !gwi.rcWorkStack.empty()) { havingFilter = new ParseTree(gwi.rcWorkStack.top()); gwi.rcWorkStack.pop(); } while (!gwi.ptWorkStack.empty()) { havingFilter = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); if (gwi.ptWorkStack.empty()) break; ptp = new ParseTree(new LogicOperator("and")); ptp->left(havingFilter); rhs = gwi.ptWorkStack.top(); gwi.ptWorkStack.pop(); ptp->right(rhs); gwi.ptWorkStack.push(ptp); } } // for post process expressions on the select list // error out post process for union and sub select unit if (isUnion || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT) { if (funcFieldVec.size() != 0 && !gwi.fatalParseError) { string emsg("Fatal parse error in vtable mode: Unsupported Items in union or sub select unit"); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg); return ER_CHECK_NOT_IMPLEMENTED; } } for (uint32_t i = 0; i < funcFieldVec.size(); i++) { SimpleColumn* sc = buildSimpleColumn(funcFieldVec[i], gwi); if (!sc || gwi.fatalParseError) { string emsg; if (gwi.parseErrorText.empty()) { emsg = "un-recognized column"; if (funcFieldVec[i]->name.length) emsg += string(funcFieldVec[i]->name.str); } else { emsg = gwi.parseErrorText; } setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); return ER_INTERNAL_ERROR; } String str; funcFieldVec[i]->print(&str, QT_INFINIDB_NO_QUOTE); sc->alias(string(str.c_ptr())); //sc->tableAlias(funcFieldVec[i]->table_name); sc->tableAlias(sc->tableAlias()); SRCP srcp(sc); uint32_t j = 0; for (; j < gwi.returnedCols.size(); j++) { if (sc->sameColumn(gwi.returnedCols[j].get())) { SimpleColumn* field = dynamic_cast(gwi.returnedCols[j].get()); if (field && field->alias() == sc->alias()) break; } } if (j == gwi.returnedCols.size()) { gwi.returnedCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(funcFieldVec[i]->field_name.str), srcp)); if (sel_cols_in_create.length() != 0) sel_cols_in_create += ", "; string fullname; fullname = str.c_ptr(); sel_cols_in_create += fullname + " `" + escapeBackTick(fullname.c_str()) + "`"; TABLE_LIST* tmp = (funcFieldVec[i]->cached_table ? funcFieldVec[i]->cached_table : 0); gwi.tableMap[make_aliastable(sc->schemaName(), sc->tableName(), sc->tableAlias(), sc->isInfiniDB())] = make_pair(1, tmp); } } // post-process Order by list and expressions on select by redo phase1. only for vtable // ignore ORDER BY clause for union select unit string ord_cols = ""; // for normal select phase SRCP minSc; // min width projected column. for count(*) use // Group by list. not valid for union main query if (gwi.thd->infinidb_vtable.vtable_state == THD::INFINIDB_CREATE_VTABLE && !unionSel) { gwi.clauseType = GROUP_BY; Item* nonSupportItem = NULL; ORDER* groupcol = reinterpret_cast(gi.groupByGroup); // check if window functions are in order by. InfiniDB process order by list if // window functions are involved, either in order by or projection. bool hasWindowFunc = gwi.hasWindowFunc; gwi.hasWindowFunc = false; for (; groupcol; groupcol = groupcol->next) { if ((*(groupcol->item))->type() == Item::WINDOW_FUNC_ITEM) gwi.hasWindowFunc = true; } if (gwi.hasWindowFunc) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_WF_NOT_ALLOWED, "GROUP BY clause"); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } gwi.hasWindowFunc = hasWindowFunc; groupcol = reinterpret_cast(gi.groupByGroup); for (; groupcol; groupcol = groupcol->next) { Item* groupItem = *(groupcol->item); // @bug5993. Could be nested ref. while (groupItem->type() == Item::REF_ITEM) groupItem = (*((Item_ref*)groupItem)->ref); if (groupItem->type() == Item::FUNC_ITEM) { Item_func* ifp = (Item_func*)groupItem; // call buildFunctionColumn here mostly for finding out // non-support column on GB list. Should be simplified. ReturnedColumn* fc = buildFunctionColumn(ifp, gwi, gwi.fatalParseError); if (!fc || gwi.fatalParseError) { nonSupportItem = ifp; break; } if (groupcol->in_field_list && groupcol->counter_used) { delete fc; fc = gwi.returnedCols[groupcol->counter - 1].get(); SRCP srcp(fc->clone()); // check if no column parm for (uint32_t i = 0; i < gwi.no_parm_func_list.size(); i++) { if (gwi.no_parm_func_list[i]->expressionId() == fc->expressionId()) { gwi.no_parm_func_list.push_back(dynamic_cast(srcp.get())); break; } } srcp->orderPos(groupcol->counter - 1); gwi.groupByCols.push_back(srcp); continue; } else if (!groupItem->is_autogenerated_name) // alias { uint32_t i = 0; for (; i < gwi.returnedCols.size(); i++) { if (string(groupItem->name.str) == gwi.returnedCols[i]->alias()) { ReturnedColumn* rc = gwi.returnedCols[i]->clone(); rc->orderPos(i); gwi.groupByCols.push_back(SRCP(rc)); delete fc; break; } } if (i == gwi.returnedCols.size()) { nonSupportItem = groupItem; break; } } else { uint32_t i = 0; for (; i < gwi.returnedCols.size(); i++) { if (fc->operator==(gwi.returnedCols[i].get())) { ReturnedColumn* rc = gwi.returnedCols[i]->clone(); rc->orderPos(i); gwi.groupByCols.push_back(SRCP(rc)); delete fc; break; } } if (i == gwi.returnedCols.size()) { gwi.groupByCols.push_back(SRCP(fc)); break; } } } else if (groupItem->type() == Item::FIELD_ITEM) { Item_field* ifp = (Item_field*)groupItem; // this GB col could be an alias of F&E on the SELECT clause, not necessarily a field. ReturnedColumn* rc = buildSimpleColumn(ifp, gwi); SimpleColumn* sc = dynamic_cast(rc); for (uint32_t j = 0; j < gwi.returnedCols.size(); j++) { if (sc) { if (sc->sameColumn(gwi.returnedCols[j].get())) { sc->orderPos(j); break; } else if (strcasecmp(sc->alias().c_str(), gwi.returnedCols[j]->alias().c_str()) == 0) { rc = gwi.returnedCols[j].get()->clone(); rc->orderPos(j); break; } } else { if (ifp->name.length && string(ifp->name.str) == gwi.returnedCols[j].get()->alias()) { rc = gwi.returnedCols[j].get()->clone(); rc->orderPos(j); break; } } } if (!rc) { nonSupportItem = ifp; break; } SRCP srcp(rc); // bug 3151 AggregateColumn* ac = dynamic_cast(rc); if (ac) { nonSupportItem = ifp; break; } gwi.groupByCols.push_back(srcp); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(string(ifp->field_name.str), srcp)); } // @bug5638. The group by column is constant but not counter, alias has to match a column // on the select list else if (!groupcol->counter_used && (groupItem->type() == Item::INT_ITEM || groupItem->type() == Item::STRING_ITEM || groupItem->type() == Item::REAL_ITEM || groupItem->type() == Item::DECIMAL_ITEM)) { ReturnedColumn* rc = 0; for (uint32_t j = 0; j < gwi.returnedCols.size(); j++) { if (groupItem->name.length && string(groupItem->name.str) == gwi.returnedCols[j].get()->alias()) { rc = gwi.returnedCols[j].get()->clone(); rc->orderPos(j); break; } } if (!rc) { nonSupportItem = groupItem; break; } gwi.groupByCols.push_back(SRCP(rc)); } else if ((*(groupcol->item))->type() == Item::SUBSELECT_ITEM) { if (!groupcol->in_field_list || !groupItem->name.length) { nonSupportItem = groupItem; } else { uint32_t i = 0; for (; i < gwi.returnedCols.size(); i++) { if (string(groupItem->name.str) == gwi.returnedCols[i]->alias()) { ReturnedColumn* rc = gwi.returnedCols[i]->clone(); rc->orderPos(i); gwi.groupByCols.push_back(SRCP(rc)); break; } } if (i == gwi.returnedCols.size()) { nonSupportItem = groupItem; } } } // @bug 3761. else if (groupcol->counter_used) { if (gwi.returnedCols.size() <= (uint32_t)(groupcol->counter - 1)) { nonSupportItem = groupItem; } else { gwi.groupByCols.push_back(SRCP(gwi.returnedCols[groupcol->counter - 1]->clone())); } } else { nonSupportItem = groupItem; } } // @bug 4756. Add internal groupby column for correlated join to the groupby list if (gwi.aggOnSelect && !gwi.subGroupByCols.empty()) gwi.groupByCols.insert(gwi.groupByCols.end(), gwi.subGroupByCols.begin(), gwi.subGroupByCols.end()); // this is window func on SELECT becuase ORDER BY has not been processed if (!gwi.windowFuncList.empty() && !gwi.subGroupByCols.empty()) { for (uint32_t i = 0; i < gwi.windowFuncList.size(); i++) { if (gwi.windowFuncList[i]->hasWindowFunc()) { vector windowFunctions = gwi.windowFuncList[i]->windowfunctionColumnList(); for (uint32_t j = 0; j < windowFunctions.size(); j++) windowFunctions[j]->addToPartition(gwi.subGroupByCols); } } } if (nonSupportItem) { Message::Args args; if (nonSupportItem->name.length) args.add("'" + string(nonSupportItem->name.str) + "'"); else args.add(""); gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_GROUP_BY, args); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } } // GROUP processing ends here if (gwi.thd->infinidb_vtable.vtable_state == THD::INFINIDB_CREATE_VTABLE) { ORDER* ordercol = reinterpret_cast(gi.groupByOrder); string create_query(gwi.thd->infinidb_vtable.create_vtable_query.c_ptr()); string select_query(gwi.thd->infinidb_vtable.select_vtable_query.c_ptr()); string lower_create_query(gwi.thd->infinidb_vtable.create_vtable_query.c_ptr()); string lower_select_query(gwi.thd->infinidb_vtable.select_vtable_query.c_ptr()); boost::algorithm::to_lower(lower_create_query); boost::algorithm::to_lower(lower_select_query); // check if window functions are in order by. InfiniDB process order by list if // window functions are involved, either in order by or projection. for (; ordercol; ordercol = ordercol->next) { if ((*(ordercol->item))->type() == Item::WINDOW_FUNC_ITEM) gwi.hasWindowFunc = true; } // re-visit the first of ordercol list ordercol = reinterpret_cast(gi.groupByOrder); // for subquery, order+limit by will be supported in infinidb. build order by columns // @todo union order by and limit support //if (gwi.hasWindowFunc || gwi.subSelectType != CalpontSelectExecutionPlan::MAIN_SELECT) for (; ordercol; ordercol = ordercol->next) { ReturnedColumn* rc = NULL; if (ordercol->in_field_list && ordercol->counter_used) { rc = gwi.returnedCols[ordercol->counter - 1]->clone(); rc->orderPos(ordercol->counter - 1); // can not be optimized off if used in order by with counter. // set with self derived table alias if it's derived table gwi.returnedCols[ordercol->counter - 1]->incRefCount(); } else { Item* ord_item = *(ordercol->item); bool nonAggField = true; // ignore not_used column on order by. if (ord_item->type() == Item::INT_ITEM && ord_item->full_name() && string(ord_item->full_name()) == "Not_used") continue; else if (ord_item->type() == Item::INT_ITEM) rc = gwi.returnedCols[((Item_int*)ord_item)->val_int() - 1]->clone(); else if (ord_item->type() == Item::SUBSELECT_ITEM) gwi.fatalParseError = true; else if (ordercol->in_field_list && ord_item->type() == Item::FIELD_ITEM) { rc = buildReturnedColumn(ord_item, gwi, gwi.fatalParseError); Item_field* ifp = static_cast(ord_item); // The item must be an alias for a projected column // and extended SELECT list must contain a proper rc // either aggregation or a field. if (!rc && ifp->name.length) { gwi.fatalParseError = false; execplan::CalpontSelectExecutionPlan::ReturnedColumnList::iterator iter = gwi.returnedCols.begin(); for ( ; iter != gwi.returnedCols.end(); iter++ ) { if ( (*iter).get()->alias() == ord_item->name.str ) { rc = (*iter).get()->clone(); nonAggField = rc->hasAggregate() ? false : true; break; } } } } else rc = buildReturnedColumn(ord_item, gwi, gwi.fatalParseError); // Looking for a match for this item in GROUP BY list. if ( rc && ord_item->type() == Item::FIELD_ITEM && nonAggField) { execplan::CalpontSelectExecutionPlan::ReturnedColumnList::iterator iter = gwi.groupByCols.begin(); for ( ; iter != gwi.groupByCols.end(); iter++ ) { if ( rc->sameColumn((*iter).get()) ) break; } // MCOL-1052 Find and remove the optimized field // from ORDER using cond_pushed filters. if (buildConstColFromFilter( dynamic_cast(rc), gwi, gi)) { break; } // MCOL-1052 GROUP BY items list doesn't contain // this ORDER BY item. if ( iter == gwi.groupByCols.end() ) { Item_ident* iip = reinterpret_cast(ord_item); std::ostringstream ostream; ostream << "'"; if (iip->db_name) ostream << iip->db_name << '.'; else ostream << "unknown db" << '.'; if (iip->table_name) ostream << iip->table_name << '.'; else ostream << "unknown table" << '.'; if (iip->field_name.length) ostream << iip->field_name.str; else ostream << "unknown field"; ostream << "'"; Message::Args args; args.add(ostream.str()); string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NOT_GROUPBY_EXPRESSION, args); gwi.parseErrorText = emsg; setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); return ERR_NOT_GROUPBY_EXPRESSION; } } // @bug5501 try item_ptr if item can not be fixed. For some // weird dml statement state, item can not be fixed but the // infomation is available in item_ptr. if (!rc || gwi.fatalParseError) { gwi.fatalParseError = false; Item* item_ptr = ordercol->item_ptr; while (item_ptr->type() == Item::REF_ITEM) item_ptr = *(((Item_ref*)item_ptr)->ref); rc = buildReturnedColumn(item_ptr, gwi, gwi.fatalParseError); } // This ORDER BY item must be an agg function - // the ordercol->item_ptr and exteded SELECT list // must contain the corresponding item. if (!rc) { Item* item_ptr = ordercol->item_ptr; if (item_ptr) rc = buildReturnedColumn(item_ptr, gwi, gwi.fatalParseError); } if (!rc) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); gwi.parseErrorText = emsg; setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } } if (ordercol->direction == ORDER::ORDER_ASC) rc->asc(true); else rc->asc(false); gwi.orderByCols.push_back(SRCP(rc)); } // make sure columnmap, returnedcols and count(*) arg_list are not empty TableMap::iterator tb_iter = gwi.tableMap.begin(); try { for (; tb_iter != gwi.tableMap.end(); tb_iter++) { if ((*tb_iter).second.first == 1) continue; CalpontSystemCatalog::TableAliasName tan = (*tb_iter).first; CalpontSystemCatalog::TableName tn = make_table((*tb_iter).first.schema, (*tb_iter).first.table); SimpleColumn* sc = getSmallestColumn(csc, tn, tan, (*tb_iter).second.second->table, gwi); SRCP srcp(sc); gwi.columnMap.insert(CalpontSelectExecutionPlan::ColumnMap::value_type(sc->columnName(), srcp)); (*tb_iter).second.first = 1; } } catch (runtime_error& e) { setError(gwi.thd, ER_INTERNAL_ERROR, e.what(), gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } catch (...) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_LOST_CONN_EXEMGR); setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } if (!gwi.count_asterisk_list.empty() || !gwi.no_parm_func_list.empty() || gwi.returnedCols.empty()) { // get the smallest column from colmap CalpontSelectExecutionPlan::ColumnMap::const_iterator iter; int minColWidth = 0; CalpontSystemCatalog::ColType ct; try { for (iter = gwi.columnMap.begin(); iter != gwi.columnMap.end(); ++iter) { // should always not null SimpleColumn* sc = dynamic_cast(iter->second.get()); if (sc && !(sc->joinInfo() & JOIN_CORRELATED)) { ct = csc->colType(sc->oid()); if (minColWidth == 0) { minColWidth = ct.colWidth; minSc = iter->second; } else if (ct.colWidth < minColWidth) { minColWidth = ct.colWidth; minSc = iter->second; } } } } catch (...) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_LOST_CONN_EXEMGR); setError(gwi.thd, ER_INTERNAL_ERROR, emsg, gwi); CalpontSystemCatalog::removeCalpontSystemCatalog(sessionID); return ER_INTERNAL_ERROR; } if (gwi.returnedCols.empty() && gwi.additionalRetCols.empty()) gwi.returnedCols.push_back(minSc); } if (!isUnion && !gwi.hasWindowFunc && gwi.subSelectType == CalpontSelectExecutionPlan::MAIN_SELECT) { std::ostringstream vtb; vtb << "infinidb_vtable.$vtable_" << gwi.thd->thread_id; //vtb << "$vtable_" << gwi.thd->thread_id; // re-construct the select query and redo phase 1 if (redo) { // select now() from region case. returnedCols should have minSc. if (sel_cols_in_create.length() == 0) { SimpleColumn* sc = dynamic_cast(gwi.returnedCols[0].get()); if (sc) sel_cols_in_create = dynamic_cast(gwi.returnedCols[0].get())->columnName(); else sel_cols_in_create = gwi.returnedCols[0]->alias(); } // select * from derived table case if (gwi.selectCols.empty()) sel_cols_in_create = " * "; create_query = "create temporary table " + vtb.str() + " engine = aria as select " + sel_cols_in_create + " from "; TABLE_LIST* table_ptr = gi.groupByTables; bool firstTb = true; // put all tables, derived tables and views on the list //TABLE_LIST* table_ptr = select_lex.get_table_list(); set aliasSet; // to avoid duplicate table alias for (; table_ptr; table_ptr = table_ptr->next_local) { if (string(table_ptr->table_name.str).find("$vtable") != string::npos) continue; if (table_ptr->derived) { if (aliasSet.find(table_ptr->alias.str) != aliasSet.end()) continue; String str; (table_ptr->derived->first_select())->print(gwi.thd, &str, QT_INFINIDB_DERIVED); if (!firstTb) create_query += ", "; create_query += "(" + string(str.c_ptr()) + ") " + string(table_ptr->alias.str); firstTb = false; aliasSet.insert(table_ptr->alias.str); } else if (table_ptr->view) { if (aliasSet.find(table_ptr->alias.str) != aliasSet.end()) continue; if (!firstTb) create_query += ", "; create_query += string(table_ptr->db.str) + "." + string(table_ptr->table_name.str) + string(" `") + escapeBackTick(table_ptr->alias.str) + string("`"); aliasSet.insert(table_ptr->alias.str); firstTb = false; } else { // table referenced by view is represented by viewAlias_tableAlias. // consistent with item.cc field print. if (table_ptr->referencing_view) { if (aliasSet.find(string(table_ptr->referencing_view->alias.str) + "_" + string(table_ptr->alias.str)) != aliasSet.end()) continue; if (!firstTb) create_query += ", "; create_query += string(table_ptr->db.str) + "." + string(table_ptr->table_name.str) + string(" "); create_query += string(" `") + escapeBackTick(table_ptr->referencing_view->alias.str) + "_" + escapeBackTick(table_ptr->alias.str) + string("`"); aliasSet.insert(string(table_ptr->referencing_view->alias.str) + "_" + string(table_ptr->alias.str)); } else { if (aliasSet.find(table_ptr->alias.str) != aliasSet.end()) continue; if (!firstTb) create_query += ", "; create_query += string(table_ptr->db.str) + "." + string(table_ptr->table_name.str) + string(" "); create_query += string("`") + escapeBackTick(table_ptr->alias.str) + string("`"); aliasSet.insert(table_ptr->alias.str); } firstTb = false; } } gwi.thd->infinidb_vtable.create_vtable_query.free(); gwi.thd->infinidb_vtable.create_vtable_query.append(create_query.c_str(), create_query.length()); gwi.thd->infinidb_vtable.vtable_state = THD::INFINIDB_REDO_PHASE1; // redo phase 1 // turn off select distinct from post process unless there're post process functions // on the select list. string sel_query = "select "; if (gi.groupByDistinct && redo) sel_query = "select distinct "; else sel_query = "select "; // select * from derived table... if (gwi.selectCols.size() == 0) sel_query += " * "; for (uint32_t i = 0; i < gwi.selectCols.size(); i++) { sel_query += gwi.selectCols[i]; if ( i + 1 != gwi.selectCols.size()) sel_query += ", "; } //select_query.replace(lower_select_query.find("select *"), string("select *").length(), sel_query); } else { // remove order by clause in case this phase has been executed before. // need a better fix later, like skip all the other non-optimized phase. size_t pos = lower_select_query.find("order by"); if (pos != string::npos) select_query.replace(pos, lower_select_query.length() - pos, ""); //select_query = "select * from " + vtb.str(); // MCOL-1052 if (unionSel) { ordercol = reinterpret_cast(gi.groupByOrder); //order_list = gi.groupByOrder; } else ordercol = 0; ord_cols = ""; for (; ordercol; ordercol = ordercol->next) { Item* ord_item = *(ordercol->item); // @bug 1706. re-construct the order by item one by one, because the ord_cols constucted so far // is for REDO phase. if (ord_cols.length() != 0) ord_cols += ", "; if (ordercol->in_field_list && ordercol->counter_used) { ostringstream oss; oss << ordercol->counter; ord_cols += oss.str(); } else if (ord_item->type() == Item::NULL_ITEM) { // MCOL-793 Do nothing for an ORDER BY NULL } else if (ord_item->type() == Item::SUM_FUNC_ITEM) { Item_sum* ifp = (Item_sum*)(*(ordercol->item)); ReturnedColumn* fc = buildAggregateColumn(ifp, gwi); for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if (fc->operator==(gwi.returnedCols[i].get())) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); break; } } //continue; } // @bug 3518. if order by clause = selected column, use position. else if (ord_item->name.length && ord_item->type() == Item::FIELD_ITEM) { Item_field* field = reinterpret_cast(ord_item); string fullname; if (field->db_name) fullname += string(field->db_name) + "."; if (field->table_name) fullname += string(field->table_name) + "."; if (field->field_name.length) fullname += string(field->field_name.str); uint32_t i = 0; for (i = 0; i < gwi.returnedCols.size(); i++) { SimpleColumn* sc = dynamic_cast(gwi.returnedCols[i].get()); if (sc && ((Item_field*)ord_item)->cached_table && (strcasecmp(getViewName(((Item_field*)ord_item)->cached_table).c_str(), sc->viewName().c_str()) != 0)) continue; if (strcasecmp(fullname.c_str(), gwi.returnedCols[i]->alias().c_str()) == 0 || strcasecmp(ord_item->name.str, gwi.returnedCols[i]->alias().c_str()) == 0) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); break; } } if (i == gwi.returnedCols.size()) ord_cols += string(" `") + escapeBackTick(ord_item->name.str) + '`'; } else if (ord_item->name.length) { // for union order by 1 case. For unknown reason, it doesn't show in_field_list if (ord_item->type() == Item::INT_ITEM) { ord_cols += ord_item->name.str; } else if (ord_item->type() == Item::SUBSELECT_ITEM) { string emsg = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_ORDER_BY); setError(gwi.thd, ER_CHECK_NOT_IMPLEMENTED, emsg, gwi); return ER_CHECK_NOT_IMPLEMENTED; } else { ord_cols += string(" `") + escapeBackTick(ord_item->name.str) + '`'; } } else if (ord_item->type() == Item::FUNC_ITEM) { // @bug5636. check if this order by column is on the select list ReturnedColumn* rc = buildFunctionColumn((Item_func*)(ord_item), gwi, gwi.fatalParseError); for (uint32_t i = 0; i < gwi.returnedCols.size(); i++) { if (rc && rc->operator==(gwi.returnedCols[i].get())) { ostringstream oss; oss << i + 1; ord_cols += oss.str(); break; } } } else { String str; ord_item->print(&str, QT_INFINIDB_NO_QUOTE); ord_cols += string(str.c_ptr()); } if (ordercol->direction != ORDER::ORDER_ASC) ord_cols += " desc"; } } if ( gwi.orderByCols.size() ) // has order by { gwi.thd->infinidb_vtable.has_order_by = true; csep->hasOrderBy(true); csep->specHandlerProcessed(true); } } // LIMIT and OFFSET are extracted from TABLE_LIST elements. // All of JOIN-ed tables contain relevant limit and offset. uint64_t limit = (uint64_t)-1; if (gi.groupByTables->select_lex->select_limit && ( limit = static_cast(gi.groupByTables->select_lex->select_limit)->val_int() ) && limit != (uint64_t)-1 ) { csep->limitNum(limit); } else if (csep->hasOrderBy()) { // We use LimitedOrderBy so set the limit to // go through the check in addOrderByAndLimit csep->limitNum((uint64_t) - 2); } if (gi.groupByTables->select_lex->offset_limit) { csep->limitStart(((Item_int*)gi.groupByTables->select_lex->offset_limit)->val_int()); } //gwi.thd->infinidb_vtable.select_vtable_query.free(); //gwi.thd->infinidb_vtable.select_vtable_query.append(select_query.c_str(), select_query.length()); // We don't currently support limit with correlated subquery if (csep->limitNum() != (uint64_t) - 1 && gwi.subQuery && !gwi.correlatedTbNameVec.empty()) { gwi.fatalParseError = true; gwi.parseErrorText = IDBErrorInfo::instance()->errorMsg(ERR_NON_SUPPORT_LIMIT_SUB); setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } } // ORDER BY processing ends here if ( gi.groupByDistinct ) csep->distinct(true); // add the smallest column to count(*) parm. // select constant in subquery case std::vector::iterator coliter; if (!minSc) { if (!gwi.returnedCols.empty()) minSc = gwi.returnedCols[0]; else if (!gwi.additionalRetCols.empty()) minSc = gwi.additionalRetCols[0]; } // @bug3523, count(*) on subquery always pick column[0]. SimpleColumn* sc = dynamic_cast(minSc.get()); if (sc && sc->schemaName().empty()) { if (gwi.derivedTbList.size() >= 1) { SimpleColumn* sc1 = new SimpleColumn(); sc1->columnName(sc->columnName()); sc1->tableName(sc->tableName()); sc1->tableAlias(sc->tableAlias()); sc1->viewName(lower(sc->viewName())); sc1->colPosition(0); minSc.reset(sc1); } } for (coliter = gwi.count_asterisk_list.begin(); coliter != gwi.count_asterisk_list.end(); ++coliter) { // @bug5977 @note should never throw this, but checking just in case. // When ExeMgr fix is ready, this should not error out... if (dynamic_cast(minSc.get())) { gwi.fatalParseError = true; gwi.parseErrorText = "No project column found for aggregate function"; setError(gwi.thd, ER_INTERNAL_ERROR, gwi.parseErrorText, gwi); return ER_CHECK_NOT_IMPLEMENTED; } // Replace the last (presumably constant) object with minSc if ((*coliter)->aggParms().empty()) { (*coliter)->aggParms().push_back(minSc); } else { (*coliter)->aggParms()[0] = minSc; } } std::vector::iterator funciter; SPTP sptp(new ParseTree(minSc.get()->clone())); for (funciter = gwi.no_parm_func_list.begin(); funciter != gwi.no_parm_func_list.end(); ++funciter) { FunctionParm funcParms = (*funciter)->functionParms(); funcParms.push_back(sptp); (*funciter)->functionParms(funcParms); } // set sequence# for subquery localCols for (uint32_t i = 0; i < gwi.localCols.size(); i++) gwi.localCols[i]->sequence(i); // append additionalRetCols to returnedCols gwi.returnedCols.insert(gwi.returnedCols.begin(), gwi.additionalRetCols.begin(), gwi.additionalRetCols.end()); csep->groupByCols(gwi.groupByCols); csep->orderByCols(gwi.orderByCols); csep->returnedCols(gwi.returnedCols); csep->columnMap(gwi.columnMap); csep->having(havingFilter); csep->derivedTableList(gwi.derivedTbList); csep->selectSubList(selectSubList); csep->subSelectList(gwi.subselectList); gwi.thd->infinidb_vtable.duplicate_field_name = false; clearStacks(gwi); return 0; } } // vim:ts=4 sw=4: