mirror of
https://github.com/MariaDB/server.git
synced 2025-01-31 02:51:44 +01:00
ea37b14409
This is joint work with Thirunarayanan Balathandayuthapani.
The MDL interface between InnoDB and the rest of the server
(in storage/innobase/dict/dict0dict.cc and in include/)
is my work, while most everything else is Thiru's.
The collection of InnoDB persistent statistics and the
defragmentation were not refactored to use MDL. They will
keep relying on lower-level interlocking with
fil_check_pending_operations().
The purge of transaction history and the background operations on
fulltext indexes will use MDL. We will revert
commit 2c4844c9e7
(MDEV-17813) because thanks to MDL, purge cannot conflict
with DDL operations anymore. For a similar reason, we will remove
the MDEV-16222 test case from gcol.innodb_virtual_debug_purge.
Purge is essentially replacing all use of the global dict_sys.latch
with MDL. Purge will skip the undo log records for tables whose names
start with #sql-ib or #sql2. Theoretically, such tables might
be renamed back to visible table names if TRUNCATE fails to
create a new table, or the final rename in ALTER TABLE...ALGORITHM=COPY
fails. In that case, purge could permanently leave some garbage
in the table. Such garbage will be tolerated; the table would not
be considered corrupted.
To avoid repeated MDL releases and acquisitions,
trx_purge_attach_undo_recs() will sort undo log records by table_id,
and purge_node_t will keep the MDL and table handle open for multiple
successive undo log records.
get_purge_table(): A new accessor, used during the purge of
history for indexed virtual columns. This interface should ideally
not exist at all.
thd_mdl_context(): Accessor of THD::mdl_context.
Wrapped in a new thd_mdl_service.
dict_get_db_name_len(): Define inline.
dict_acquire_mdl_shared(): Acquire explicit shared MDL on a table name
if needed.
dict_table_open_on_id(): Return MDL_ticket, if requested.
dict_table_close(): Release MDL ticket, if requested.
dict_fts_index_syncing(), dict_index_t::index_fts_syncing: Remove.
row_drop_table_for_mysql() no longer needs to check these, because
MDL guarantees that a fulltext index sync will not be in progress
while MDL_EXCLUSIVE is protecting a DDL operation.
dict_table_t::parse_name(): Parse the table name for acquiring MDL.
purge_node_t::undo_recs: Change the type to std::list<trx_purge_rec_t*>
(different container, and storing also roll_ptr).
purge_node_t: Add mdl_ticket, last_table_id, purge_thd, mdl_hold_recs
for acquiring MDL and for keeping the table open across multiple
undo log records.
purge_vcol_info_t, row_purge_store_vsec_cur(), row_purge_restore_vsec_cur():
Remove. We will acquire the MDL earlier.
purge_sys_t::heap: Added, for reading undo log records.
fts_sync_during_ddl(): Invoked during ALGORITHM=INPLACE operations
to ensure that fts_sync_table() will not conflict with MDL_EXCLUSIVE.
Uses fts_t::sync_message for bookkeeping.
266 lines
8.4 KiB
C++
266 lines
8.4 KiB
C++
/*****************************************************************************
|
|
|
|
Copyright (c) 1996, 2016, Oracle and/or its affiliates. All Rights Reserved.
|
|
Copyright (c) 2017, 2019, MariaDB Corporation.
|
|
|
|
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, Fifth Floor, Boston, MA 02110-1335 USA
|
|
|
|
*****************************************************************************/
|
|
|
|
/**************************************************//**
|
|
@file include/trx0purge.h
|
|
Purge old versions
|
|
|
|
Created 3/26/1996 Heikki Tuuri
|
|
*******************************************************/
|
|
|
|
#ifndef trx0purge_h
|
|
#define trx0purge_h
|
|
|
|
#include "trx0rseg.h"
|
|
#include "que0types.h"
|
|
|
|
#include <queue>
|
|
|
|
/** A dummy undo record used as a return value when we have a whole undo log
|
|
which needs no purge */
|
|
extern trx_undo_rec_t trx_purge_dummy_rec;
|
|
|
|
/********************************************************************//**
|
|
Calculates the file address of an undo log header when we have the file
|
|
address of its history list node.
|
|
@return file address of the log */
|
|
UNIV_INLINE
|
|
fil_addr_t
|
|
trx_purge_get_log_from_hist(
|
|
/*========================*/
|
|
fil_addr_t node_addr); /*!< in: file address of the history
|
|
list node of the log */
|
|
/** Prepend the history list with an undo log.
|
|
Remove the undo log segment from the rseg slot if it is too big for reuse.
|
|
@param[in] trx transaction
|
|
@param[in,out] undo undo log
|
|
@param[in,out] mtr mini-transaction */
|
|
void
|
|
trx_purge_add_undo_to_history(const trx_t* trx, trx_undo_t*& undo, mtr_t* mtr);
|
|
/**
|
|
Run a purge batch.
|
|
@param n_tasks number of purge tasks to submit to the queue
|
|
@param truncate whether to truncate the history at the end of the batch
|
|
@return number of undo log pages handled in the batch */
|
|
ulint trx_purge(ulint n_tasks, bool truncate);
|
|
|
|
/** Rollback segements from a given transaction with trx-no
|
|
scheduled for purge. */
|
|
class TrxUndoRsegs {
|
|
private:
|
|
typedef std::vector<trx_rseg_t*, ut_allocator<trx_rseg_t*> >
|
|
trx_rsegs_t;
|
|
public:
|
|
typedef trx_rsegs_t::iterator iterator;
|
|
typedef trx_rsegs_t::const_iterator const_iterator;
|
|
|
|
/** Default constructor */
|
|
TrxUndoRsegs() {}
|
|
/** Constructor */
|
|
TrxUndoRsegs(trx_rseg_t& rseg)
|
|
: m_commit(rseg.last_commit), m_rsegs(1, &rseg) {}
|
|
/** Constructor */
|
|
TrxUndoRsegs(trx_id_t trx_no, trx_rseg_t& rseg)
|
|
: m_commit(trx_no << 1), m_rsegs(1, &rseg) {}
|
|
|
|
/** @return the transaction commit identifier */
|
|
trx_id_t trx_no() const { return m_commit >> 1; }
|
|
|
|
bool operator!=(const TrxUndoRsegs& other) const
|
|
{ return m_commit != other.m_commit; }
|
|
bool empty() const { return m_rsegs.empty(); }
|
|
void erase(iterator& it) { m_rsegs.erase(it); }
|
|
iterator begin() { return(m_rsegs.begin()); }
|
|
iterator end() { return(m_rsegs.end()); }
|
|
const_iterator begin() const { return m_rsegs.begin(); }
|
|
const_iterator end() const { return m_rsegs.end(); }
|
|
|
|
/** Compare two TrxUndoRsegs based on trx_no.
|
|
@param elem1 first element to compare
|
|
@param elem2 second element to compare
|
|
@return true if elem1 > elem2 else false.*/
|
|
bool operator()(const TrxUndoRsegs& lhs, const TrxUndoRsegs& rhs)
|
|
{
|
|
return(lhs.m_commit > rhs.m_commit);
|
|
}
|
|
|
|
private:
|
|
/** Copy trx_rseg_t::last_commit */
|
|
trx_id_t m_commit;
|
|
/** Rollback segments of a transaction, scheduled for purge. */
|
|
trx_rsegs_t m_rsegs;
|
|
};
|
|
|
|
typedef std::priority_queue<
|
|
TrxUndoRsegs,
|
|
std::vector<TrxUndoRsegs, ut_allocator<TrxUndoRsegs> >,
|
|
TrxUndoRsegs> purge_pq_t;
|
|
|
|
/** Chooses the rollback segment with the oldest committed transaction */
|
|
struct TrxUndoRsegsIterator {
|
|
/** Constructor */
|
|
TrxUndoRsegsIterator();
|
|
/** Sets the next rseg to purge in purge_sys.
|
|
Executed in the purge coordinator thread.
|
|
@return whether anything is to be purged */
|
|
inline bool set_next();
|
|
|
|
private:
|
|
// Disable copying
|
|
TrxUndoRsegsIterator(const TrxUndoRsegsIterator&);
|
|
TrxUndoRsegsIterator& operator=(const TrxUndoRsegsIterator&);
|
|
|
|
/** The current element to process */
|
|
TrxUndoRsegs m_rsegs;
|
|
/** Track the current element in m_rsegs */
|
|
TrxUndoRsegs::const_iterator m_iter;
|
|
};
|
|
|
|
/** The control structure used in the purge operation */
|
|
class purge_sys_t
|
|
{
|
|
public:
|
|
/** latch protecting view, m_enabled */
|
|
MY_ALIGNED(CACHE_LINE_SIZE)
|
|
rw_lock_t latch;
|
|
private:
|
|
/** whether purge is enabled; protected by latch and std::atomic */
|
|
std::atomic<bool> m_enabled;
|
|
/** number of pending stop() calls without resume() */
|
|
Atomic_counter<int32_t> m_paused;
|
|
public:
|
|
que_t* query; /*!< The query graph which will do the
|
|
parallelized purge operation */
|
|
MY_ALIGNED(CACHE_LINE_SIZE)
|
|
ReadView view; /*!< The purge will not remove undo logs
|
|
which are >= this view (purge view) */
|
|
|
|
/** Iterator to the undo log records of committed transactions */
|
|
struct iterator
|
|
{
|
|
bool operator<=(const iterator& other) const
|
|
{
|
|
if (commit < other.commit) return true;
|
|
if (commit > other.commit) return false;
|
|
return undo_no <= other.undo_no;
|
|
}
|
|
|
|
/** @return the commit number of the transaction */
|
|
trx_id_t trx_no() const { return commit >> 1; }
|
|
void reset_trx_no(trx_id_t trx_no) { commit = trx_no << 1; }
|
|
|
|
/** 2 * trx_t::no + old_insert of the committed transaction */
|
|
trx_id_t commit;
|
|
/** The record number within the committed transaction's undo
|
|
log, increasing, purged from from 0 onwards */
|
|
undo_no_t undo_no;
|
|
};
|
|
|
|
/** The tail of the purge queue; the last parsed undo log of a
|
|
committed transaction. */
|
|
iterator tail;
|
|
/** The head of the purge queue; any older undo logs of committed
|
|
transactions may be discarded (history list truncation). */
|
|
iterator head;
|
|
/*-----------------------------*/
|
|
bool next_stored; /*!< whether rseg holds the next record
|
|
to purge */
|
|
trx_rseg_t* rseg; /*!< Rollback segment for the next undo
|
|
record to purge */
|
|
uint32_t page_no; /*!< Page number for the next undo
|
|
record to purge, page number of the
|
|
log header, if dummy record */
|
|
uint32_t hdr_page_no; /*!< Header page of the undo log where
|
|
the next record to purge belongs */
|
|
uint16_t offset; /*!< Page offset for the next undo
|
|
record to purge, 0 if the dummy
|
|
record */
|
|
uint16_t hdr_offset; /*!< Header byte offset on the page */
|
|
|
|
|
|
TrxUndoRsegsIterator
|
|
rseg_iter; /*!< Iterator to get the next rseg
|
|
to process */
|
|
|
|
purge_pq_t purge_queue; /*!< Binary min-heap, ordered on
|
|
TrxUndoRsegs::trx_no. It is protected
|
|
by the pq_mutex */
|
|
PQMutex pq_mutex; /*!< Mutex protecting purge_queue */
|
|
|
|
/** Undo tablespace file truncation (only accessed by the
|
|
srv_purge_coordinator_thread) */
|
|
struct {
|
|
/** The undo tablespace that is currently being truncated */
|
|
fil_space_t* current;
|
|
/** The undo tablespace that was last truncated */
|
|
fil_space_t* last;
|
|
} truncate;
|
|
|
|
/** Heap for reading the undo log records */
|
|
mem_heap_t* heap;
|
|
/**
|
|
Constructor.
|
|
|
|
Some members may require late initialisation, thus we just mark object as
|
|
uninitialised. Real initialisation happens in create().
|
|
*/
|
|
|
|
purge_sys_t(): m_enabled(false), heap(nullptr) {}
|
|
|
|
/** Create the instance */
|
|
void create();
|
|
|
|
/** Close the purge system on shutdown */
|
|
void close();
|
|
|
|
/** @return whether purge is enabled */
|
|
bool enabled() { return m_enabled.load(std::memory_order_relaxed); }
|
|
/** @return whether the purge coordinator is paused */
|
|
bool paused()
|
|
{ return m_paused != 0; }
|
|
|
|
/** Enable purge at startup. Not protected by latch; the main thread
|
|
will wait for purge_sys.enabled() in srv_start() */
|
|
void coordinator_startup()
|
|
{
|
|
ut_ad(!enabled());
|
|
m_enabled.store(true, std::memory_order_relaxed);
|
|
}
|
|
|
|
/** Disable purge at shutdown */
|
|
void coordinator_shutdown()
|
|
{
|
|
ut_ad(enabled());
|
|
m_enabled.store(false, std::memory_order_relaxed);
|
|
}
|
|
|
|
/** @return whether the purge coordinator thread is active */
|
|
bool running();
|
|
/** Stop purge during FLUSH TABLES FOR EXPORT */
|
|
void stop();
|
|
/** Resume purge at UNLOCK TABLES after FLUSH TABLES FOR EXPORT */
|
|
void resume();
|
|
};
|
|
|
|
/** The global data structure coordinating a purge */
|
|
extern purge_sys_t purge_sys;
|
|
|
|
#include "trx0purge.ic"
|
|
|
|
#endif /* trx0purge_h */
|