mirror of
https://github.com/MariaDB/server.git
synced 2025-01-16 03:52:35 +01:00
70823e1d91
The reason for the failure was a bug in an include file on debian that causes 'struct stat' to have different sized depending on the environment. This patch fixes so that we always include my_global.h or my_config.h before we include any other files. Other things: - Removed #include <my_global.h> in some include files; Better to always do this at the top level to have as few "always-include-this-file-first' files as possible. - Removed usage of some include files that where already included by my_global.h or by other files. client/mysql_plugin.c: Use my_global.h first client/mysqlslap.c: Remove duplicated include files extra/comp_err.c: Remove duplicated include files include/m_string.h: Remove duplicated include files include/maria.h: Remove duplicated include files libmysqld/emb_qcache.cc: Use my_global.h first plugin/semisync/semisync.h: Use my_pthread.h first sql/datadict.cc: Use my_global.h first sql/debug_sync.cc: Use my_global.h first sql/derror.cc: Use my_global.h first sql/des_key_file.cc: Use my_global.h first sql/discover.cc: Use my_global.h first sql/event_data_objects.cc: Use my_global.h first sql/event_db_repository.cc: Use my_global.h first sql/event_parse_data.cc: Use my_global.h first sql/event_queue.cc: Use my_global.h first sql/event_scheduler.cc: Use my_global.h first sql/events.cc: Use my_global.h first sql/field.cc: Use my_global.h first Remove duplicated include files sql/field_conv.cc: Use my_global.h first sql/filesort.cc: Use my_global.h first Remove duplicated include files sql/gstream.cc: Use my_global.h first sql/ha_ndbcluster.cc: Use my_global.h first sql/ha_ndbcluster_binlog.cc: Use my_global.h first sql/ha_ndbcluster_cond.cc: Use my_global.h first sql/ha_partition.cc: Use my_global.h first sql/handler.cc: Use my_global.h first sql/hash_filo.cc: Use my_global.h first sql/hostname.cc: Use my_global.h first sql/init.cc: Use my_global.h first sql/item.cc: Use my_global.h first sql/item_buff.cc: Use my_global.h first sql/item_cmpfunc.cc: Use my_global.h first sql/item_create.cc: Use my_global.h first sql/item_geofunc.cc: Use my_global.h first sql/item_inetfunc.cc: Use my_global.h first sql/item_row.cc: Use my_global.h first sql/item_strfunc.cc: Use my_global.h first sql/item_subselect.cc: Use my_global.h first sql/item_sum.cc: Use my_global.h first sql/item_timefunc.cc: Use my_global.h first sql/item_xmlfunc.cc: Use my_global.h first sql/key.cc: Use my_global.h first sql/lock.cc: Use my_global.h first sql/log.cc: Use my_global.h first sql/log_event.cc: Use my_global.h first sql/log_event_old.cc: Use my_global.h first sql/mf_iocache.cc: Use my_global.h first sql/mysql_install_db.cc: Remove duplicated include files sql/mysqld.cc: Remove duplicated include files sql/net_serv.cc: Remove duplicated include files sql/opt_range.cc: Use my_global.h first sql/opt_subselect.cc: Use my_global.h first sql/opt_sum.cc: Use my_global.h first sql/parse_file.cc: Use my_global.h first sql/partition_info.cc: Use my_global.h first sql/procedure.cc: Use my_global.h first sql/protocol.cc: Use my_global.h first sql/records.cc: Use my_global.h first sql/records.h: Don't include my_global.h Better to do this at the upper level sql/repl_failsafe.cc: Use my_global.h first sql/rpl_filter.cc: Use my_global.h first sql/rpl_gtid.cc: Use my_global.h first sql/rpl_handler.cc: Use my_global.h first sql/rpl_injector.cc: Use my_global.h first sql/rpl_record.cc: Use my_global.h first sql/rpl_record_old.cc: Use my_global.h first sql/rpl_reporting.cc: Use my_global.h first sql/rpl_rli.cc: Use my_global.h first sql/rpl_tblmap.cc: Use my_global.h first sql/rpl_utility.cc: Use my_global.h first sql/set_var.cc: Added comment sql/slave.cc: Use my_global.h first sql/sp.cc: Use my_global.h first sql/sp_cache.cc: Use my_global.h first sql/sp_head.cc: Use my_global.h first sql/sp_pcontext.cc: Use my_global.h first sql/sp_rcontext.cc: Use my_global.h first sql/spatial.cc: Use my_global.h first sql/sql_acl.cc: Use my_global.h first sql/sql_admin.cc: Use my_global.h first sql/sql_analyse.cc: Use my_global.h first sql/sql_audit.cc: Use my_global.h first sql/sql_base.cc: Use my_global.h first sql/sql_binlog.cc: Use my_global.h first sql/sql_bootstrap.cc: Use my_global.h first Use my_global.h first sql/sql_cache.cc: Use my_global.h first sql/sql_class.cc: Use my_global.h first sql/sql_client.cc: Use my_global.h first sql/sql_connect.cc: Use my_global.h first sql/sql_crypt.cc: Use my_global.h first sql/sql_cursor.cc: Use my_global.h first sql/sql_db.cc: Use my_global.h first sql/sql_delete.cc: Use my_global.h first sql/sql_derived.cc: Use my_global.h first sql/sql_do.cc: Use my_global.h first sql/sql_error.cc: Use my_global.h first sql/sql_explain.cc: Use my_global.h first sql/sql_expression_cache.cc: Use my_global.h first sql/sql_handler.cc: Use my_global.h first sql/sql_help.cc: Use my_global.h first sql/sql_insert.cc: Use my_global.h first sql/sql_lex.cc: Use my_global.h first sql/sql_load.cc: Use my_global.h first sql/sql_locale.cc: Use my_global.h first sql/sql_manager.cc: Use my_global.h first sql/sql_parse.cc: Use my_global.h first sql/sql_partition.cc: Use my_global.h first sql/sql_plugin.cc: Added comment sql/sql_prepare.cc: Use my_global.h first sql/sql_priv.h: Added error if we use this before including my_global.h This check is here becasue so many files includes sql_priv.h first. sql/sql_profile.cc: Use my_global.h first sql/sql_reload.cc: Use my_global.h first sql/sql_rename.cc: Use my_global.h first sql/sql_repl.cc: Use my_global.h first sql/sql_select.cc: Use my_global.h first sql/sql_servers.cc: Use my_global.h first sql/sql_show.cc: Added comment sql/sql_signal.cc: Use my_global.h first sql/sql_statistics.cc: Use my_global.h first sql/sql_table.cc: Use my_global.h first sql/sql_tablespace.cc: Use my_global.h first sql/sql_test.cc: Use my_global.h first sql/sql_time.cc: Use my_global.h first sql/sql_trigger.cc: Use my_global.h first sql/sql_udf.cc: Use my_global.h first sql/sql_union.cc: Use my_global.h first sql/sql_update.cc: Use my_global.h first sql/sql_view.cc: Use my_global.h first sql/sys_vars.cc: Added comment sql/table.cc: Use my_global.h first sql/thr_malloc.cc: Use my_global.h first sql/transaction.cc: Use my_global.h first sql/uniques.cc: Use my_global.h first sql/unireg.cc: Use my_global.h first sql/unireg.h: Removed inclusion of my_global.h storage/archive/ha_archive.cc: Added comment storage/blackhole/ha_blackhole.cc: Use my_global.h first storage/csv/ha_tina.cc: Use my_global.h first storage/csv/transparent_file.cc: Use my_global.h first storage/federated/ha_federated.cc: Use my_global.h first storage/federatedx/federatedx_io.cc: Use my_global.h first storage/federatedx/federatedx_io_mysql.cc: Use my_global.h first storage/federatedx/federatedx_io_null.cc: Use my_global.h first storage/federatedx/federatedx_txn.cc: Use my_global.h first storage/heap/ha_heap.cc: Use my_global.h first storage/innobase/handler/handler0alter.cc: Use my_global.h first storage/maria/ha_maria.cc: Use my_global.h first storage/maria/unittest/ma_maria_log_cleanup.c: Remove duplicated include files storage/maria/unittest/test_file.c: Added comment storage/myisam/ha_myisam.cc: Move sql_plugin.h first as this includes my_global.h storage/myisammrg/ha_myisammrg.cc: Use my_global.h first storage/oqgraph/oqgraph_thunk.cc: Use my_config.h and my_global.h first One could not include my_global.h before oqgraph_thunk.h (don't know why) storage/spider/ha_spider.cc: Use my_global.h first storage/spider/hs_client/config.cpp: Use my_global.h first storage/spider/hs_client/escape.cpp: Use my_global.h first storage/spider/hs_client/fatal.cpp: Use my_global.h first storage/spider/hs_client/hstcpcli.cpp: Use my_global.h first storage/spider/hs_client/socket.cpp: Use my_global.h first storage/spider/hs_client/string_util.cpp: Use my_global.h first storage/spider/spd_conn.cc: Use my_global.h first storage/spider/spd_copy_tables.cc: Use my_global.h first storage/spider/spd_db_conn.cc: Use my_global.h first storage/spider/spd_db_handlersocket.cc: Use my_global.h first storage/spider/spd_db_mysql.cc: Use my_global.h first storage/spider/spd_db_oracle.cc: Use my_global.h first storage/spider/spd_direct_sql.cc: Use my_global.h first storage/spider/spd_i_s.cc: Use my_global.h first storage/spider/spd_malloc.cc: Use my_global.h first storage/spider/spd_param.cc: Use my_global.h first storage/spider/spd_ping_table.cc: Use my_global.h first storage/spider/spd_sys_table.cc: Use my_global.h first storage/spider/spd_table.cc: Use my_global.h first storage/spider/spd_trx.cc: Use my_global.h first storage/xtradb/handler/handler0alter.cc: Use my_global.h first storage/xtradb/handler/i_s.cc: Use my_global.h first
485 lines
11 KiB
C++
485 lines
11 KiB
C++
/* Copyright (c) 2002, 2010, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
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 St, Fifth Floor, Boston, MA 02110-1301 USA */
|
|
|
|
#include <my_global.h>
|
|
#include "sql_priv.h"
|
|
#include "unireg.h"
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
|
#pragma implementation
|
|
#endif
|
|
|
|
#include "sp_pcontext.h"
|
|
#include "sp_head.h"
|
|
|
|
bool sp_condition_value::equals(const sp_condition_value *cv) const
|
|
{
|
|
DBUG_ASSERT(cv);
|
|
|
|
if (this == cv)
|
|
return true;
|
|
|
|
if (type != cv->type)
|
|
return false;
|
|
|
|
switch (type)
|
|
{
|
|
case sp_condition_value::ERROR_CODE:
|
|
return (mysqlerr == cv->mysqlerr);
|
|
|
|
case sp_condition_value::SQLSTATE:
|
|
return (strcmp(sql_state, cv->sql_state) == 0);
|
|
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
void sp_pcontext::init(uint var_offset,
|
|
uint cursor_offset,
|
|
int num_case_expressions)
|
|
{
|
|
m_var_offset= var_offset;
|
|
m_cursor_offset= cursor_offset;
|
|
m_num_case_exprs= num_case_expressions;
|
|
|
|
m_labels.empty();
|
|
}
|
|
|
|
|
|
sp_pcontext::sp_pcontext()
|
|
: Sql_alloc(),
|
|
m_max_var_index(0), m_max_cursor_index(0),
|
|
m_parent(NULL), m_pboundary(0),
|
|
m_scope(REGULAR_SCOPE)
|
|
{
|
|
init(0, 0, 0);
|
|
}
|
|
|
|
|
|
sp_pcontext::sp_pcontext(sp_pcontext *prev, sp_pcontext::enum_scope scope)
|
|
: Sql_alloc(),
|
|
m_max_var_index(0), m_max_cursor_index(0),
|
|
m_parent(prev), m_pboundary(0),
|
|
m_scope(scope)
|
|
{
|
|
init(prev->m_var_offset + prev->m_max_var_index,
|
|
prev->current_cursor_count(),
|
|
prev->get_num_case_exprs());
|
|
}
|
|
|
|
|
|
sp_pcontext::~sp_pcontext()
|
|
{
|
|
for (size_t i= 0; i < m_children.elements(); ++i)
|
|
delete m_children.at(i);
|
|
}
|
|
|
|
|
|
sp_pcontext *sp_pcontext::push_context(THD *thd, sp_pcontext::enum_scope scope)
|
|
{
|
|
sp_pcontext *child= new (thd->mem_root) sp_pcontext(this, scope);
|
|
|
|
if (child)
|
|
m_children.append(child);
|
|
return child;
|
|
}
|
|
|
|
|
|
sp_pcontext *sp_pcontext::pop_context()
|
|
{
|
|
m_parent->m_max_var_index+= m_max_var_index;
|
|
|
|
uint submax= max_cursor_index();
|
|
if (submax > m_parent->m_max_cursor_index)
|
|
m_parent->m_max_cursor_index= submax;
|
|
|
|
if (m_num_case_exprs > m_parent->m_num_case_exprs)
|
|
m_parent->m_num_case_exprs= m_num_case_exprs;
|
|
|
|
return m_parent;
|
|
}
|
|
|
|
|
|
uint sp_pcontext::diff_handlers(const sp_pcontext *ctx, bool exclusive) const
|
|
{
|
|
uint n= 0;
|
|
const sp_pcontext *pctx= this;
|
|
const sp_pcontext *last_ctx= NULL;
|
|
|
|
while (pctx && pctx != ctx)
|
|
{
|
|
n+= pctx->m_handlers.elements();
|
|
last_ctx= pctx;
|
|
pctx= pctx->parent_context();
|
|
}
|
|
if (pctx)
|
|
return (exclusive && last_ctx ? n - last_ctx->m_handlers.elements() : n);
|
|
return 0; // Didn't find ctx
|
|
}
|
|
|
|
|
|
uint sp_pcontext::diff_cursors(const sp_pcontext *ctx, bool exclusive) const
|
|
{
|
|
uint n= 0;
|
|
const sp_pcontext *pctx= this;
|
|
const sp_pcontext *last_ctx= NULL;
|
|
|
|
while (pctx && pctx != ctx)
|
|
{
|
|
n+= pctx->m_cursors.elements();
|
|
last_ctx= pctx;
|
|
pctx= pctx->parent_context();
|
|
}
|
|
if (pctx)
|
|
return (exclusive && last_ctx ? n - last_ctx->m_cursors.elements() : n);
|
|
return 0; // Didn't find ctx
|
|
}
|
|
|
|
|
|
sp_variable *sp_pcontext::find_variable(LEX_STRING name,
|
|
bool current_scope_only) const
|
|
{
|
|
uint i= m_vars.elements() - m_pboundary;
|
|
|
|
while (i--)
|
|
{
|
|
sp_variable *p= m_vars.at(i);
|
|
|
|
if (my_strnncoll(system_charset_info,
|
|
(const uchar *)name.str, name.length,
|
|
(const uchar *)p->name.str, p->name.length) == 0)
|
|
{
|
|
return p;
|
|
}
|
|
}
|
|
|
|
return (!current_scope_only && m_parent) ?
|
|
m_parent->find_variable(name, false) :
|
|
NULL;
|
|
}
|
|
|
|
|
|
sp_variable *sp_pcontext::find_variable(uint offset) const
|
|
{
|
|
if (m_var_offset <= offset && offset < m_var_offset + m_vars.elements())
|
|
return m_vars.at(offset - m_var_offset); // This frame
|
|
|
|
return m_parent ?
|
|
m_parent->find_variable(offset) : // Some previous frame
|
|
NULL; // Index out of bounds
|
|
}
|
|
|
|
|
|
sp_variable *sp_pcontext::add_variable(THD *thd,
|
|
LEX_STRING name,
|
|
enum enum_field_types type,
|
|
sp_variable::enum_mode mode)
|
|
{
|
|
sp_variable *p=
|
|
new (thd->mem_root) sp_variable(name, type,mode, current_var_count());
|
|
|
|
if (!p)
|
|
return NULL;
|
|
|
|
++m_max_var_index;
|
|
|
|
return m_vars.append(p) ? NULL : p;
|
|
}
|
|
|
|
|
|
sp_label *sp_pcontext::push_label(THD *thd, LEX_STRING name, uint ip)
|
|
{
|
|
sp_label *label=
|
|
new (thd->mem_root) sp_label(name, ip, sp_label::IMPLICIT, this);
|
|
|
|
if (!label)
|
|
return NULL;
|
|
|
|
m_labels.push_front(label);
|
|
|
|
return label;
|
|
}
|
|
|
|
|
|
sp_label *sp_pcontext::find_label(LEX_STRING name)
|
|
{
|
|
List_iterator_fast<sp_label> li(m_labels);
|
|
sp_label *lab;
|
|
|
|
while ((lab= li++))
|
|
{
|
|
if (my_strcasecmp(system_charset_info, name.str, lab->name.str) == 0)
|
|
return lab;
|
|
}
|
|
|
|
/*
|
|
Note about exception handlers.
|
|
See SQL:2003 SQL/PSM (ISO/IEC 9075-4:2003),
|
|
section 13.1 <compound statement>,
|
|
syntax rule 4.
|
|
In short, a DECLARE HANDLER block can not refer
|
|
to labels from the parent context, as they are out of scope.
|
|
*/
|
|
return (m_parent && (m_scope == REGULAR_SCOPE)) ?
|
|
m_parent->find_label(name) :
|
|
NULL;
|
|
}
|
|
|
|
|
|
bool sp_pcontext::add_condition(THD *thd,
|
|
LEX_STRING name,
|
|
sp_condition_value *value)
|
|
{
|
|
sp_condition *p= new (thd->mem_root) sp_condition(name, value);
|
|
|
|
if (p == NULL)
|
|
return true;
|
|
|
|
return m_conditions.append(p);
|
|
}
|
|
|
|
|
|
sp_condition_value *sp_pcontext::find_condition(LEX_STRING name,
|
|
bool current_scope_only) const
|
|
{
|
|
uint i= m_conditions.elements();
|
|
|
|
while (i--)
|
|
{
|
|
sp_condition *p= m_conditions.at(i);
|
|
|
|
if (my_strnncoll(system_charset_info,
|
|
(const uchar *) name.str, name.length,
|
|
(const uchar *) p->name.str, p->name.length) == 0)
|
|
{
|
|
return p->value;
|
|
}
|
|
}
|
|
|
|
return (!current_scope_only && m_parent) ?
|
|
m_parent->find_condition(name, false) :
|
|
NULL;
|
|
}
|
|
|
|
|
|
sp_handler *sp_pcontext::add_handler(THD *thd,
|
|
sp_handler::enum_type type)
|
|
{
|
|
sp_handler *h= new (thd->mem_root) sp_handler(type);
|
|
|
|
if (!h)
|
|
return NULL;
|
|
|
|
return m_handlers.append(h) ? NULL : h;
|
|
}
|
|
|
|
|
|
bool sp_pcontext::check_duplicate_handler(
|
|
const sp_condition_value *cond_value) const
|
|
{
|
|
for (size_t i= 0; i < m_handlers.elements(); ++i)
|
|
{
|
|
sp_handler *h= m_handlers.at(i);
|
|
|
|
List_iterator_fast<sp_condition_value> li(h->condition_values);
|
|
sp_condition_value *cv;
|
|
|
|
while ((cv= li++))
|
|
{
|
|
if (cond_value->equals(cv))
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
sp_handler*
|
|
sp_pcontext::find_handler(const char *sql_state,
|
|
uint sql_errno,
|
|
Sql_condition::enum_warning_level level) const
|
|
{
|
|
sp_handler *found_handler= NULL;
|
|
sp_condition_value *found_cv= NULL;
|
|
|
|
for (size_t i= 0; i < m_handlers.elements(); ++i)
|
|
{
|
|
sp_handler *h= m_handlers.at(i);
|
|
|
|
List_iterator_fast<sp_condition_value> li(h->condition_values);
|
|
sp_condition_value *cv;
|
|
|
|
while ((cv= li++))
|
|
{
|
|
switch (cv->type)
|
|
{
|
|
case sp_condition_value::ERROR_CODE:
|
|
if (sql_errno == cv->mysqlerr &&
|
|
(!found_cv ||
|
|
found_cv->type > sp_condition_value::ERROR_CODE))
|
|
{
|
|
found_cv= cv;
|
|
found_handler= h;
|
|
}
|
|
break;
|
|
|
|
case sp_condition_value::SQLSTATE:
|
|
if (strcmp(sql_state, cv->sql_state) == 0 &&
|
|
(!found_cv ||
|
|
found_cv->type > sp_condition_value::SQLSTATE))
|
|
{
|
|
found_cv= cv;
|
|
found_handler= h;
|
|
}
|
|
break;
|
|
|
|
case sp_condition_value::WARNING:
|
|
if ((is_sqlstate_warning(sql_state) ||
|
|
level == Sql_condition::WARN_LEVEL_WARN) && !found_cv)
|
|
{
|
|
found_cv= cv;
|
|
found_handler= h;
|
|
}
|
|
break;
|
|
|
|
case sp_condition_value::NOT_FOUND:
|
|
if (is_sqlstate_not_found(sql_state) && !found_cv)
|
|
{
|
|
found_cv= cv;
|
|
found_handler= h;
|
|
}
|
|
break;
|
|
|
|
case sp_condition_value::EXCEPTION:
|
|
if (is_sqlstate_exception(sql_state) &&
|
|
level == Sql_condition::WARN_LEVEL_ERROR && !found_cv)
|
|
{
|
|
found_cv= cv;
|
|
found_handler= h;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (found_handler)
|
|
return found_handler;
|
|
|
|
|
|
// There is no appropriate handler in this parsing context. We need to look up
|
|
// in parent contexts. There might be two cases here:
|
|
//
|
|
// 1. The current context has REGULAR_SCOPE. That means, it's a simple
|
|
// BEGIN..END block:
|
|
// ...
|
|
// BEGIN
|
|
// ... # We're here.
|
|
// END
|
|
// ...
|
|
// In this case we simply call find_handler() on parent's context recursively.
|
|
//
|
|
// 2. The current context has HANDLER_SCOPE. That means, we're inside an
|
|
// SQL-handler block:
|
|
// ...
|
|
// DECLARE ... HANDLER FOR ...
|
|
// BEGIN
|
|
// ... # We're here.
|
|
// END
|
|
// ...
|
|
// In this case we can not just call parent's find_handler(), because
|
|
// parent's handler don't catch conditions from this scope. Instead, we should
|
|
// try to find first parent context (we might have nested handler
|
|
// declarations), which has REGULAR_SCOPE (i.e. which is regular BEGIN..END
|
|
// block).
|
|
|
|
const sp_pcontext *p= this;
|
|
|
|
while (p && p->m_scope == HANDLER_SCOPE)
|
|
p= p->m_parent;
|
|
|
|
if (!p || !p->m_parent)
|
|
return NULL;
|
|
|
|
return p->m_parent->find_handler(sql_state, sql_errno, level);
|
|
}
|
|
|
|
|
|
bool sp_pcontext::add_cursor(LEX_STRING name)
|
|
{
|
|
if (m_cursors.elements() == m_max_cursor_index)
|
|
++m_max_cursor_index;
|
|
|
|
return m_cursors.append(name);
|
|
}
|
|
|
|
|
|
bool sp_pcontext::find_cursor(LEX_STRING name,
|
|
uint *poff,
|
|
bool current_scope_only) const
|
|
{
|
|
uint i= m_cursors.elements();
|
|
|
|
while (i--)
|
|
{
|
|
LEX_STRING n= m_cursors.at(i);
|
|
|
|
if (my_strnncoll(system_charset_info,
|
|
(const uchar *) name.str, name.length,
|
|
(const uchar *) n.str, n.length) == 0)
|
|
{
|
|
*poff= m_cursor_offset + i;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return (!current_scope_only && m_parent) ?
|
|
m_parent->find_cursor(name, poff, false) :
|
|
false;
|
|
}
|
|
|
|
|
|
void sp_pcontext::retrieve_field_definitions(
|
|
List<Create_field> *field_def_lst) const
|
|
{
|
|
/* Put local/context fields in the result list. */
|
|
|
|
for (size_t i= 0; i < m_vars.elements(); ++i)
|
|
{
|
|
sp_variable *var_def= m_vars.at(i);
|
|
|
|
field_def_lst->push_back(&var_def->field_def);
|
|
}
|
|
|
|
/* Put the fields of the enclosed contexts in the result list. */
|
|
|
|
for (size_t i= 0; i < m_children.elements(); ++i)
|
|
m_children.at(i)->retrieve_field_definitions(field_def_lst);
|
|
}
|
|
|
|
|
|
const LEX_STRING *sp_pcontext::find_cursor(uint offset) const
|
|
{
|
|
if (m_cursor_offset <= offset &&
|
|
offset < m_cursor_offset + m_cursors.elements())
|
|
{
|
|
return &m_cursors.at(offset - m_cursor_offset); // This frame
|
|
}
|
|
|
|
return m_parent ?
|
|
m_parent->find_cursor(offset) : // Some previous frame
|
|
NULL; // Index out of bounds
|
|
}
|