mirror of
https://github.com/MariaDB/server.git
synced 2025-01-19 13:32:33 +01:00
726 lines
21 KiB
C++
726 lines
21 KiB
C++
/*****************************************************************************
|
|
|
|
Copyright (c) 1996, 2011, Oracle and/or its affiliates. All Rights Reserved.
|
|
|
|
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; version 2 of the License.
|
|
|
|
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.,
|
|
51 Franklin Street, Suite 500, Boston, MA 02110-1335 USA
|
|
|
|
*****************************************************************************/
|
|
|
|
/**************************************************//**
|
|
@file read/read0read.cc
|
|
Cursor read
|
|
|
|
Created 2/16/1997 Heikki Tuuri
|
|
*******************************************************/
|
|
|
|
#include "read0read.h"
|
|
|
|
#ifdef UNIV_NONINL
|
|
#include "read0read.ic"
|
|
#endif
|
|
|
|
#include "srv0srv.h"
|
|
#include "trx0sys.h"
|
|
|
|
/*
|
|
-------------------------------------------------------------------------------
|
|
FACT A: Cursor read view on a secondary index sees only committed versions
|
|
-------
|
|
of the records in the secondary index or those versions of rows created
|
|
by transaction which created a cursor before cursor was created even
|
|
if transaction which created the cursor has changed that clustered index page.
|
|
|
|
PROOF: We must show that read goes always to the clustered index record
|
|
to see that record is visible in the cursor read view. Consider e.g.
|
|
following table and SQL-clauses:
|
|
|
|
create table t1(a int not null, b int, primary key(a), index(b));
|
|
insert into t1 values (1,1),(2,2);
|
|
commit;
|
|
|
|
Now consider that we have a cursor for a query
|
|
|
|
select b from t1 where b >= 1;
|
|
|
|
This query will use secondary key on the table t1. Now after the first fetch
|
|
on this cursor if we do a update:
|
|
|
|
update t1 set b = 5 where b = 2;
|
|
|
|
Now second fetch of the cursor should not see record (2,5) instead it should
|
|
see record (2,2).
|
|
|
|
We also should show that if we have delete t1 where b = 5; we still
|
|
can see record (2,2).
|
|
|
|
When we access a secondary key record maximum transaction id is fetched
|
|
from this record and this trx_id is compared to up_limit_id in the view.
|
|
If trx_id in the record is greater or equal than up_limit_id in the view
|
|
cluster record is accessed. Because trx_id of the creating
|
|
transaction is stored when this view was created to the list of
|
|
trx_ids not seen by this read view previous version of the
|
|
record is requested to be built. This is build using clustered record.
|
|
If the secondary key record is delete-marked, its corresponding
|
|
clustered record can be already be purged only if records
|
|
trx_id < low_limit_no. Purge can't remove any record deleted by a
|
|
transaction which was active when cursor was created. But, we still
|
|
may have a deleted secondary key record but no clustered record. But,
|
|
this is not a problem because this case is handled in
|
|
row_sel_get_clust_rec() function which is called
|
|
whenever we note that this read view does not see trx_id in the
|
|
record. Thus, we see correct version. Q. E. D.
|
|
|
|
-------------------------------------------------------------------------------
|
|
FACT B: Cursor read view on a clustered index sees only committed versions
|
|
-------
|
|
of the records in the clustered index or those versions of rows created
|
|
by transaction which created a cursor before cursor was created even
|
|
if transaction which created the cursor has changed that clustered index page.
|
|
|
|
PROOF: Consider e.g.following table and SQL-clauses:
|
|
|
|
create table t1(a int not null, b int, primary key(a));
|
|
insert into t1 values (1),(2);
|
|
commit;
|
|
|
|
Now consider that we have a cursor for a query
|
|
|
|
select a from t1 where a >= 1;
|
|
|
|
This query will use clustered key on the table t1. Now after the first fetch
|
|
on this cursor if we do a update:
|
|
|
|
update t1 set a = 5 where a = 2;
|
|
|
|
Now second fetch of the cursor should not see record (5) instead it should
|
|
see record (2).
|
|
|
|
We also should show that if we have execute delete t1 where a = 5; after
|
|
the cursor is opened we still can see record (2).
|
|
|
|
When accessing clustered record we always check if this read view sees
|
|
trx_id stored to clustered record. By default we don't see any changes
|
|
if record trx_id >= low_limit_id i.e. change was made transaction
|
|
which started after transaction which created the cursor. If row
|
|
was changed by the future transaction a previous version of the
|
|
clustered record is created. Thus we see only committed version in
|
|
this case. We see all changes made by committed transactions i.e.
|
|
record trx_id < up_limit_id. In this case we don't need to do anything,
|
|
we already see correct version of the record. We don't see any changes
|
|
made by active transaction except creating transaction. We have stored
|
|
trx_id of creating transaction to list of trx_ids when this view was
|
|
created. Thus we can easily see if this record was changed by the
|
|
creating transaction. Because we already have clustered record we can
|
|
access roll_ptr. Using this roll_ptr we can fetch undo record.
|
|
We can now check that undo_no of the undo record is less than undo_no of the
|
|
trancaction which created a view when cursor was created. We see this
|
|
clustered record only in case when record undo_no is less than undo_no
|
|
in the view. If this is not true we build based on undo_rec previous
|
|
version of the record. This record is found because purge can't remove
|
|
records accessed by active transaction. Thus we see correct version. Q. E. D.
|
|
-------------------------------------------------------------------------------
|
|
FACT C: Purge does not remove any delete-marked row that is visible
|
|
-------
|
|
in any cursor read view.
|
|
|
|
PROOF: We know that:
|
|
1: Currently active read views in trx_sys_t::view_list are ordered by
|
|
read_view_t::low_limit_no in descending order, that is,
|
|
newest read view first.
|
|
|
|
2: Purge clones the oldest read view and uses that to determine whether there
|
|
are any active transactions that can see the to be purged records.
|
|
|
|
Therefore any joining or active transaction will not have a view older
|
|
than the purge view, according to 1.
|
|
|
|
When purge needs to remove a delete-marked row from a secondary index,
|
|
it will first check that the DB_TRX_ID value of the corresponding
|
|
record in the clustered index is older than the purge view. It will
|
|
also check if there is a newer version of the row (clustered index
|
|
record) that is not delete-marked in the secondary index. If such a
|
|
row exists and is collation-equal to the delete-marked secondary index
|
|
record then purge will not remove the secondary index record.
|
|
|
|
Delete-marked clustered index records will be removed by
|
|
row_purge_remove_clust_if_poss(), unless the clustered index record
|
|
(and its DB_ROLL_PTR) has been updated. Every new version of the
|
|
clustered index record will update DB_ROLL_PTR, pointing to a new UNDO
|
|
log entry that allows the old version to be reconstructed. The
|
|
DB_ROLL_PTR in the oldest remaining version in the old-version chain
|
|
may be pointing to garbage (an undo log record discarded by purge),
|
|
but it will never be dereferenced, because the purge view is older
|
|
than any active transaction.
|
|
|
|
For details see: row_vers_old_has_index_entry() and row_purge_poss_sec()
|
|
|
|
Some additional issues:
|
|
|
|
What if trx_sys->view_list == NULL and some transaction T1 and Purge both
|
|
try to open read_view at same time. Only one can acquire trx_sys->mutex.
|
|
In which order will the views be opened? Should it matter? If no, why?
|
|
|
|
The order does not matter. No new transactions can be created and no running
|
|
transaction can commit or rollback (or free views).
|
|
*/
|
|
|
|
#ifdef UNIV_DEBUG
|
|
/*********************************************************************//**
|
|
Validates a read view object. */
|
|
static
|
|
ibool
|
|
read_view_validate(
|
|
/*===============*/
|
|
const read_view_t* view) /*!< in: view to validate */
|
|
{
|
|
ulint i;
|
|
|
|
ut_ad(mutex_own(&trx_sys->mutex));
|
|
|
|
/* Check that the view->trx_ids array is in descending order. */
|
|
for (i = 1; i < view->n_trx_ids; ++i) {
|
|
|
|
ut_a(view->trx_ids[i] < view->trx_ids[i - 1]);
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/** Functor to validate the view list. */
|
|
struct Check {
|
|
|
|
Check() : m_prev_view(0) { }
|
|
|
|
void operator()(const read_view_t* view)
|
|
{
|
|
ut_a(m_prev_view == NULL
|
|
|| m_prev_view->low_limit_no >= view->low_limit_no);
|
|
|
|
m_prev_view = view;
|
|
}
|
|
|
|
const read_view_t* m_prev_view;
|
|
};
|
|
|
|
/*********************************************************************//**
|
|
Validates a read view list. */
|
|
static
|
|
ibool
|
|
read_view_list_validate(void)
|
|
/*=========================*/
|
|
{
|
|
ut_ad(mutex_own(&trx_sys->mutex));
|
|
|
|
ut_list_map(trx_sys->view_list, &read_view_t::view_list, Check());
|
|
|
|
return(TRUE);
|
|
}
|
|
#endif
|
|
|
|
/*********************************************************************//**
|
|
Creates a read view object.
|
|
@return own: read view struct */
|
|
UNIV_INLINE
|
|
read_view_t*
|
|
read_view_create_low(
|
|
/*=================*/
|
|
ulint n, /*!< in: number of cells in the trx_ids array */
|
|
mem_heap_t* heap) /*!< in: memory heap from which allocated */
|
|
{
|
|
read_view_t* view;
|
|
|
|
view = static_cast<read_view_t*>(
|
|
mem_heap_alloc(
|
|
heap, sizeof(*view) + n * sizeof(*view->trx_ids)));
|
|
|
|
view->n_trx_ids = n;
|
|
view->trx_ids = (trx_id_t*) &view[1];
|
|
|
|
return(view);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Clones a read view object. This function will allocate space for two read
|
|
views contiguously, one identical in size and content as @param view (starting
|
|
at returned pointer) and another view immediately following the trx_ids array.
|
|
The second view will have space for an extra trx_id_t element.
|
|
@return read view struct */
|
|
UNIV_INLINE
|
|
read_view_t*
|
|
read_view_clone(
|
|
/*============*/
|
|
const read_view_t* view, /*!< in: view to clone */
|
|
mem_heap_t* heap) /*!< in: memory heap
|
|
from which allocated */
|
|
{
|
|
ulint sz;
|
|
read_view_t* clone;
|
|
read_view_t* new_view;
|
|
|
|
ut_ad(mutex_own(&trx_sys->mutex));
|
|
|
|
/* Allocate space for two views. */
|
|
|
|
sz = sizeof(*view) + view->n_trx_ids * sizeof(*view->trx_ids);
|
|
|
|
/* Add an extra trx_id_t slot for the new view. */
|
|
|
|
clone = static_cast<read_view_t*>(
|
|
mem_heap_alloc(heap, (sz * 2) + sizeof(trx_id_t)));
|
|
|
|
/* Only the contents of the old view are important, the new view
|
|
will be created from this and so we don't copy that across. */
|
|
|
|
memcpy(clone, view, sz);
|
|
|
|
clone->trx_ids = (trx_id_t*) &clone[1];
|
|
|
|
new_view = (read_view_t*) &clone->trx_ids[clone->n_trx_ids];
|
|
new_view->trx_ids = (trx_id_t*) &new_view[1];
|
|
new_view->n_trx_ids = clone->n_trx_ids + 1;
|
|
|
|
ut_a(new_view->n_trx_ids == view->n_trx_ids + 1);
|
|
|
|
return(clone);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Insert the view in the proper order into the trx_sys->view_list. The
|
|
read view list is ordered by read_view_t::low_limit_no in descending order. */
|
|
static
|
|
void
|
|
read_view_add(
|
|
/*==========*/
|
|
read_view_t* view) /*!< in: view to add to */
|
|
{
|
|
read_view_t* elem;
|
|
read_view_t* prev_elem;
|
|
|
|
ut_ad(mutex_own(&trx_sys->mutex));
|
|
ut_ad(read_view_validate(view));
|
|
|
|
/* Find the correct slot for insertion. */
|
|
for (elem = UT_LIST_GET_FIRST(trx_sys->view_list), prev_elem = NULL;
|
|
elem != NULL && view->low_limit_no < elem->low_limit_no;
|
|
prev_elem = elem, elem = UT_LIST_GET_NEXT(view_list, elem)) {
|
|
/* No op */
|
|
}
|
|
|
|
if (prev_elem == NULL) {
|
|
UT_LIST_ADD_FIRST(view_list, trx_sys->view_list, view);
|
|
} else {
|
|
UT_LIST_INSERT_AFTER(
|
|
view_list, trx_sys->view_list, prev_elem, view);
|
|
}
|
|
|
|
ut_ad(read_view_list_validate());
|
|
}
|
|
|
|
/** Functor to create thew view trx_ids array. */
|
|
struct CreateView {
|
|
|
|
CreateView(read_view_t* view)
|
|
: m_view(view)
|
|
{
|
|
m_n_trx = m_view->n_trx_ids;
|
|
m_view->n_trx_ids = 0;
|
|
}
|
|
|
|
void operator()(const trx_t* trx)
|
|
{
|
|
ut_ad(mutex_own(&trx_sys->mutex));
|
|
ut_ad(trx->in_rw_trx_list);
|
|
|
|
/* trx->state cannot change from or to NOT_STARTED
|
|
while we are holding the trx_sys->mutex. It may change
|
|
from ACTIVE to PREPARED or COMMITTED. */
|
|
|
|
if (trx->id != m_view->creator_trx_id
|
|
&& !trx_state_eq(trx, TRX_STATE_COMMITTED_IN_MEMORY)) {
|
|
|
|
ut_ad(m_n_trx > m_view->n_trx_ids);
|
|
|
|
m_view->trx_ids[m_view->n_trx_ids++] = trx->id;
|
|
|
|
/* NOTE that a transaction whose trx number is <
|
|
trx_sys->max_trx_id can still be active, if it is
|
|
in the middle of its commit! Note that when a
|
|
transaction starts, we initialize trx->no to
|
|
IB_ULONGLONG_MAX. */
|
|
|
|
/* trx->no is protected by trx_sys->mutex, which
|
|
we are holding. It is assigned by trx_commit()
|
|
before lock_trx_release_locks() assigns
|
|
trx->state = TRX_STATE_COMMITTED_IN_MEMORY. */
|
|
|
|
if (m_view->low_limit_no > trx->no) {
|
|
m_view->low_limit_no = trx->no;
|
|
}
|
|
}
|
|
}
|
|
|
|
read_view_t* m_view;
|
|
ulint m_n_trx;
|
|
};
|
|
|
|
/*********************************************************************//**
|
|
Opens a read view where exactly the transactions serialized before this
|
|
point in time are seen in the view.
|
|
@return own: read view struct */
|
|
static
|
|
read_view_t*
|
|
read_view_open_now_low(
|
|
/*===================*/
|
|
trx_id_t cr_trx_id, /*!< in: trx_id of creating
|
|
transaction, or 0 used in purge */
|
|
mem_heap_t* heap) /*!< in: memory heap from which
|
|
allocated */
|
|
{
|
|
read_view_t* view;
|
|
ulint n_trx = UT_LIST_GET_LEN(trx_sys->rw_trx_list);
|
|
|
|
ut_ad(mutex_own(&trx_sys->mutex));
|
|
|
|
view = read_view_create_low(n_trx, heap);
|
|
|
|
view->undo_no = 0;
|
|
view->type = VIEW_NORMAL;
|
|
view->creator_trx_id = cr_trx_id;
|
|
|
|
/* No future transactions should be visible in the view */
|
|
|
|
view->low_limit_no = trx_sys->max_trx_id;
|
|
view->low_limit_id = view->low_limit_no;
|
|
|
|
/* No active transaction should be visible, except cr_trx */
|
|
|
|
ut_list_map(trx_sys->rw_trx_list, &trx_t::trx_list, CreateView(view));
|
|
|
|
if (view->n_trx_ids > 0) {
|
|
/* The last active transaction has the smallest id: */
|
|
view->up_limit_id = view->trx_ids[view->n_trx_ids - 1];
|
|
} else {
|
|
view->up_limit_id = view->low_limit_id;
|
|
}
|
|
|
|
/* Purge views are not added to the view list. */
|
|
if (cr_trx_id > 0) {
|
|
read_view_add(view);
|
|
}
|
|
|
|
return(view);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Opens a read view where exactly the transactions serialized before this
|
|
point in time are seen in the view.
|
|
@return own: read view struct */
|
|
UNIV_INTERN
|
|
read_view_t*
|
|
read_view_open_now(
|
|
/*===============*/
|
|
trx_id_t cr_trx_id, /*!< in: trx_id of creating
|
|
transaction, or 0 used in purge */
|
|
mem_heap_t* heap) /*!< in: memory heap from which
|
|
allocated */
|
|
{
|
|
read_view_t* view;
|
|
|
|
mutex_enter(&trx_sys->mutex);
|
|
|
|
view = read_view_open_now_low(cr_trx_id, heap);
|
|
|
|
mutex_exit(&trx_sys->mutex);
|
|
|
|
return(view);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Makes a copy of the oldest existing read view, with the exception that also
|
|
the creating trx of the oldest view is set as not visible in the 'copied'
|
|
view. Opens a new view if no views currently exist. The view must be closed
|
|
with ..._close. This is used in purge.
|
|
@return own: read view struct */
|
|
UNIV_INTERN
|
|
read_view_t*
|
|
read_view_purge_open(
|
|
/*=================*/
|
|
mem_heap_t* heap) /*!< in: memory heap from which
|
|
allocated */
|
|
{
|
|
ulint i;
|
|
read_view_t* view;
|
|
read_view_t* oldest_view;
|
|
trx_id_t creator_trx_id;
|
|
ulint insert_done = 0;
|
|
|
|
mutex_enter(&trx_sys->mutex);
|
|
|
|
oldest_view = UT_LIST_GET_LAST(trx_sys->view_list);
|
|
|
|
if (oldest_view == NULL) {
|
|
|
|
view = read_view_open_now_low(0, heap);
|
|
|
|
mutex_exit(&trx_sys->mutex);
|
|
|
|
return(view);
|
|
}
|
|
|
|
/* Allocate space for both views, the oldest and the new purge view. */
|
|
|
|
oldest_view = read_view_clone(oldest_view, heap);
|
|
|
|
ut_ad(read_view_validate(oldest_view));
|
|
|
|
mutex_exit(&trx_sys->mutex);
|
|
|
|
ut_a(oldest_view->creator_trx_id > 0);
|
|
creator_trx_id = oldest_view->creator_trx_id;
|
|
|
|
view = (read_view_t*) &oldest_view->trx_ids[oldest_view->n_trx_ids];
|
|
|
|
/* Add the creator transaction id in the trx_ids array in the
|
|
correct slot. */
|
|
|
|
for (i = 0; i < oldest_view->n_trx_ids; ++i) {
|
|
trx_id_t id;
|
|
|
|
id = oldest_view->trx_ids[i - insert_done];
|
|
|
|
if (insert_done == 0 && creator_trx_id > id) {
|
|
id = creator_trx_id;
|
|
insert_done = 1;
|
|
}
|
|
|
|
view->trx_ids[i] = id;
|
|
}
|
|
|
|
if (insert_done == 0) {
|
|
view->trx_ids[i] = creator_trx_id;
|
|
} else {
|
|
ut_a(i > 0);
|
|
view->trx_ids[i] = oldest_view->trx_ids[i - 1];
|
|
}
|
|
|
|
view->creator_trx_id = 0;
|
|
|
|
view->low_limit_no = oldest_view->low_limit_no;
|
|
view->low_limit_id = oldest_view->low_limit_id;
|
|
|
|
if (view->n_trx_ids > 0) {
|
|
/* The last active transaction has the smallest id: */
|
|
|
|
view->up_limit_id = view->trx_ids[view->n_trx_ids - 1];
|
|
} else {
|
|
view->up_limit_id = oldest_view->up_limit_id;
|
|
}
|
|
|
|
return(view);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Remove a read view from the trx_sys->view_list. */
|
|
UNIV_INTERN
|
|
void
|
|
read_view_remove(
|
|
/*=============*/
|
|
read_view_t* view) /*!< in: read view */
|
|
{
|
|
mutex_enter(&trx_sys->mutex);
|
|
|
|
ut_ad(read_view_validate(view));
|
|
|
|
UT_LIST_REMOVE(view_list, trx_sys->view_list, view);
|
|
|
|
ut_ad(read_view_list_validate());
|
|
|
|
mutex_exit(&trx_sys->mutex);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Closes a consistent read view for MySQL. This function is called at an SQL
|
|
statement end if the trx isolation level is <= TRX_ISO_READ_COMMITTED. */
|
|
UNIV_INTERN
|
|
void
|
|
read_view_close_for_mysql(
|
|
/*======================*/
|
|
trx_t* trx) /*!< in: trx which has a read view */
|
|
{
|
|
ut_a(trx->global_read_view);
|
|
|
|
read_view_remove(trx->global_read_view);
|
|
|
|
mem_heap_empty(trx->global_read_view_heap);
|
|
|
|
trx->read_view = NULL;
|
|
trx->global_read_view = NULL;
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Prints a read view to stderr. */
|
|
UNIV_INTERN
|
|
void
|
|
read_view_print(
|
|
/*============*/
|
|
const read_view_t* view) /*!< in: read view */
|
|
{
|
|
ulint n_ids;
|
|
ulint i;
|
|
|
|
if (view->type == VIEW_HIGH_GRANULARITY) {
|
|
fprintf(stderr,
|
|
"High-granularity read view undo_n:o " TRX_ID_FMT "\n",
|
|
view->undo_no);
|
|
} else {
|
|
fprintf(stderr, "Normal read view\n");
|
|
}
|
|
|
|
fprintf(stderr, "Read view low limit trx n:o " TRX_ID_FMT "\n",
|
|
view->low_limit_no);
|
|
|
|
fprintf(stderr, "Read view up limit trx id " TRX_ID_FMT "\n",
|
|
view->up_limit_id);
|
|
|
|
fprintf(stderr, "Read view low limit trx id " TRX_ID_FMT "\n",
|
|
view->low_limit_id);
|
|
|
|
fprintf(stderr, "Read view individually stored trx ids:\n");
|
|
|
|
n_ids = view->n_trx_ids;
|
|
|
|
for (i = 0; i < n_ids; i++) {
|
|
fprintf(stderr, "Read view trx id " TRX_ID_FMT "\n",
|
|
view->trx_ids[i]);
|
|
}
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Create a high-granularity consistent cursor view for mysql to be used
|
|
in cursors. In this consistent read view modifications done by the
|
|
creating transaction after the cursor is created or future transactions
|
|
are not visible. */
|
|
UNIV_INTERN
|
|
cursor_view_t*
|
|
read_cursor_view_create_for_mysql(
|
|
/*==============================*/
|
|
trx_t* cr_trx) /*!< in: trx where cursor view is created */
|
|
{
|
|
read_view_t* view;
|
|
mem_heap_t* heap;
|
|
ulint n_trx;
|
|
cursor_view_t* curview;
|
|
|
|
/* Use larger heap than in trx_create when creating a read_view
|
|
because cursors are quite long. */
|
|
|
|
heap = mem_heap_create(512);
|
|
|
|
curview = (cursor_view_t*) mem_heap_alloc(heap, sizeof(*curview));
|
|
|
|
curview->heap = heap;
|
|
|
|
/* Drop cursor tables from consideration when evaluating the
|
|
need of auto-commit */
|
|
|
|
curview->n_mysql_tables_in_use = cr_trx->n_mysql_tables_in_use;
|
|
|
|
cr_trx->n_mysql_tables_in_use = 0;
|
|
|
|
mutex_enter(&trx_sys->mutex);
|
|
|
|
n_trx = UT_LIST_GET_LEN(trx_sys->rw_trx_list);
|
|
|
|
curview->read_view = read_view_create_low(n_trx, curview->heap);
|
|
|
|
view = curview->read_view;
|
|
view->undo_no = cr_trx->undo_no;
|
|
view->type = VIEW_HIGH_GRANULARITY;
|
|
view->creator_trx_id = UINT64_UNDEFINED;
|
|
|
|
/* No future transactions should be visible in the view */
|
|
|
|
view->low_limit_no = trx_sys->max_trx_id;
|
|
view->low_limit_id = view->low_limit_no;
|
|
|
|
/* No active transaction should be visible */
|
|
|
|
ut_list_map(trx_sys->rw_trx_list, &trx_t::trx_list, CreateView(view));
|
|
|
|
view->creator_trx_id = cr_trx->id;
|
|
|
|
if (view->n_trx_ids > 0) {
|
|
/* The last active transaction has the smallest id: */
|
|
|
|
view->up_limit_id = view->trx_ids[view->n_trx_ids - 1];
|
|
} else {
|
|
view->up_limit_id = view->low_limit_id;
|
|
}
|
|
|
|
read_view_add(view);
|
|
|
|
mutex_exit(&trx_sys->mutex);
|
|
|
|
return(curview);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
Close a given consistent cursor view for mysql and restore global read view
|
|
back to a transaction read view. */
|
|
UNIV_INTERN
|
|
void
|
|
read_cursor_view_close_for_mysql(
|
|
/*=============================*/
|
|
trx_t* trx, /*!< in: trx */
|
|
cursor_view_t* curview)/*!< in: cursor view to be closed */
|
|
{
|
|
ut_a(curview);
|
|
ut_a(curview->read_view);
|
|
ut_a(curview->heap);
|
|
|
|
/* Add cursor's tables to the global count of active tables that
|
|
belong to this transaction */
|
|
trx->n_mysql_tables_in_use += curview->n_mysql_tables_in_use;
|
|
|
|
read_view_remove(curview->read_view);
|
|
|
|
trx->read_view = trx->global_read_view;
|
|
|
|
mem_heap_free(curview->heap);
|
|
}
|
|
|
|
/*********************************************************************//**
|
|
This function sets a given consistent cursor view to a transaction
|
|
read view if given consistent cursor view is not NULL. Otherwise, function
|
|
restores a global read view to a transaction read view. */
|
|
UNIV_INTERN
|
|
void
|
|
read_cursor_set_for_mysql(
|
|
/*======================*/
|
|
trx_t* trx, /*!< in: transaction where cursor is set */
|
|
cursor_view_t* curview)/*!< in: consistent cursor view to be set */
|
|
{
|
|
ut_a(trx);
|
|
|
|
mutex_enter(&trx_sys->mutex);
|
|
|
|
if (UNIV_LIKELY(curview != NULL)) {
|
|
trx->read_view = curview->read_view;
|
|
} else {
|
|
trx->read_view = trx->global_read_view;
|
|
}
|
|
|
|
ut_ad(read_view_validate(trx->read_view));
|
|
|
|
mutex_exit(&trx_sys->mutex);
|
|
}
|