sp_rcontext.h 5.22 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
/* -*- C++ -*- */
/* 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 */

#ifndef _SP_RCONTEXT_H_
#define _SP_RCONTEXT_H_

21
#ifdef USE_PRAGMA_INTERFACE
22 23 24 25
#pragma interface			/* gcc class implementation */
#endif

struct sp_cond_type;
unknown's avatar
unknown committed
26
class sp_cursor;
27
struct sp_pvar;
28
class sp_lex_keeper;
29
class sp_instr_cpush;
30 31 32 33 34 35 36 37 38 39 40 41 42 43

#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;

44 45 46 47 48 49 50 51 52 53

/*
  This is a run context? of one SP ?
  THis is 
   - a stack of cursors? 
   - a stack of handlers?
   - a stack of Items ?
   - a stack of instruction locations in SP?
*/

54 55 56 57 58 59 60
class sp_rcontext : public Sql_alloc
{
  sp_rcontext(const sp_rcontext &); /* Prevent use of these */
  void operator=(sp_rcontext &);

 public:

61
  bool in_handler;
62 63 64 65 66 67 68
  /*
    Arena used to (re) allocate items on . E.g. reallocate INOUT/OUT
    SP parameters when they don't fit into prealloced items. This
    is common situation with String items. It is used mainly in
    sp_eval_func_item().
  */
  Query_arena *callers_arena;
69

70
  sp_rcontext(uint fsize, uint hmax, uint cmax);
71 72 73 74 75

  ~sp_rcontext()
  {
    // Not needed?
    //sql_element_free(m_frame);
76
    //m_saved.empty();
77 78 79 80 81
  }

  inline void
  push_item(Item *i)
  {
82
    if (m_count < m_fsize)
83
      m_frame[m_count++]= i;
84 85 86 87 88 89
  }

  inline void
  set_item(uint idx, Item *i)
  {
    if (idx < m_count)
90
      m_frame[idx]= i;
91 92
  }

93 94
  /* Returns 0 on success, -1 on (eval) failure */
  int
95
  set_item_eval(THD *thd, uint idx, Item **i, enum_field_types type);
96

97 98 99 100 101 102
  inline Item *
  get_item(uint idx)
  {
    return m_frame[idx];
  }

103 104 105 106 107 108 109
  inline Item **
  get_item_addr(uint idx)
  {
    return m_frame + idx;
  }


110 111 112 113 114 115 116 117 118 119 120 121
  inline void
  set_result(Item *it)
  {
    m_result= it;
  }

  inline Item *
  get_result()
  {
    return m_result;
  }

122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
  inline void
  push_handler(struct sp_cond_type *cond, uint h, int type, uint f)
  {
    m_handler[m_hcount].cond= cond;
    m_handler[m_hcount].handler= h;
    m_handler[m_hcount].type= type;
    m_handler[m_hcount].foffset= f;
    m_hcount+= 1;
  }

  inline void
  pop_handlers(uint count)
  {
    m_hcount-= count;
  }

  // Returns 1 if a handler was found, 0 otherwise.
139
  bool
unknown's avatar
unknown committed
140
  find_handler(uint sql_errno,MYSQL_ERROR::enum_warning_level level);
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179

  // Returns handler type and sets *ip to location if one was found
  inline int
  found_handler(uint *ip, uint *fp)
  {
    if (m_hfound < 0)
      return SP_HANDLER_NONE;
    *ip= m_handler[m_hfound].handler;
    *fp= m_handler[m_hfound].foffset;
    return m_handler[m_hfound].type;
  }

  // Clears the handler find state
  inline void
  clear_handler()
  {
    m_hfound= -1;
  }

  inline void
  push_hstack(uint h)
  {
    m_hstack[m_hsp++]= h;
  }

  inline uint
  pop_hstack()
  {
    return m_hstack[--m_hsp];
  }

  // Save variables starting at fp and up
  void
  save_variables(uint fp);

  // Restore variables down to fp
  void
  restore_variables(uint fp);

180
  void
181
  push_cursor(sp_lex_keeper *lex_keeper, sp_instr_cpush *i);
182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197

  void
  pop_cursors(uint count);

  void
  pop_all_cursors()
  {
    pop_cursors(m_ccount);
  }

  inline sp_cursor *
  get_cursor(uint i)
  {
    return m_cstack[i];
  }

198 199 200
private:

  uint m_count;
201
  uint m_fsize;
202
  Item **m_frame;
203

204
  Item *m_result;		// For FUNCTIONs
205

206 207 208 209 210 211
  sp_handler_t *m_handler;
  uint m_hcount;
  uint *m_hstack;
  uint m_hsp;
  int m_hfound;			// Set by find_handler; -1 if not found
  List<Item> m_saved;		// Saved variables
212

213 214 215
  sp_cursor **m_cstack;
  uint m_ccount;

216 217
}; // class sp_rcontext : public Sql_alloc

218 219 220 221 222

class sp_cursor : public Sql_alloc
{
public:

223
  sp_cursor(sp_lex_keeper *lex_keeper, sp_instr_cpush *i);
224 225 226 227 228 229 230 231

  virtual ~sp_cursor()
  {
    destroy();
  }

  // We have split this in two to make it easy for sp_instr_copen
  // to reuse the sp_instr::exec_stmt() code.
232
  sp_lex_keeper *
233 234
  pre_open(THD *thd);
  void
235
  post_open(THD *thd, my_bool was_opened);
236 237 238 239 240 241 242 243 244 245 246 247 248

  int
  close(THD *thd);

  inline my_bool
  is_open()
  {
    return m_isopen;
  }

  int
  fetch(THD *, List<struct sp_pvar> *vars);

249 250 251 252 253 254
  inline sp_instr_cpush *
  get_instr()
  {
    return m_i;
  }
  
255 256 257
private:

  MEM_ROOT m_mem_root;		// My own mem_root
258
  sp_lex_keeper *m_lex_keeper;
259 260
  Protocol_cursor *m_prot;
  my_bool m_isopen;
261
  my_bool m_nseof;		// Original no_send_eof
262 263
  Protocol *m_oprot;		// Original protcol
  MYSQL_ROWS *m_current_row;
264
  sp_instr_cpush *m_i;		// My push instruction
265 266 267 268 269 270
  
  void
  destroy();

}; // class sp_cursor : public Sql_alloc

271
#endif /* _SP_RCONTEXT_H_ */