1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 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 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
|
/*****************************************************************************
Copyright (c) 1997, 2016, Oracle and/or its affiliates. All Rights Reserved.
Copyright (c) 2018, 2022, 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/read0types.h
Cursor read
Created 2/16/1997 Heikki Tuuri
*******************************************************/
#pragma once
#include "dict0mem.h"
#include "trx0types.h"
#include "srw_lock.h"
#include <algorithm>
/**
Read view lists the trx ids of those transactions for which a consistent read
should not see the modifications to the database.
*/
class ReadViewBase
{
/**
The read should not see any transaction with trx id >= this value.
In other words, this is the "high water mark".
*/
trx_id_t m_low_limit_id= 0;
/**
The read should see all trx ids which are strictly
smaller (<) than this value. In other words, this is the
low water mark".
*/
trx_id_t m_up_limit_id;
/** Set of RW transactions that was active when this snapshot was taken */
trx_ids_t m_ids;
/**
The view does not need to see the undo logs for transactions whose
transaction number is strictly smaller (<) than this value: they can be
removed in purge if not needed by other views.
*/
trx_id_t m_low_limit_no;
protected:
bool empty() { return m_ids.empty(); }
/** @return the up limit id */
trx_id_t up_limit_id() const { return m_up_limit_id; }
public:
/**
Append state from another view.
This method is used to find min(m_low_limit_no), min(m_low_limit_id) and
all transaction ids below min(m_low_limit_id). These values effectively
form oldest view.
@param other view to copy from
*/
void append(const ReadViewBase &other)
{
ut_ad(&other != this);
if (m_low_limit_no > other.m_low_limit_no)
m_low_limit_no= other.m_low_limit_no;
if (m_low_limit_id > other.m_low_limit_id)
m_low_limit_id= other.m_low_limit_id;
trx_ids_t::iterator dst= m_ids.begin();
for (const trx_id_t id : other.m_ids)
{
if (id >= m_low_limit_id)
break;
loop:
if (dst == m_ids.end())
{
m_ids.push_back(id);
dst= m_ids.end();
continue;
}
if (*dst < id)
{
dst++;
goto loop;
}
else if (*dst > id)
dst= m_ids.insert(dst, id) + 1;
}
m_ids.erase(std::lower_bound(dst, m_ids.end(), m_low_limit_id),
m_ids.end());
m_up_limit_id= m_ids.empty() ? m_low_limit_id : m_ids.front();
ut_ad(m_up_limit_id <= m_low_limit_id);
}
/**
Creates a snapshot where exactly the transactions serialized before this
point in time are seen in the view.
@param[in,out] trx transaction
*/
inline void snapshot(trx_t *trx);
/**
Check whether the changes by id are visible.
@param[in] id transaction id to check against the view
@return whether the view sees the modifications of id.
*/
bool changes_visible(trx_id_t id) const
MY_ATTRIBUTE((warn_unused_result))
{
if (id >= m_low_limit_id)
return false;
return id < m_up_limit_id ||
m_ids.empty() ||
!std::binary_search(m_ids.begin(), m_ids.end(), id);
}
/**
@param id transaction to check
@return true if view sees transaction id
*/
bool sees(trx_id_t id) const { return id < m_up_limit_id; }
/** @return the low limit no */
trx_id_t low_limit_no() const { return m_low_limit_no; }
/** @return the low limit id */
trx_id_t low_limit_id() const { return m_low_limit_id; }
/** Clamp the low limit id for purge_sys.end_view */
void clamp_low_limit_id(trx_id_t limit)
{
if (m_low_limit_id > limit)
m_low_limit_id= limit;
}
};
/** A ReadView with extra members required for trx_t::read_view. */
class ReadView: public ReadViewBase
{
/**
View state.
Implemented as atomic to allow mutex-free view close and re-use.
Non-owner thread is allowed to call is_open() alone without mutex
protection as well. E.g. trx_sys.view_count() does this.
If non-owner thread intends to access other members as well, both
is_open() and other members accesses must be protected by m_mutex.
E.g. copy_to().
*/
std::atomic<bool> m_open;
/** For synchronisation with purge coordinator. */
mutable srw_mutex m_mutex;
/**
trx id of creating transaction.
Used exclusively by the read view owner thread.
*/
trx_id_t m_creator_trx_id;
public:
ReadView()
{
memset(reinterpret_cast<void*>(this), 0, sizeof *this);
m_mutex.init();
}
~ReadView() { m_mutex.destroy(); }
/**
Opens a read view where exactly the transactions serialized before this
point in time are seen in the view.
View becomes visible to purge thread. Intended to be called by the ReadView
owner thread.
@param[in,out] trx transaction
*/
void open(trx_t *trx);
/**
Closes the view.
View becomes not visible to purge thread. Intended to be called by the
ReadView owner thread.
*/
void close() { m_open.store(false, std::memory_order_relaxed); }
/** Returns true if view is open. */
bool is_open() const { return m_open.load(std::memory_order_relaxed); }
/**
Sets the creator transaction id.
This should be set only for views created by RW transactions.
Intended to be called by the ReadView owner thread.
*/
void set_creator_trx_id(trx_id_t id)
{
ut_ad(m_creator_trx_id == 0);
m_creator_trx_id= id;
}
/**
Writes the limits to the file.
@param file file to write to
*/
void print_limits(FILE *file) const
{
m_mutex.wr_lock();
if (is_open())
fprintf(file, "Trx read view will not see trx with"
" id >= " TRX_ID_FMT ", sees < " TRX_ID_FMT "\n",
low_limit_id(), up_limit_id());
m_mutex.wr_unlock();
}
/**
A wrapper around ReadViewBase::changes_visible().
Intended to be called by the ReadView owner thread.
*/
bool changes_visible(trx_id_t id) const
{ return id == m_creator_trx_id || ReadViewBase::changes_visible(id); }
/**
A wrapper around ReadViewBase::append().
Intended to be called by the purge coordinator task.
*/
void append_to(ReadViewBase *to) const
{
m_mutex.wr_lock();
if (is_open())
to->append(*this);
m_mutex.wr_unlock();
}
/**
Declare the object mostly unaccessible.
*/
void mem_noaccess() const
{
MEM_NOACCESS(&m_open, sizeof m_open);
/* m_mutex is accessed via trx_sys.rw_trx_hash */
MEM_NOACCESS(&m_creator_trx_id, sizeof m_creator_trx_id);
}
};
|