Commit 36d98ce8 authored by osku's avatar osku

Rewrite pars_info datatypes and APIs, add a few helper functions.

parent 1690e0de
...@@ -15,6 +15,13 @@ Created 11/19/1996 Heikki Tuuri ...@@ -15,6 +15,13 @@ Created 11/19/1996 Heikki Tuuri
#include "pars0types.h" #include "pars0types.h"
#include "row0types.h" #include "row0types.h"
#include "trx0types.h" #include "trx0types.h"
#include "ut0vec.h"
/* Type of the user functions. The first argument is always InnoDB-supplied
and varies in type, while 'user_arg' is a user-supplied argument. The
meaning of the return type also varies. See the individual use cases, e.g.
the FETCH statement, for details on them. */
typedef void* (*pars_user_func_cb_t)(void* arg, void* user_arg);
extern int yydebug; extern int yydebug;
...@@ -439,6 +446,75 @@ pars_complete_graph_for_exec( ...@@ -439,6 +446,75 @@ pars_complete_graph_for_exec(
trx_t* trx, /* in: transaction handle */ trx_t* trx, /* in: transaction handle */
mem_heap_t* heap); /* in: memory heap from which allocated */ mem_heap_t* heap); /* in: memory heap from which allocated */
/********************************************************************
Create parser info struct.*/
pars_info_t*
pars_info_create(void);
/*==================*/
/* out, own: info struct */
/********************************************************************
Free info struct and everything it contains.*/
void
pars_info_free(
/*===========*/
pars_info_t* info); /* in: info struct */
/********************************************************************
Add bound literal. */
void
pars_info_add_literal(
/*==================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: name */
const void* address, /* in: address */
ulint length, /* in: length of data */
ulint type, /* in: type, e.g. DATA_FIXBINARY */
ulint prtype); /* in: precise type, e.g.
DATA_UNSIGNED */
/********************************************************************
Equivalent to pars_info_add_literal(info, name, str, strlen(str),
DATA_VARCHAR, DATA_ENGLISH). */
void
pars_info_add_str_literal(
/*======================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: name */
const char* str); /* in: string */
/********************************************************************
Equivalent to:
char buf[4];
mach_write_to_4(buf, val);
pars_info_add_literal(info, name, buf, 4, DATA_INT, 0);
except that the buffer is dynamically allocated from the info struct's
heap. */
void
pars_info_add_int4_literal(
/*=======================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: name */
lint val); /* in: value */
/********************************************************************
Add user function. */
void
pars_info_add_function(
/*===================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: function name */
pars_user_func_cb_t func, /* in: function address */
void* arg); /* in: user-supplied argument */
/******************************************************************** /********************************************************************
Get user function with the given name.*/ Get user function with the given name.*/
...@@ -462,21 +538,19 @@ pars_info_get_bound_lit( ...@@ -462,21 +538,19 @@ pars_info_get_bound_lit(
const char* name); /* in: bound literal name to find */ const char* name); /* in: bound literal name to find */
/* Extra information supplied for pars_sql(). All data is owned by the user /* Extra information supplied for pars_sql(). */
who's responsible for freeing them as necessary.*/
struct pars_info_struct { struct pars_info_struct {
pars_user_func_t* funcs; /* user functions */ mem_heap_t* heap; /* our own memory heap */
ulint n_funcs; /* number of user functions */
pars_bound_lit_t* bound_lits; /* bound literals */ ib_vector* funcs; /* user functions, or NUll
ulint n_bound_lits; /* number of bound literals */ (pars_user_func_t*) */
}; ib_vector* bound_lits; /* bound literals, or NULL
(pars_bound_lit_t*) */
/* Type of the user functions. The first argument is always InnoDB-supplied ibool pars_sql_owns_us;
and varies in type, while 'user_arg' is a user-supplied argument. The /* if TRUE (which is the default),
meaning of the return type also varies. See the individual use cases, e.g. pars_sql() free us before exiting */
the FETCH statement, for details on them. */ };
typedef void* (*pars_user_func_cb_t)(void* arg, void* user_arg);
/* User-supplied function and argument. */ /* User-supplied function and argument. */
struct pars_user_func_struct { struct pars_user_func_struct {
......
...@@ -1878,6 +1878,10 @@ pars_sql( ...@@ -1878,6 +1878,10 @@ pars_sql(
/* fprintf(stderr, "SQL graph size %lu\n", mem_heap_get_size(heap)); */ /* fprintf(stderr, "SQL graph size %lu\n", mem_heap_get_size(heap)); */
if (info && info->pars_sql_owns_us) {
pars_info_free(info);
}
return(graph); return(graph);
} }
...@@ -1912,6 +1916,135 @@ pars_complete_graph_for_exec( ...@@ -1912,6 +1916,135 @@ pars_complete_graph_for_exec(
return(thr); return(thr);
} }
/********************************************************************
Create parser info struct.*/
pars_info_t*
pars_info_create(void)
/*==================*/
/* out, own: info struct */
{
pars_info_t* info;
mem_heap_t* heap;
heap = mem_heap_create(512);
info = mem_heap_alloc(heap, sizeof(*info));
info->heap = heap;
info->funcs = NULL;
info->bound_lits = NULL;
info->pars_sql_owns_us = TRUE;
return(info);
}
/********************************************************************
Free info struct and everything it contains.*/
void
pars_info_free(
/*===========*/
pars_info_t* info) /* in: info struct */
{
mem_heap_free(info->heap);
}
/********************************************************************
Add bound literal. */
void
pars_info_add_literal(
/*==================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: name */
const void* address, /* in: address */
ulint length, /* in: length of data */
ulint type, /* in: type, e.g. DATA_FIXBINARY */
ulint prtype) /* in: precise type, e.g.
DATA_UNSIGNED */
{
pars_bound_lit_t* pbl;
pbl = mem_heap_alloc(info->heap, sizeof(*pbl));
pbl->name = name;
pbl->address = address;
pbl->length = length;
pbl->type = type;
pbl->prtype = prtype;
if (!info->bound_lits) {
info->bound_lits = ib_vector_create(info->heap, 8);
}
ib_vector_push(info->bound_lits, pbl);
}
/********************************************************************
Equivalent to pars_info_add_literal(info, name, str, strlen(str),
DATA_VARCHAR, DATA_ENGLISH). */
void
pars_info_add_str_literal(
/*======================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: name */
const char* str) /* in: string */
{
pars_info_add_literal(info, name, str, strlen(str),
DATA_VARCHAR, DATA_ENGLISH);
}
/********************************************************************
Equivalent to:
char buf[4];
mach_write_to_4(buf, val);
pars_info_add_literal(info, name, buf, 4, DATA_INT, 0);
except that the buffer is dynamically allocated from the info struct's
heap. */
void
pars_info_add_int4_literal(
/*=======================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: name */
lint val) /* in: value */
{
byte* buf = mem_heap_alloc(info->heap, 4);
mach_write_to_4(buf, val);
pars_info_add_literal(info, name, buf, 4, DATA_INT, 0);
}
/********************************************************************
Add user function. */
void
pars_info_add_function(
/*===================*/
pars_info_t* info, /* in: info struct */
const char* name, /* in: function name */
pars_user_func_cb_t func, /* in: function address */
void* arg) /* in: user-supplied argument */
{
pars_user_func_t* puf;
puf = mem_heap_alloc(info->heap, sizeof(*puf));
puf->name = name;
puf->func = func;
puf->arg = arg;
if (!info->funcs) {
info->funcs = ib_vector_create(info->heap, 8);
}
ib_vector_push(info->funcs, puf);
}
/******************************************************************** /********************************************************************
Get user function with the given name.*/ Get user function with the given name.*/
...@@ -1923,15 +2056,20 @@ pars_info_get_user_func( ...@@ -1923,15 +2056,20 @@ pars_info_get_user_func(
pars_info_t* info, /* in: info struct */ pars_info_t* info, /* in: info struct */
const char* name) /* in: function name to find*/ const char* name) /* in: function name to find*/
{ {
ulint i; ulint i;
ib_vector* vec;
if (!info) { if (!info || !info->funcs) {
return(NULL); return(NULL);
} }
for (i = 0; i < info->n_funcs; i++) { vec = info->funcs;
if (strcmp(info->funcs[i].name, name) == 0) {
return(&info->funcs[i]); for (i = 0; i < ib_vector_size(vec); i++) {
pars_user_func_t* puf = ib_vector_get(vec, i);
if (strcmp(puf->name, name) == 0) {
return(puf);
} }
} }
...@@ -1949,15 +2087,20 @@ pars_info_get_bound_lit( ...@@ -1949,15 +2087,20 @@ pars_info_get_bound_lit(
pars_info_t* info, /* in: info struct */ pars_info_t* info, /* in: info struct */
const char* name) /* in: bound literal name to find */ const char* name) /* in: bound literal name to find */
{ {
ulint i; ulint i;
ib_vector* vec;
if (!info) { if (!info || !info->bound_lits) {
return(NULL); return(NULL);
} }
for (i = 0; i < info->n_bound_lits; i++) { vec = info->bound_lits;
if (strcmp(info->bound_lits[i].name, name) == 0) {
return(&info->bound_lits[i]); for (i = 0; i < ib_vector_size(vec); i++) {
pars_bound_lit_t* pbl = ib_vector_get(vec, i);
if (strcmp(pbl->name, name) == 0) {
return(pbl);
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment