mirror of
https://github.com/MariaDB/server.git
synced 2025-01-26 00:34:18 +01:00
64c95da886
creating indexes. Lock the user table inside the user transaction. enum trx_dict_op: Remove TRX_OP_INDEX_MAY_WAIT. ha_innobase::add_index(): Lock the user tables within prebuilt->trx. Commit the data dictionary transaction before creating indexes. ha_innobase::final_drop_index(): Lock the user table within prebuilt->trx.
205 lines
4.9 KiB
Text
205 lines
4.9 KiB
Text
/******************************************************
|
|
The transaction
|
|
|
|
(c) 1996 Innobase Oy
|
|
|
|
Created 3/26/1996 Heikki Tuuri
|
|
*******************************************************/
|
|
|
|
/*****************************************************************
|
|
Starts the transaction if it is not yet started. */
|
|
UNIV_INLINE
|
|
void
|
|
trx_start_if_not_started(
|
|
/*=====================*/
|
|
trx_t* trx) /* in: transaction */
|
|
{
|
|
ut_ad(trx->conc_state != TRX_COMMITTED_IN_MEMORY);
|
|
|
|
if (trx->conc_state == TRX_NOT_STARTED) {
|
|
|
|
trx_start(trx, ULINT_UNDEFINED);
|
|
}
|
|
}
|
|
|
|
/*****************************************************************
|
|
Starts the transaction if it is not yet started. Assumes we have reserved
|
|
the kernel mutex! */
|
|
UNIV_INLINE
|
|
void
|
|
trx_start_if_not_started_low(
|
|
/*=========================*/
|
|
trx_t* trx) /* in: transaction */
|
|
{
|
|
ut_ad(trx->conc_state != TRX_COMMITTED_IN_MEMORY);
|
|
|
|
if (trx->conc_state == TRX_NOT_STARTED) {
|
|
|
|
trx_start_low(trx, ULINT_UNDEFINED);
|
|
}
|
|
}
|
|
|
|
/*****************************************************************
|
|
Resets the new record lock info in a transaction struct. */
|
|
UNIV_INLINE
|
|
void
|
|
trx_reset_new_rec_lock_info(
|
|
/*========================*/
|
|
trx_t* trx) /* in: transaction struct */
|
|
{
|
|
trx->new_rec_locks[0] = NULL;
|
|
trx->new_rec_locks[1] = NULL;
|
|
}
|
|
|
|
/*****************************************************************
|
|
Registers that we have set a new record lock on an index. We only have space
|
|
to store 2 indexes! If this is called to store more than 2 indexes after
|
|
trx_reset_new_rec_lock_info(), then this function does nothing. */
|
|
UNIV_INLINE
|
|
void
|
|
trx_register_new_rec_lock(
|
|
/*======================*/
|
|
trx_t* trx, /* in: transaction struct */
|
|
dict_index_t* index) /* in: trx sets a new record lock on this
|
|
index */
|
|
{
|
|
if (trx->new_rec_locks[0] == NULL) {
|
|
trx->new_rec_locks[0] = index;
|
|
|
|
return;
|
|
}
|
|
|
|
if (trx->new_rec_locks[0] == index) {
|
|
|
|
return;
|
|
}
|
|
|
|
if (trx->new_rec_locks[1] != NULL) {
|
|
|
|
return;
|
|
}
|
|
|
|
trx->new_rec_locks[1] = index;
|
|
}
|
|
|
|
/*****************************************************************
|
|
Checks if trx has set a new record lock on an index. */
|
|
UNIV_INLINE
|
|
ibool
|
|
trx_new_rec_locks_contain(
|
|
/*======================*/
|
|
/* out: TRUE if trx has set a new record lock
|
|
on index */
|
|
trx_t* trx, /* in: transaction struct */
|
|
dict_index_t* index) /* in: index */
|
|
{
|
|
return(trx->new_rec_locks[0] == index
|
|
|| trx->new_rec_locks[1] == index);
|
|
}
|
|
|
|
/********************************************************************
|
|
Retrieves the error_info field from a trx. */
|
|
UNIV_INLINE
|
|
const dict_index_t*
|
|
trx_get_error_info(
|
|
/*===============*/
|
|
/* out: the error info */
|
|
const trx_t* trx) /* in: trx object */
|
|
{
|
|
return(trx->error_info);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Retrieves transacion's id, represented as unsigned long long. */
|
|
UNIV_INLINE
|
|
ullint
|
|
trx_get_id(
|
|
/*=======*/
|
|
/* out: transaction's id */
|
|
const trx_t* trx) /* in: transaction */
|
|
{
|
|
return((ullint)ut_conv_dulint_to_longlong(trx->id));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Retrieves transaction's que state in a human readable string. The string
|
|
should not be free()'d or modified. */
|
|
UNIV_INLINE
|
|
const char*
|
|
trx_get_que_state_str(
|
|
/*==================*/
|
|
/* out: string in the data segment */
|
|
const trx_t* trx) /* in: transaction */
|
|
{
|
|
/* be sure to adjust TRX_QUE_STATE_STR_MAX_LEN if you change this */
|
|
switch (trx->que_state) {
|
|
case TRX_QUE_RUNNING:
|
|
return("RUNNING");
|
|
case TRX_QUE_LOCK_WAIT:
|
|
return("LOCK WAIT");
|
|
case TRX_QUE_ROLLING_BACK:
|
|
return("ROLLING BACK");
|
|
case TRX_QUE_COMMITTING:
|
|
return("COMMITTING");
|
|
default:
|
|
return("UNKNOWN");
|
|
}
|
|
}
|
|
|
|
/**************************************************************************
|
|
Determine if a transaction is a dictionary operation. */
|
|
UNIV_INLINE
|
|
enum trx_dict_op
|
|
trx_get_dict_operation(
|
|
/*===================*/
|
|
/* out: dictionary operation mode */
|
|
const trx_t* trx) /* in: transaction */
|
|
{
|
|
enum trx_dict_op op = (enum trx_dict_op) trx->dict_operation;
|
|
|
|
#ifdef UNIV_DEBUG
|
|
switch (op) {
|
|
case TRX_DICT_OP_NONE:
|
|
case TRX_DICT_OP_TABLE:
|
|
case TRX_DICT_OP_INDEX:
|
|
return(op);
|
|
}
|
|
ut_error;
|
|
#endif /* UNIV_DEBUG */
|
|
return((enum trx_dict_op) UNIV_EXPECT(op, TRX_DICT_OP_NONE));
|
|
}
|
|
/**************************************************************************
|
|
Flag a transaction a dictionary operation. */
|
|
UNIV_INLINE
|
|
void
|
|
trx_set_dict_operation(
|
|
/*===================*/
|
|
trx_t* trx, /* in/out: transaction */
|
|
enum trx_dict_op op) /* in: operation, not
|
|
TRX_DICT_OP_NONE */
|
|
{
|
|
#ifdef UNIV_DEBUG
|
|
enum trx_dict_op old_op = trx_get_dict_operation(trx);
|
|
|
|
switch (op) {
|
|
case TRX_DICT_OP_NONE:
|
|
ut_error;
|
|
break;
|
|
case TRX_DICT_OP_TABLE:
|
|
switch (old_op) {
|
|
case TRX_DICT_OP_NONE:
|
|
case TRX_DICT_OP_INDEX:
|
|
case TRX_DICT_OP_TABLE:
|
|
goto ok;
|
|
}
|
|
ut_error;
|
|
break;
|
|
case TRX_DICT_OP_INDEX:
|
|
ut_ad(old_op == TRX_DICT_OP_NONE);
|
|
break;
|
|
}
|
|
ok:
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
trx->dict_operation = op;
|
|
}
|