mirror of
https://github.com/MariaDB/server.git
synced 2025-01-21 22:34:18 +01:00
5954e94fa7
Procedure names were unintentionally case-sensitive when read from the database (but case-insensitive when fetched from the cache). Note that the DB-part of qualified names is still case-sensitive (for consistency with other usage in mysql).
1100 lines
No EOL
41 KiB
Text
1100 lines
No EOL
41 KiB
Text
|
|
Implementation specification for Stored Procedures
|
|
==================================================
|
|
|
|
|
|
- How parsing and execution of queries work
|
|
|
|
In order to execute a query, the function sql_parse.cc:mysql_parse() is
|
|
called, which in turn calls the parser (yyparse()) with an updated Lex
|
|
structure as the result. mysql_parse() then calls mysql_execute_command()
|
|
which dispatches on the command code (in Lex) to the corresponding code for
|
|
executing that particular query.
|
|
|
|
There are three structures involved in the execution of a query which are of
|
|
interest to the stored procedure implementation:
|
|
|
|
- Lex (mentioned above) is the "compiled" query, that is the output from
|
|
the parser and what is then interpreted to do the actual work.
|
|
It constains an enum value (sql_command) which is the query type, and
|
|
all the data collected by the parser needed for the execution (table
|
|
names, fields, values, etc).
|
|
- THD is the "run-time" state of a connection, containing all that is
|
|
needed for a particular client connection, and, among other things, the
|
|
Lex structure currently being executed.
|
|
- Item_*: During parsing, all data is translated into "items", objects of
|
|
the subclasses of "Item", such as Item_int, Item_real, Item_string, etc,
|
|
for basic datatypes, and also various more specialized Item types for
|
|
expressions to be evaluated (Item_func objects).
|
|
|
|
|
|
- How to fit Stored Procedure into this scheme
|
|
|
|
- An overview of the classes and files for stored procedures
|
|
(More detailed APIs at the end of this file)
|
|
|
|
- class sp_head (sp_head.{cc,h})
|
|
This contains, among other things, an array of "instructions" and the
|
|
method for executing the procedure.
|
|
|
|
- class sp_pcontext (sp_pcontext.{cc,h}
|
|
This is the parse context for the procedure. It's primarily used during
|
|
parsing to keep track of local parameters, variables and labels, but
|
|
it's also used at CALL time do find parameters mode (IN, OUT or INOUT)
|
|
and type when setting up the runtime context.
|
|
|
|
- class sp_instr (sp_head.{cc,h})
|
|
This is the base class for "instructions", that is, what is generated
|
|
by the parser. It turns out that we only need a minimum of 5 different
|
|
sub classes:
|
|
- sp_instr_stmt
|
|
Execute a statement. This is the "call-out" any normal SQL statement,
|
|
like a SELECT, INSERT etc. It contains the Lex structure for the
|
|
statement in question.
|
|
- sp_instr_set
|
|
Set the value of a local variable (or parameter)
|
|
- sp_instr_jump
|
|
An unconditional jump.
|
|
- sp_instr_jump_if_not
|
|
Jump if condition is not true. It turns out that the negative test is
|
|
most convenient when generating the code for the flow control
|
|
constructs.
|
|
- sp_instr_freturn
|
|
Return a value from a FUNCTION and exit.
|
|
For condition HANDLERs some special instructions are also needed, see
|
|
that section below.
|
|
|
|
- class sp_rcontext (sp_rcontext.h)
|
|
This is the runtime context in the THD structure.
|
|
It contains an array of items, the parameters and local variables for
|
|
the currently executing stored procedure.
|
|
This means that variable value lookup is in runtime is constant time,
|
|
a simple index operation.
|
|
|
|
- class Item_splocal (Item.{cc,h})
|
|
This is a subclass of Item. Its sole purpose is to hide the fact that
|
|
the real Item is actually in the current frame (runtime context).
|
|
It contains the frame offset and defers all methods to the real Item
|
|
in the frame. This is what the parser generates for local variables.
|
|
|
|
- Utility functions (sp.{cc,h})
|
|
This contains functions for creating, dropping and finding a stored
|
|
procedure in the mysql.proc table (or the internal cache).
|
|
|
|
|
|
- Parsing CREATE PROCEDURE ...
|
|
|
|
When parsing a CREATE PROCEDURE the parser first initializes the
|
|
sphead and spcont (runtime context) fields in the Lex.
|
|
The sql_command code for the result of parsing a is
|
|
SQLCOM_CREATE_PROCEDURE.
|
|
|
|
The parsing of the parameter list and body is relatively
|
|
straight-forward:
|
|
|
|
- Parameters:
|
|
name, type and mode (IN/OUT/INOUT) is pushed to spcont
|
|
- Declared local variables:
|
|
Same as parameters (mode is then IN)
|
|
- Local Variable references:
|
|
If an identifier is found in in spcont, an Item_splocal is created
|
|
with the variable's frame index, otherwise an Item_field or Item_ref
|
|
is created (as before).
|
|
- Statements:
|
|
The Lex in THD is replaced by a new Lex structure and the statement,
|
|
is parsed as usual. A sp_instr_stmt is created, containing the new
|
|
Lex, and added to added to the instructions in sphead.
|
|
Afterwards, the procedure's Lex is restored in THD.
|
|
- SET var:
|
|
Setting a local variable generates a sp_instr_set instruction,
|
|
containing the variable's frame offset, the expression (an Item),
|
|
and the type.
|
|
- Flow control:
|
|
Flow control constructs like, IF, WHILE, etc, generate a conditional
|
|
and unconditional jumps in the "obvious" way, but a few notes may
|
|
be required:
|
|
- Forward jumps: When jumping forward, the exact destination is not
|
|
known at the time of the creation of the jump instruction. The
|
|
sphead therefore contains list of instruction-label pairs for
|
|
each forward reference. When the position later is known, the
|
|
instructions in the list are updated with the correct location.
|
|
- Loop constructs have optional labels. If a loop doesn't have a
|
|
label, an anonymous label is generated to simplify the parsing.
|
|
- There are two types of CASE. The "simple" case is implemented
|
|
with an anonymous variable bound to the value to be tested.
|
|
|
|
|
|
- A simple example
|
|
|
|
Parsing the procedure:
|
|
|
|
create procedure a(s char(16))
|
|
begin
|
|
declare x int;
|
|
set x = 3;
|
|
while x > 0 do
|
|
set x = x-1;
|
|
insert into db.tab values (x, s);
|
|
end while;
|
|
end
|
|
|
|
would generate the following structures:
|
|
______
|
|
thd: | | _________
|
|
| lex -+--->| | ___________________
|
|
|______| | spcont -+------------------->| "s",in,char(16):0 |
|
|
| sphead -+------ |("x",in,int :1)|
|
|
|_________| | |___________________|
|
|
____V__________________
|
|
| m_name: "a" |
|
|
| m_defstr: "create ..."|
|
|
| m_instr: ... |
|
|
|_______________________|
|
|
|
|
Note that the contents of the spcont is changing during the parsing,
|
|
at all times reflecting the state of the would-be runtime frame.
|
|
The m_instr is an array of instructions:
|
|
|
|
Pos. Instruction
|
|
0 sp_instr_set(1, '3')
|
|
1 sp_instr_jump_if_not(5, 'x>0')
|
|
2 sp_instr_set(1, 'x-1')
|
|
3 sp_instr_stmt('insert into ...')
|
|
4 sp_instr_jump(1)
|
|
5 <end>
|
|
|
|
Here, '3', 'x>0', etc, represent the Items or Lex for the respective
|
|
expressions or statements.
|
|
|
|
|
|
- Parsing CREATE FUNCTION ...
|
|
|
|
Creating a functions is essensially the same thing as for a PROCEDURE,
|
|
with the addition that a FUNCTION has a return type and a RETURN
|
|
statement, but no OUT or INOUT parameters.
|
|
|
|
The main difference during parsing is that we store the result type
|
|
in the sp_head. However, there are big differences when it comes to
|
|
invoking a FUNCTION. (See below.)
|
|
|
|
|
|
- Storing, caching, dropping...
|
|
|
|
As seen above, the entired definition string, including the "CREATE
|
|
PROCEDURE" (or "FUNCTION") is kept. The procedure definition string is
|
|
stored in the table mysql.proc with the name and type as the key, the
|
|
type being one of the enum ("procedure","function").
|
|
|
|
A PROCEDURE is just stored in the mysql.proc table. A FUNCTION has an
|
|
additional requirement. They will be called in expressions with the same
|
|
syntax as UDFs, so UDFs and stored FUNCTIONs share the namespace. Thus,
|
|
we must make sure that we do not have UDFs and FUNCTIONs with the same
|
|
name (even if they are storded in different places).
|
|
|
|
This means that we can reparse the procedure as many time as we want.
|
|
The first time, the resulting Lex is used to store the procedure in
|
|
the database (using the function sp.c:sp_create_procedure()).
|
|
|
|
The simplest way would be to just leave it at that, and re-read the
|
|
procedure from the database each time it is called. (And in fact, that's
|
|
the way the earliest implementation will work.)
|
|
However, this is not very efficient, and we can do better. The full
|
|
implementation should work like this:
|
|
|
|
1) Upon creation time, parse and store the procedure. Note that we still
|
|
need to parse it to catch syntax errors, but we can't check if called
|
|
procedures exists for instance.
|
|
2) Upon first CALL, read from the database, parse it, and cache the
|
|
resulting Lex in memory. This time we can do more error checking.
|
|
3) Upon subsequent CALLs, use the cached Lex.
|
|
|
|
Note that this implies that the Lex structure with its sphead must be
|
|
reentrant, that is, reusable and shareable between different threads
|
|
and calls. The runtime state for a procedure is kept in the sp_rcontext
|
|
in THD.
|
|
|
|
The mechanisms of storing, finding, and dropping procedures are
|
|
encapsulated in the files sp.{cc,h}.
|
|
|
|
|
|
- CALLing a procedure
|
|
|
|
A CALL is parsed just like any statement. The resulting Lex has the
|
|
sql_command SQLCOM_CALL, the procedure's name and the parameters are
|
|
pushed to the Lex' value_list.
|
|
|
|
sql_parse.cc:mysql_execute_command() then uses sp.cc:sp_find() to
|
|
get the sp_head for the procedure (which may have been read from the
|
|
database or feetched from the in-memory cache) and calls the sp_head's
|
|
method execute().
|
|
Note: It's important that substatements called by the procedure do not
|
|
do send_ok(). Fortunately, there is a flag in THD->net to disable
|
|
this during CALLs. If a substatement fails, it will however send
|
|
an error back to the client, so the CALL mechanism must return
|
|
immediately and without sending an error.
|
|
|
|
The sp_head::execute() method works as follows:
|
|
|
|
1) Keep a pointer to the old runtime context in THD (if any)
|
|
2) Create a new runtime context. The information about the required size
|
|
is in sp_head's parse time context.
|
|
3) Push each parameter (from the CALL's Lex->value_list) to the new
|
|
context. If it's an OUT or INOUT parameter, the parameter's offset
|
|
in the caller's frame is set in the new context as well.
|
|
4) For each instruction, call its execute() method.
|
|
The result is a pointer to the next instruction to execute (or NULL)
|
|
if an error occured.
|
|
5) On success, set the new values of the OUT and INOUT parameters in
|
|
the caller's frame.
|
|
|
|
- USE database
|
|
|
|
Before executing the instruction we also keeps the current default
|
|
database (if any). If this was changed during execution (i.e. a "USE"
|
|
statement has been executed), we restore the current database to the
|
|
original.
|
|
|
|
This is the most useful way to handle USE in procedures. If we didn't,
|
|
the caller would find himself in a different database after calling
|
|
a function, which can be confusing.
|
|
Restoring the database also gives full freedom to the procedure writer:
|
|
- It's possible to write "general" procedures that are independent of
|
|
the actual database name.
|
|
- It's possible to write procedures that work on a particular database
|
|
by calling USE, without having to use fully qualified table names
|
|
everywhere (which doesn't help if you want to call other, "general",
|
|
procedures anyway).
|
|
|
|
- Evaluating Items
|
|
|
|
There are three occasions where we need to evaluate an expression:
|
|
|
|
- When SETing a variable
|
|
- When CALLing a procedure
|
|
- When testing an expression for a branch (in IF, WHILE, etc)
|
|
|
|
The semantics in stored procedures is "call-by-value", so we have to
|
|
evaluate any "func" Items at the point of the CALL or SET, otherwise
|
|
we would get a kind of "lazy" evaluation with unexpected results with
|
|
respect to OUT parameters for instance.
|
|
For this the support function, sp_head.cc:eval_func_item() is needed.
|
|
|
|
|
|
- Calling a FUNCTION
|
|
|
|
Functions don't have an explicit call keyword like procedures. Instead,
|
|
they appear in expressions with the conventional syntax "fun(arg, ...)".
|
|
The problem is that we already have User Defined Functions (UDFs) which
|
|
are called the same way. A UDF is detected by the lexical analyzer (not
|
|
the parser!), in the find_keyword() function, and returns a UDF_*_FUNC
|
|
or UDA_*_SUM token with the udf_func object as the yylval.
|
|
|
|
So, stored functions must be handled in a simpilar way, and as a
|
|
consequence, UDFs and functions must not have the same name.
|
|
|
|
- Detecting and parsing a FUNCTION invocation
|
|
|
|
The existance of UDFs are checked during the lexical analysis (in
|
|
sql_lex.cc:find_keyword()). This has the drawback that they must
|
|
exist before they are refered to, which was ok before SPs existed,
|
|
but then it becomes a problem. The first implementation of SP FUNCTIONs
|
|
will work the same way, but this should be fixed a.s.a.p. (This will
|
|
required some reworking of the way UDFs are handled, which is why it's
|
|
not done from the start.)
|
|
For the time being, a FUNCTION is detected the same way, and returns
|
|
the token SP_FUNC. During the parsing we only check for the *existance*
|
|
of the function, we don't parse it, since wa can't call the parser
|
|
recursively.
|
|
|
|
When encountering a SP_FUNC with parameters in the expression parser,
|
|
an instance of the new Item_func_sp class is created. Unlike UDFs, we
|
|
don't have different classes for different return types, since we at
|
|
this point don't know the type.
|
|
|
|
- Collecting FUNCTIONs to invoke
|
|
|
|
A FUNCTION differs from a PROCEDURE in one important aspect: Whereas a
|
|
PROCEDURE is CALLed as statement by itself, a FUNCTION is invoked
|
|
"on-the-fly" during the execution of *another* statement.
|
|
This makes things a lot more complicated compared to CALL:
|
|
- We can't read and parse the FUNCTION from the mysql.proc table at the
|
|
point of invocation; the server requires that all tables used are
|
|
opened and locked at the beginning of the query execution.
|
|
One "obvious" solution would be to simply push "mysql.proc" to the list
|
|
of tables used by the query, but this implies a "join" with this table
|
|
if the query is a select, so it doesn't work (and we can't exclude this
|
|
table easily; since a priviledged used might in fact want to search
|
|
the proc table).
|
|
Another solution would of course be to allow the opening and closing
|
|
of the mysql.proc table during a query execution, but this it not
|
|
possible at the present.
|
|
|
|
So, the solution is to collect the names of the refered FUNCTIONs during
|
|
parsing in the lex.
|
|
Then, before doing anything else in mysql_execute_command(), read all
|
|
functions from the database an keep them in the THD, where the function
|
|
sp_find_function() can find them during the execution.
|
|
Note: Even with an in-memory cache, we must still make sure that the
|
|
functions are indeed read and cached at this point.
|
|
The code that read and cache functions from the database must also be
|
|
invoked recursively for each read FUNCTION to make sure we have *all* the
|
|
functions we need.
|
|
|
|
|
|
- Parsing DROP PROCEDURE/FUNCTION
|
|
|
|
The procedure name is pushed to Lex->value_list.
|
|
The sql_command code for the result of parsing a is
|
|
SQLCOM_DROP_PROCEDURE/SQLCOM_DROP_FUNCTION.
|
|
|
|
Dropping is done by simply getting the procedure with the sp_find()
|
|
function and calling sp_drop() (both in sp.{cc,h}).
|
|
|
|
DROP PROCEDURE/FUNCTION also supports the non-standard "IF EXISTS",
|
|
analogous to other DROP statements in MySQL.
|
|
|
|
|
|
- Condition and Handlers
|
|
|
|
Condition names are lexical entities and are kept in the parser context
|
|
just like variables. But, condition are just "aliases" for SQLSTATE
|
|
strings, or mysqld error codes (which is a non-standard extension in
|
|
MySQL), and are only used during parsing.
|
|
|
|
Handlers comes in three types, CONTINUE, EXIT and UNDO. The latter is
|
|
like an EXIT handler with an implicit rollback, and is currently not
|
|
implemented.
|
|
The EXIT handler jumps to the end of its BEGIN-END block when finished.
|
|
The CONTINUE handler returns to the statement following that which
|
|
invoked the handler.
|
|
|
|
The handlers in effect at any point is part of each thread's runtime
|
|
state, so we need to push and pop handlers in the sp_rcontext during
|
|
execution. We use special instructions for this:
|
|
- sp_instr_hpush_jump
|
|
Push a handler. The instruction contains the necessary information,
|
|
like which conditions we handle and the location of the handler.
|
|
The jump takes us to the location after the handler code.
|
|
- sp_instr_hpop
|
|
Pop the handlers of the current frame (which we are just leaving).
|
|
|
|
It might seems strange to jump past the handlers like that, but there's
|
|
no extra cost in doing this, and for technical reasons it's easiest for
|
|
the parser to generate the handler instructions when they occur in the
|
|
source.
|
|
|
|
When an error occurs, one of the error routines is called and an error
|
|
message is normally sent back to the client immediately.
|
|
Catching a condition must be done in these error routines (there are
|
|
quite a few) to prevent them from doing this. We do this by calling
|
|
a method in the THD's sp_rcontext (if there is one). If a handler is
|
|
found, this is recorded in the context and the routine returns without
|
|
sending the error message.
|
|
The exectution loop (sp_head::execute()) checks for this after each
|
|
statement and invokes the handler that has been found. If several
|
|
errors or warnings occurs during one statement, only the first is
|
|
caught, the rest are ignored.
|
|
|
|
Invoking and returning from a handler is trivial in the EXIT case.
|
|
We simply jump to it, and it will have an sp_instr_jump as its last
|
|
instruction.
|
|
|
|
Calling and returning from a CONTINUE handler poses some special
|
|
problems. Since we need to return to the point after its invokation,
|
|
we push the return location on a stack in the sp_rcontext (this is
|
|
done by the exectution loop). The handler then ends with a special
|
|
instruction, sp_instr_hreturn, which returns to this location.
|
|
|
|
CONTINUE handlers have one additional problem: They are parsed at
|
|
the lexical level where they occur, so variable offsets will assume
|
|
that it's actually called at that level. However, a handler might be
|
|
invoked from a sub-block where additional local variables have been
|
|
declared, which will then share the location of any local variables
|
|
in the handler itself. So, when calling a CONTINUE handler, we need
|
|
to save any local variables above the handler's frame offset, and
|
|
restore them upon return. (This is not a problem for EXIT handlers,
|
|
since they will leave the block anyway.)
|
|
This is taken care of by the execution loop and the sp_instr_hreturn
|
|
instruction.
|
|
|
|
- Examples:
|
|
|
|
- EXIT handler
|
|
begin
|
|
declare x int default 0;
|
|
|
|
begin
|
|
declare exit handler for 'XXXXX' set x = 1;
|
|
|
|
(statement1);
|
|
(statement2);
|
|
end;
|
|
(statement3);
|
|
end
|
|
|
|
Pos. Instruction
|
|
0 sp_instr_set(0, '0')
|
|
1 sp_instr_hpush_jump(4, 1) # location and frame size
|
|
2 sp_instr_set(0, '1')
|
|
3 sp_instr_jump(6)
|
|
4 sp_instr_stmt('statement1')
|
|
5 sp_instr_stmt('statement2')
|
|
6 sp_instr_hpop(1)
|
|
7 sp_instr_stmt('statement3')
|
|
|
|
- CONTINUE handler
|
|
create procedure hndlr1(val int)
|
|
begin
|
|
declare x int default 0;
|
|
declare foo condition for 1146;
|
|
declare continue handler for foo set x = 1;
|
|
|
|
insert into t3 values ("hndlr1", val); # Non-existing table?
|
|
if x>0 then
|
|
insert into t1 values ("hndlr1", val); # This instead then
|
|
end if;
|
|
end|
|
|
|
|
Pos. Instruction
|
|
0 sp_instr_set(1, '0')
|
|
1 sp_instr_hpush_jump(4, 2)
|
|
2 sp_instr_set(1, '1')
|
|
3 sp_instr_hreturn(2) # frame size
|
|
4 sp_instr_stmt('insert ... t3 ...')
|
|
5 sp_instr_jump_if_not(7, 'x>0')
|
|
6 sp_instr_stmt('insert ... t1 ...')
|
|
7 sp_instr_hpop(2)
|
|
|
|
|
|
- Cursors
|
|
|
|
For stored procedures to be really useful, you want to have cursors.
|
|
MySQL doesn't yet have "real" cursor support (with API and ODBC support,
|
|
allowing updating, arbitrary scrolling, etc), but a simple asensitive,
|
|
non-scrolling, read-only cursor can be implemented in SPs using the
|
|
class Protocol_cursor.
|
|
This class intecepts the creation and sending of results sets and instead
|
|
stores it in-memory, as MYSQL_FIELDS and MYSQL_ROWS (as in the client API).
|
|
|
|
To support this, we need the usual name binding support in sp_pcontext
|
|
(similar to variables and conditions) to keep track on declared cursor
|
|
names, and a corresponding run-time mechanism in sp_rcontext.
|
|
Cursors are lexically scoped like everything with a body or BEGIN/END
|
|
block, so they are pushed and poped as usual (see conditions and variables
|
|
above).
|
|
The basic operations on a cursor are OPEN, FETCH and CLOSE, which will
|
|
each have a corresponding instruction. In addition, we need instructions
|
|
to push a new cursor (this will encapsulate the LEX of the SELECT statement
|
|
of the cursor), and a pop instruction:
|
|
- sp_instr_cpush
|
|
Push a cursor to the sp_rcontext. This instruction contains the LEX
|
|
for the select statement
|
|
- sp_instr_cpop
|
|
Pop a number of cursors from the sp_rcontext.
|
|
- sp_instr_copen
|
|
Open a cursor: This will execute the select and get the result set
|
|
in a sepeate memroot.
|
|
- sp_instr_cfetch
|
|
Fetch the next row from the in-memory result set. The instruction
|
|
contains a list of the variables (frame offsets) to set.
|
|
- sp_instr_cclose
|
|
Free the result set.
|
|
|
|
A cursor is a separate class, sp_cursor (defined in sp_rcontex.h) which
|
|
encapsulates the basic operations used by the above instructions.
|
|
This class contains the LEX, Protocol_cursor object, and its memroot,
|
|
as well as the cursor's current state.
|
|
Compiling and executing is fairly straight-forward. sp_instr_copen is
|
|
a subclass of sp_instr_stmt and uses its mechanism to execute a
|
|
substatement.
|
|
|
|
- Example:
|
|
|
|
begin
|
|
declare x int;
|
|
declare c cursor for select a from t1;
|
|
|
|
open c;
|
|
fetch c into x;
|
|
close c;
|
|
end
|
|
|
|
Pos. Instruction
|
|
0 sp_instr_cpush('select a from ...')
|
|
1 sp_instr_copen(0) # The 0'th cursor
|
|
2 sp_instr_cfetch(0) # Contains the variable list
|
|
3 sp_instr_cclose(0)
|
|
4 sp_instr_cpop(1)
|
|
|
|
|
|
|
|
- The SP cache
|
|
|
|
There are two ways to cache SPs:
|
|
|
|
1) one global cache, share by all threads/connections,
|
|
2) one cache per thread.
|
|
|
|
There are pros and cons with both methods:
|
|
|
|
1) Pros: Save memory, each SP only read from table once,
|
|
Cons: Needs locking (= serialization at access), requires thread-safe
|
|
data structures,
|
|
2) Pros: Fast, no locking required (almost), limited thread-safe
|
|
requirement,
|
|
Cons: Uses more memory, each SP read from table once per thread.
|
|
|
|
Unfortunately, we cannot use alternative 1 for the time being, as most
|
|
of the datastructures to be cached (lex and items) are not reentrant
|
|
and thread-safe. (Things are modifed at execution, we have THD pointers
|
|
stored everywhere, etc.)
|
|
This leaves us with alternative 2, one cache per thread; or actually
|
|
two, since we keep FUNCTIONs and PROCEDUREs in separate caches.
|
|
This is not that terrible; the only case when it will perform
|
|
significantly worse than a global cache is when we have an application
|
|
where new threads are connecting, calling a procedure, and disconnecting,
|
|
over and over again.
|
|
|
|
The cache implementation itself is simple and straightforward, a hashtable
|
|
wrapped in a class and a C API (see APIs below).
|
|
|
|
There is however one issue with multiple caches: dropping and altering
|
|
procedures. Normally, this should be a very rare event in a running
|
|
system; it's typically something you do during development and testing,
|
|
so it's not unthinkable that we would simply ignore the issue and let
|
|
any threads running with a cached version of an SP keep doing so until
|
|
its disconnected.
|
|
But assuming we want to keep the caches consistent with respect to drop
|
|
and alter, it can be done:
|
|
|
|
1) A global counter is needed, initialized to 0 at start.
|
|
2) At each DROP or ALTER, increase the counter by one.
|
|
3) Each cache has its own copy of the counter, copied at the last read.
|
|
4) When looking up a name in the cache, first check if the global counter
|
|
is larger than the local copy.
|
|
If so, clear the cache and return "not found", and update the local
|
|
counter; otherwise, lookup as usual.
|
|
|
|
This minimizes the cost to a single brief lock for the access of an
|
|
integer when operating normally. Only in the event of an actual drop or
|
|
alter, is the cache cleared. This may seem to be drastic, but since we
|
|
assume that this is a rare event, it's not a problem.
|
|
It would of course be possible to have a much more fine-grained solution,
|
|
keeping track of each SP, but the overhead of doing so is not worth the
|
|
effort.
|
|
|
|
|
|
- Class and function APIs
|
|
This is an outline of the key types. Some types and other details
|
|
in the actual files have been omitted for readability.
|
|
|
|
- The parser context: sp_pcontext.h
|
|
|
|
typedef enum
|
|
{
|
|
sp_param_in,
|
|
sp_param_out,
|
|
sp_param_inout
|
|
} sp_param_mode_t;
|
|
|
|
typedef struct
|
|
{
|
|
LEX_STRING name;
|
|
enum enum_field_types type;
|
|
sp_param_mode_t mode;
|
|
uint offset; // Offset in current frame
|
|
my_bool isset;
|
|
} sp_pvar_t;
|
|
|
|
typedef struct sp_cond_type
|
|
{
|
|
enum { number, state, warning, notfound, exception } type;
|
|
char sqlstate[6];
|
|
uint mysqlerr;
|
|
} sp_cond_type_t;
|
|
|
|
class sp_pcontext
|
|
{
|
|
sp_pcontext();
|
|
|
|
// Return the maximum frame size
|
|
uint max_framesize();
|
|
|
|
// Return the current frame size
|
|
uint current_framesize();
|
|
|
|
// Return the number of parameters
|
|
uint params();
|
|
|
|
// Set the number of parameters to the current frame size
|
|
void set_params();
|
|
|
|
// Set type of the variable at offset 'i' in the frame
|
|
void set_type(uint i, enum enum_field_types type);
|
|
|
|
// Mark the i:th variable to "set" (i.e. having a value) with
|
|
// 'val' true.
|
|
void set_isset(uint i, my_bool val);
|
|
|
|
// Push the variable 'name' to the frame.
|
|
void push_var(LEX_STRING *name,
|
|
enum enum_field_types type, sp_param_mode_t mode);
|
|
|
|
// Pop 'num' variables from the frame.
|
|
void pop_var(uint num = 1);
|
|
|
|
// Find variable by name
|
|
sp_pvar_t *find_pvar(LEX_STRING *name);
|
|
|
|
// Find variable by index
|
|
sp_pvar_t *find_pvar(uint i);
|
|
|
|
// Push label 'name' of instruction index 'ip' to the label context
|
|
sp_label_t *push_label(char *name, uint ip);
|
|
|
|
// Find label 'name' in the context
|
|
sp_label_t *find_label(char *name);
|
|
|
|
// Return the last pushed label
|
|
sp_label_t *last_label();
|
|
|
|
// Return and remove the last pushed label.
|
|
sp_label_t *pop_label();
|
|
|
|
// Push a condition to the context
|
|
void push_cond(LEX_STRING *name, sp_cond_type_t *val);
|
|
|
|
// Pop a 'num' condition from the context
|
|
void pop_cond(uint num);
|
|
|
|
// Find a condition in the context
|
|
sp_cond_type_t *find_cond(LEX_STRING *name);
|
|
|
|
// Increase the handler count
|
|
void add_handler();
|
|
|
|
// Returns the handler count
|
|
uint handlers();
|
|
|
|
// Push a cursor
|
|
void push_cursor(LEX_STRING *name);
|
|
|
|
// Find a cursor
|
|
my_bool find_cursor(LEX_STRING *name, uint *poff);
|
|
|
|
// Pop 'num' cursors
|
|
void pop_cursor(uint num);
|
|
|
|
// Return the number of cursors
|
|
uint cursors();
|
|
}
|
|
|
|
|
|
- The run-time context (call frame): sp_rcontext.h
|
|
|
|
#define SP_HANDLER_NONE 0
|
|
#define SP_HANDLER_EXIT 1
|
|
#define SP_HANDLER_CONTINUE 2
|
|
#define SP_HANDLER_UNDO 3
|
|
|
|
typedef struct
|
|
{
|
|
struct sp_cond_type *cond;
|
|
uint handler; // Location of handler
|
|
int type;
|
|
uint foffset; // Frame offset for the handlers declare level
|
|
} sp_handler_t;
|
|
|
|
class sp_rcontext
|
|
{
|
|
// 'fsize' is the max size of the context, 'hmax' the number of handlers,
|
|
// 'cmax' the number of cursors
|
|
sp_rcontext(uint fsize, uint hmax, , uint cmax);
|
|
|
|
// Push value (parameter) 'i' to the frame
|
|
void push_item(Item *i);
|
|
|
|
// Set slot 'idx' to value 'i'
|
|
void set_item(uint idx, Item *i);
|
|
|
|
// Return the item in slot 'idx'
|
|
Item *get_item(uint idx);
|
|
|
|
// Set the "out" index 'oidx' for slot 'idx. If it's an IN slot,
|
|
// use 'oidx' -1.
|
|
void set_oindex(uint idx, int oidx);
|
|
|
|
// Return the "out" index for slot 'idx'
|
|
int get_oindex(uint idx);
|
|
|
|
// Set the FUNCTION result
|
|
void set_result(Item *i);
|
|
|
|
// Get the FUNCTION result
|
|
Item *get_result();
|
|
|
|
// Push handler at location 'h' for condition 'cond'. 'f' is the
|
|
// current variable frame size.
|
|
void push_handler(sp_cond_type_t *cond, uint h, int type, uint f);
|
|
|
|
// Pop 'count' handlers
|
|
void pop_handlers(uint count);
|
|
|
|
// Find a handler for this error. This sets the state for a found
|
|
// handler in the context. If called repeatedly without clearing,
|
|
// only the first call's state is kept.
|
|
int find_handler(uint sql_errno);
|
|
|
|
// Returns 1 if a handler has been found, with '*ip' and '*fp' set
|
|
// to the handler location and frame size respectively.
|
|
int found_handler(uint *ip, uint *fp);
|
|
|
|
// Clear the found handler state.
|
|
void clear_handler();
|
|
|
|
// Push a return address for a CONTINUE handler
|
|
void push_hstack(uint ip);
|
|
|
|
// Pop the CONTINUE handler return stack
|
|
uint pop_hstack();
|
|
|
|
// Save variables from frame index 'fp' and up.
|
|
void save_variables(uint fp);
|
|
|
|
// Restore saved variables from to frame index 'fp' and up.
|
|
void restore_variables(uint fp);
|
|
|
|
// Push a cursor for the statement (lex)
|
|
void push_cursor(LEX *lex);
|
|
|
|
// Pop 'count' cursors
|
|
void pop_cursors(uint count);
|
|
|
|
// Pop all cursors
|
|
void pop_all_cursors();
|
|
|
|
// Get the 'i'th cursor
|
|
sp_cursor *get_cursor(uint i);
|
|
|
|
}
|
|
|
|
|
|
- The procedure: sp_head.h
|
|
|
|
#define TYPE_ENUM_FUNCTION 1
|
|
#define TYPE_ENUM_PROCEDURE 2
|
|
|
|
class sp_head
|
|
{
|
|
int m_type; // TYPE_ENUM_FUNCTION or TYPE_ENUM_PROCEDURE
|
|
|
|
sp_head();
|
|
|
|
void init(LEX_STRING *name, LEX *lex, LEX_STRING *comment, char suid);
|
|
|
|
// Store this procedure in the database. This is a wrapper around
|
|
// the function sp_create_procedure().
|
|
int create(THD *);
|
|
|
|
// Invoke a FUNCTION
|
|
int
|
|
execute_function(THD *thd, Item **args, uint argcount, Item **resp);
|
|
|
|
// CALL a PROCEDURE
|
|
int
|
|
execute_procedure(THD *thd, List<Item> *args);
|
|
|
|
// Add the instruction to this procedure.
|
|
void add_instr(sp_instr *);
|
|
|
|
// Returns the number of instructions.
|
|
uint instructions();
|
|
|
|
// Returns the last instruction
|
|
sp_instr *last_instruction();
|
|
|
|
// Resets lex in 'thd' and keeps a copy of the old one.
|
|
void reset_lex(THD *);
|
|
|
|
// Restores lex in 'thd' from our copy, but keeps some status from the
|
|
// one in 'thd', like ptr, tables, fields, etc.
|
|
void restore_lex(THD *);
|
|
|
|
// Put the instruction on the backpatch list, associated with
|
|
// the label.
|
|
void push_backpatch(sp_instr *, struct sp_label *);
|
|
|
|
// Update all instruction with this label in the backpatch list to
|
|
// the current position.
|
|
void backpatch(struct sp_label *);
|
|
|
|
// Returns the SP name (with optional length in '*lenp').
|
|
char *name(uint *lenp = 0);
|
|
|
|
// Returns the result type for a function
|
|
Item_result result();
|
|
|
|
// Sets various attributes
|
|
void sp_set_info(char *creator, uint creatorlen,
|
|
longlong created, longlong modified,
|
|
bool suid, char *comment, uint commentlen);
|
|
}
|
|
|
|
|
|
- Instructions
|
|
|
|
- The base class:
|
|
class sp_instr
|
|
{
|
|
// 'ip' is the index of this instruction
|
|
sp_instr(uint ip);
|
|
|
|
// Execute this instrution.
|
|
// '*nextp' will be set to the index of the next instruction
|
|
// to execute. (For most instruction this will be the
|
|
// instruction following this one.)
|
|
// Returns 0 on success, non-zero if some error occured.
|
|
virtual int execute(THD *, uint *nextp)
|
|
}
|
|
|
|
- Statement instruction:
|
|
class sp_instr_stmt : public sp_instr
|
|
{
|
|
sp_instr_stmt(uint ip);
|
|
|
|
int execute(THD *, uint *nextp);
|
|
|
|
// Set the statement's Lex
|
|
void set_lex(LEX *);
|
|
|
|
// Return the statement's Lex
|
|
LEX *get_lex();
|
|
}
|
|
|
|
- SET instruction:
|
|
class sp_instr_set : public sp_instr
|
|
{
|
|
// 'offset' is the variable's frame offset, 'val' the value,
|
|
// and 'type' the variable type.
|
|
sp_instr_set(uint ip,
|
|
uint offset, Item *val, enum enum_field_types type);
|
|
|
|
int execute(THD *, uint *nextp);
|
|
}
|
|
|
|
- Unconditional jump
|
|
class sp_instr_jump : public sp_instr
|
|
{
|
|
// No destination, must be set.
|
|
sp_instr_jump(uint ip);
|
|
|
|
// 'dest' is the destination instruction index.
|
|
sp_instr_jump(uint ip, uint dest);
|
|
|
|
int execute(THD *, uint *nextp);
|
|
|
|
// Set the destination instruction 'dest'.
|
|
void set_destination(uint dest);
|
|
}
|
|
|
|
- Conditional jump
|
|
class sp_instr_jump_if_not : public sp_instr_jump
|
|
{
|
|
// Jump if 'i' evaluates to false. Destination not set yet.
|
|
sp_instr_jump_if_not(uint ip, Item *i);
|
|
|
|
// Jump to 'dest' if 'i' evaluates to false.
|
|
sp_instr_jump_if_not(uint ip, Item *i, uint dest)
|
|
|
|
int execute(THD *, uint *nextp);
|
|
}
|
|
|
|
- Return a function value
|
|
class sp_instr_freturn : public sp_instr
|
|
{
|
|
// Return the value 'val'
|
|
sp_instr_freturn(uint ip, Item *val, enum enum_field_types type);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Push a handler and jump
|
|
class sp_instr_hpush_jump : public sp_instr_jump
|
|
{
|
|
// Push handler of type 'htype', with current frame size 'fp'
|
|
sp_instr_hpush_jump(uint ip, int htype, uint fp);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
|
|
// Add condition for this handler
|
|
void add_condition(struct sp_cond_type *cond);
|
|
}
|
|
|
|
- Pops handlers
|
|
class sp_instr_hpop : public sp_instr
|
|
{
|
|
// Pop 'count' handlers
|
|
sp_instr_hpop(uint ip, uint count);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Return from a CONTINUE handler
|
|
class sp_instr_hreturn : public sp_instr
|
|
{
|
|
// Return from handler, and restore variables to 'fp'.
|
|
sp_instr_hreturn(uint ip, uint fp);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Push a CURSOR
|
|
class sp_instr_cpush : public sp_instr_stmt
|
|
{
|
|
// Push a cursor for statement 'lex'
|
|
sp_instr_cpush(uint ip, LEX *lex)
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Pop CURSORs
|
|
class sp_instr_cpop : public sp_instr_stmt
|
|
{
|
|
// Pop 'count' cursors
|
|
sp_instr_cpop(uint ip, uint count)
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Open a CURSOR
|
|
class sp_instr_copen : public sp_instr_stmt
|
|
{
|
|
// Open the 'c'th cursor
|
|
sp_instr_copen(uint ip, uint c);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Close a CURSOR
|
|
class sp_instr_cclose : public sp_instr
|
|
{
|
|
// Close the 'c'th cursor
|
|
sp_instr_cclose(uint ip, uint c);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
}
|
|
|
|
- Fetch a row with CURSOR
|
|
class sp_instr_cfetch : public sp_instr
|
|
{
|
|
// Fetch next with the 'c'th cursor
|
|
sp_instr_cfetch(uint ip, uint c);
|
|
|
|
int execute(THD *thd, uint *nextp);
|
|
|
|
// Add a target variable for the fetch
|
|
void add_to_varlist(struct sp_pvar *var);
|
|
}
|
|
|
|
|
|
- Utility functions: sp.h
|
|
|
|
#define SP_OK 0
|
|
#define SP_KEY_NOT_FOUND -1
|
|
#define SP_OPEN_TABLE_FAILED -2
|
|
#define SP_WRITE_ROW_FAILED -3
|
|
#define SP_DELETE_ROW_FAILED -4
|
|
#define SP_GET_FIELD_FAILED -5
|
|
#define SP_PARSE_ERROR -6
|
|
|
|
// Finds a stored procedure given its name. Returns NULL if not found.
|
|
sp_head *sp_find_procedure(THD *, LEX_STRING *name);
|
|
|
|
// Store the procedure 'name' in the database. 'def' is the complete
|
|
// definition string ("create procedure ...").
|
|
int sp_create_procedure(THD *,
|
|
char *name, uint namelen,
|
|
char *def, uint deflen,
|
|
char *comment, uint commentlen, bool suid);
|
|
|
|
// Drop the procedure 'name' from the database.
|
|
int sp_drop_procedure(THD *, char *name, uint namelen);
|
|
|
|
// Finds a stored function given its name. Returns NULL if not found.
|
|
sp_head *sp_find_function(THD *, LEX_STRING *name);
|
|
|
|
// Store the function 'name' in the database. 'def' is the complete
|
|
// definition string ("create function ...").
|
|
int sp_create_function(THD *,
|
|
char *name, uint namelen,
|
|
char *def, uint deflen,
|
|
char *comment, uint commentlen, bool suid);
|
|
|
|
// Drop the function 'name' from the database.
|
|
int sp_drop_function(THD *, char *name, uint namelen);
|
|
|
|
|
|
- The cache: sp_cache.h
|
|
|
|
/* Initialize the SP caching once at startup */
|
|
void sp_cache_init();
|
|
|
|
/* Clear the cache *cp and set *cp to NULL */
|
|
void sp_cache_clear(sp_cache **cp);
|
|
|
|
/* Insert an SP to cache. If **cp points to NULL, it's set to a
|
|
new cache */
|
|
void sp_cache_insert(sp_cache **cp, sp_head *sp);
|
|
|
|
/* Lookup an SP in cache */
|
|
sp_head *sp_cache_lookup(sp_cache **cp, char *name, uint namelen);
|
|
|
|
/* Remove an SP from cache */
|
|
void sp_cache_remove(sp_cache **cp, sp_head *sp);
|
|
|
|
|
|
- The mysql.proc schema:
|
|
|
|
CREATE TABLE proc (
|
|
db char(64) binary DEFAULT '' NOT NULL,
|
|
name char(64) DEFAULT '' NOT NULL,
|
|
type enum('FUNCTION','PROCEDURE') NOT NULL,
|
|
specific_name char(64) DEFAULT '' NOT NULL,
|
|
language enum('SQL') DEFAULT 'SQL' NOT NULL,
|
|
sql_data_access enum('CONTAINS_SQL') DEFAULT 'CONTAINS_SQL' NOT NULL,
|
|
is_deterministic enum('YES','NO') DEFAULT 'NO' NOT NULL,
|
|
security_type enum('INVOKER','DEFINER') DEFAULT 'DEFINER' NOT NULL,
|
|
param_list blob DEFAULT '' NOT NULL,
|
|
returns char(64) DEFAULT '' NOT NULL,
|
|
body blob DEFAULT '' NOT NULL,
|
|
definer char(77) binary DEFAULT '' NOT NULL,
|
|
created timestamp,
|
|
modified timestamp,
|
|
sql_mode set(
|
|
'REAL_AS_FLOAT',
|
|
'PIPES_AS_CONCAT',
|
|
'ANSI_QUOTES',
|
|
'IGNORE_SPACE',
|
|
'NOT_USED',
|
|
'ONLY_FULL_GROUP_BY',
|
|
'NO_UNSIGNED_SUBTRACTION',
|
|
'NO_DIR_IN_CREATE',
|
|
'POSTGRESQL',
|
|
'ORACLE',
|
|
'MSSQL',
|
|
'DB2',
|
|
'MAXDB',
|
|
'NO_KEY_OPTIONS',
|
|
'NO_TABLE_OPTIONS',
|
|
'NO_FIELD_OPTIONS',
|
|
'MYSQL323',
|
|
'MYSQL40',
|
|
'ANSI',
|
|
'NO_AUTO_VALUE_ON_ZERO'
|
|
) DEFAULT 0 NOT NULL,
|
|
comment char(64) binary DEFAULT '' NOT NULL,
|
|
PRIMARY KEY (db,name,type)
|
|
) comment='Stored Procedures';
|
|
|
|
--
|
|
|