Commit 40f7819e authored by sunny's avatar sunny

branches/5.1: Fix Bug#38187 Error 153 when creating savepoints

InnoDB previously treated savepoints as a stack e.g.,
  SAVEPOINT a;
  SAVEPOINT b;
  SAVEPOINT c;
  SAVEPOINT b; <- This would delete b and c.

This fix changes the behavior to:
  SAVEPOINT a;
  SAVEPOINT b;
  SAVEPOINT c;
  SAVEPOINT b; <- Does not delete savepoint c
parent d3244010
...@@ -15,6 +15,8 @@ Created 3/26/1996 Heikki Tuuri ...@@ -15,6 +15,8 @@ Created 3/26/1996 Heikki Tuuri
#include "mtr0mtr.h" #include "mtr0mtr.h"
#include "trx0sys.h" #include "trx0sys.h"
#define trx_roll_free_all_savepoints(s) trx_roll_savepoints_free((s), NULL)
/*********************************************************************** /***********************************************************************
Returns a transaction savepoint taken at this point in time. */ Returns a transaction savepoint taken at this point in time. */
...@@ -237,7 +239,17 @@ trx_release_savepoint_for_mysql( ...@@ -237,7 +239,17 @@ trx_release_savepoint_for_mysql(
const char* savepoint_name); /* in: savepoint name */ const char* savepoint_name); /* in: savepoint name */
/*********************************************************************** /***********************************************************************
Frees savepoint structs. */ Frees a single savepoint struct. */
void
trx_roll_savepoint_free(
/*=====================*/
trx_t* trx, /* in: transaction handle */
trx_named_savept_t* savep); /* in: savepoint to free */
/***********************************************************************
Frees savepoint structs starting from savep, if savep == NULL then
free all savepoints. */
void void
trx_roll_savepoints_free( trx_roll_savepoints_free(
......
...@@ -185,7 +185,25 @@ trx_rollback_last_sql_stat_for_mysql( ...@@ -185,7 +185,25 @@ trx_rollback_last_sql_stat_for_mysql(
} }
/*********************************************************************** /***********************************************************************
Frees savepoint structs. */ Frees a single savepoint struct. */
void
trx_roll_savepoint_free(
/*=====================*/
trx_t* trx, /* in: transaction handle */
trx_named_savept_t* savep) /* in: savepoint to free */
{
ut_a(savep != NULL);
ut_a(UT_LIST_GET_LEN(trx->trx_savepoints) > 0);
UT_LIST_REMOVE(trx_savepoints, trx->trx_savepoints, savep);
mem_free(savep->name);
mem_free(savep);
}
/***********************************************************************
Frees savepoint structs starting from savep, if savep == NULL then
free all savepoints. */
void void
trx_roll_savepoints_free( trx_roll_savepoints_free(
...@@ -206,9 +224,7 @@ trx_roll_savepoints_free( ...@@ -206,9 +224,7 @@ trx_roll_savepoints_free(
while (savep != NULL) { while (savep != NULL) {
next_savep = UT_LIST_GET_NEXT(trx_savepoints, savep); next_savep = UT_LIST_GET_NEXT(trx_savepoints, savep);
UT_LIST_REMOVE(trx_savepoints, trx->trx_savepoints, savep); trx_roll_savepoint_free(trx, savep);
mem_free(savep->name);
mem_free(savep);
savep = next_savep; savep = next_savep;
} }
...@@ -343,8 +359,8 @@ trx_savepoint_for_mysql( ...@@ -343,8 +359,8 @@ trx_savepoint_for_mysql(
} }
/*********************************************************************** /***********************************************************************
Releases a named savepoint. Savepoints which Releases only the named savepoint. Savepoints which were set after this
were set after this savepoint are deleted. */ savepoint are left as is. */
ulint ulint
trx_release_savepoint_for_mysql( trx_release_savepoint_for_mysql(
...@@ -360,31 +376,16 @@ trx_release_savepoint_for_mysql( ...@@ -360,31 +376,16 @@ trx_release_savepoint_for_mysql(
savep = UT_LIST_GET_FIRST(trx->trx_savepoints); savep = UT_LIST_GET_FIRST(trx->trx_savepoints);
/* Search for the savepoint by name and free if found. */
while (savep != NULL) { while (savep != NULL) {
if (0 == ut_strcmp(savep->name, savepoint_name)) { if (0 == ut_strcmp(savep->name, savepoint_name)) {
/* Found */ trx_roll_savepoint_free(trx, savep);
break; return(DB_SUCCESS);
} }
savep = UT_LIST_GET_NEXT(trx_savepoints, savep); savep = UT_LIST_GET_NEXT(trx_savepoints, savep);
} }
if (savep == NULL) { return(DB_NO_SAVEPOINT);
return(DB_NO_SAVEPOINT);
}
/* We can now free all savepoints strictly later than this one */
trx_roll_savepoints_free(trx, savep);
/* Now we can free this savepoint too */
UT_LIST_REMOVE(trx_savepoints, trx->trx_savepoints, savep);
mem_free(savep->name);
mem_free(savep);
return(DB_SUCCESS);
} }
/*********************************************************************** /***********************************************************************
......
...@@ -930,8 +930,8 @@ trx_commit_off_kernel( ...@@ -930,8 +930,8 @@ trx_commit_off_kernel(
mutex_enter(&kernel_mutex); mutex_enter(&kernel_mutex);
} }
/* Free savepoints */ /* Free all savepoints */
trx_roll_savepoints_free(trx, NULL); trx_roll_free_all_savepoints(trx);
trx->conc_state = TRX_NOT_STARTED; trx->conc_state = TRX_NOT_STARTED;
trx->rseg = NULL; trx->rseg = NULL;
......
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