Simplistic, experimental framework for Stored Procedures (SPs).
Implements creation and dropping of PROCEDUREs, IN, OUT, and INOUT parameters,
single-statement procedures, rudimentary multi-statement (begin-end) prodedures
(when the client can handle it), and local variables.
Missing most of the embedded SQL language, all attributes, FUNCTIONs, error handling,
reparses procedures at each call (no caching), etc, etc.
Certainly buggy too, but procedures can actually be created and called....
2002-12-08 19:59:22 +01:00
|
|
|
/* Copyright (C) 2002 MySQL AB
|
|
|
|
|
|
|
|
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; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
#pragma implementation
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(WIN32) || defined(__WIN__)
|
|
|
|
#undef SAFEMALLOC /* Problems with threads */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "mysql_priv.h"
|
|
|
|
#include "sp_pcontext.h"
|
2002-12-11 14:24:29 +01:00
|
|
|
#include "sp_head.h"
|
Simplistic, experimental framework for Stored Procedures (SPs).
Implements creation and dropping of PROCEDUREs, IN, OUT, and INOUT parameters,
single-statement procedures, rudimentary multi-statement (begin-end) prodedures
(when the client can handle it), and local variables.
Missing most of the embedded SQL language, all attributes, FUNCTIONs, error handling,
reparses procedures at each call (no caching), etc, etc.
Certainly buggy too, but procedures can actually be created and called....
2002-12-08 19:59:22 +01:00
|
|
|
|
|
|
|
sp_pcontext::sp_pcontext()
|
2002-12-11 14:24:29 +01:00
|
|
|
: m_params(0), m_framesize(0), m_i(0), m_genlab(0)
|
Simplistic, experimental framework for Stored Procedures (SPs).
Implements creation and dropping of PROCEDUREs, IN, OUT, and INOUT parameters,
single-statement procedures, rudimentary multi-statement (begin-end) prodedures
(when the client can handle it), and local variables.
Missing most of the embedded SQL language, all attributes, FUNCTIONs, error handling,
reparses procedures at each call (no caching), etc, etc.
Certainly buggy too, but procedures can actually be created and called....
2002-12-08 19:59:22 +01:00
|
|
|
{
|
|
|
|
m_pvar_size = 16;
|
|
|
|
m_pvar = (sp_pvar_t *)my_malloc(m_pvar_size * sizeof(sp_pvar_t), MYF(MY_WME));
|
|
|
|
if (m_pvar)
|
|
|
|
memset(m_pvar, 0, m_pvar_size * sizeof(sp_pvar_t));
|
2002-12-11 14:24:29 +01:00
|
|
|
m_label.empty();
|
Simplistic, experimental framework for Stored Procedures (SPs).
Implements creation and dropping of PROCEDUREs, IN, OUT, and INOUT parameters,
single-statement procedures, rudimentary multi-statement (begin-end) prodedures
(when the client can handle it), and local variables.
Missing most of the embedded SQL language, all attributes, FUNCTIONs, error handling,
reparses procedures at each call (no caching), etc, etc.
Certainly buggy too, but procedures can actually be created and called....
2002-12-08 19:59:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sp_pcontext::grow()
|
|
|
|
{
|
|
|
|
uint sz = m_pvar_size + 8;
|
|
|
|
sp_pvar_t *a = (sp_pvar_t *)my_realloc((char *)m_pvar,
|
|
|
|
sz * sizeof(sp_pvar_t),
|
|
|
|
MYF(MY_WME | MY_ALLOW_ZERO_PTR));
|
|
|
|
|
|
|
|
if (a)
|
|
|
|
{
|
|
|
|
m_pvar_size = sz;
|
|
|
|
m_pvar = a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This does a linear search (from newer to older variables, in case
|
|
|
|
** we have shadowed names).
|
|
|
|
** It's possible to have a more efficient allocation and search method,
|
|
|
|
** but it might not be worth it. The typical number of parameters and
|
|
|
|
** variables will in most cases be low (a handfull).
|
|
|
|
** And this is only called during parsing.
|
|
|
|
*/
|
|
|
|
sp_pvar_t *
|
|
|
|
sp_pcontext::find_pvar(LEX_STRING *name)
|
|
|
|
{
|
|
|
|
String n(name->str, name->length, default_charset_info);
|
|
|
|
uint i = m_i;
|
|
|
|
|
|
|
|
while (i-- > 0)
|
|
|
|
{
|
|
|
|
if (stringcmp(&n, m_pvar[i].name->const_string()) == 0)
|
|
|
|
return m_pvar + i;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
sp_pcontext::push(LEX_STRING *name, enum enum_field_types type,
|
|
|
|
sp_param_mode_t mode)
|
|
|
|
{
|
|
|
|
if (m_i >= m_pvar_size)
|
|
|
|
grow();
|
|
|
|
if (m_i < m_pvar_size)
|
|
|
|
{
|
|
|
|
if (m_i == m_framesize)
|
|
|
|
m_framesize += 1;
|
|
|
|
m_pvar[m_i].name= new Item_string(name->str, name->length,
|
|
|
|
default_charset_info);
|
|
|
|
m_pvar[m_i].type= type;
|
|
|
|
m_pvar[m_i].mode= mode;
|
|
|
|
m_pvar[m_i].offset= m_i;
|
|
|
|
m_pvar[m_i].isset= (mode == sp_param_out ? FALSE : TRUE);
|
|
|
|
m_i += 1;
|
|
|
|
}
|
|
|
|
}
|
2002-12-11 14:24:29 +01:00
|
|
|
|
2002-12-16 15:40:44 +01:00
|
|
|
sp_label_t *
|
2002-12-11 14:24:29 +01:00
|
|
|
sp_pcontext::push_label(char *name, uint ip)
|
|
|
|
{
|
|
|
|
sp_label_t *lab = (sp_label_t *)my_malloc(sizeof(sp_label_t), MYF(MY_WME));
|
|
|
|
|
|
|
|
if (lab)
|
|
|
|
{
|
|
|
|
lab->name= name;
|
|
|
|
lab->ip= ip;
|
|
|
|
m_label.push_front(lab);
|
|
|
|
}
|
2002-12-16 15:40:44 +01:00
|
|
|
return lab;
|
2002-12-11 14:24:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
sp_label_t *
|
|
|
|
sp_pcontext::find_label(char *name)
|
|
|
|
{
|
|
|
|
List_iterator_fast<sp_label_t> li(m_label);
|
|
|
|
sp_label_t *lab;
|
|
|
|
|
|
|
|
while ((lab= li++))
|
|
|
|
if (strcasecmp(name, lab->name) == 0)
|
|
|
|
return lab;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|