8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-23 07:23:04 +01:00

Avoid replication of empty (read-only) transactions. Fixed memory leak. Misc corrections.

This commit is contained in:
Dmitry Yemanov 2020-08-23 18:16:52 +03:00
parent 46570bce9a
commit 46284645e5
4 changed files with 55 additions and 39 deletions

View File

@ -78,6 +78,7 @@ namespace Replication
}
Firebird::UCharBuffer* getBuffer();
void releaseBuffer(Firebird::UCharBuffer* buffer);
void flush(Firebird::UCharBuffer* buffer, bool sync);
@ -94,7 +95,6 @@ namespace Replication
private:
void logError(const Firebird::IStatus* status);
void releaseBuffer(Firebird::UCharBuffer* buffer);
void bgWriter();

View File

@ -49,7 +49,7 @@ namespace Replication
const ULONG BLOCK_BEGIN_TRANS = 1;
const ULONG BLOCK_END_TRANS = 2;
enum Operation
enum Operation: UCHAR
{
opStartTransaction = 1,
opPrepareTransaction = 2,

View File

@ -119,20 +119,6 @@ void Replicator::postError(const Exception& ex)
newErrors.copyTo(&m_status);
}
// IDisposable implementation
void Replicator::dispose()
{
try
{
delete this;
}
catch (const Exception& ex)
{
postError(ex);
}
}
// IReplicatedSession implementation
IReplicatedTransaction* Replicator::startTransaction(SINT64 number)
@ -194,6 +180,14 @@ bool Replicator::commitTransaction(Transaction* transaction)
auto& txnData = transaction->getData();
// Do not replicate this transaction if it's de-facto read-only.
// If there were no flushes yet and the buffer contains just one tag
// (this should be opStartTransaction), it means nothing was changed.
const auto dataLength = txnData.buffer->getCount() - sizeof(Block);
if (txnData.flushes || dataLength > sizeof(UCHAR))
{
for (const auto& generator : m_generators)
{
fb_assert(generator.name.hasData());
@ -207,12 +201,11 @@ bool Replicator::commitTransaction(Transaction* transaction)
txnData.putTag(opCommitTransaction);
flush(txnData, FLUSH_SYNC, BLOCK_END_TRANS);
FB_SIZE_T pos;
if (m_transactions.find(transaction, pos))
m_transactions.remove(pos);
transaction->dispose();
}
else
{
fb_assert((*txnData.buffer)[sizeof(Block)] == opStartTransaction);
}
}
catch (const Exception& ex)
{
@ -220,6 +213,7 @@ bool Replicator::commitTransaction(Transaction* transaction)
return false;
}
transaction->dispose();
return true;
}
@ -236,12 +230,10 @@ bool Replicator::rollbackTransaction(Transaction* transaction)
txnData.putTag(opRollbackTransaction);
flush(txnData, FLUSH_SYNC, BLOCK_END_TRANS);
}
FB_SIZE_T pos;
if (m_transactions.find(transaction, pos))
m_transactions.remove(pos);
transaction->dispose();
else
{
fb_assert((*txnData.buffer)[sizeof(Block)] == opStartTransaction);
}
}
catch (const Exception& ex)
{
@ -249,9 +241,27 @@ bool Replicator::rollbackTransaction(Transaction* transaction)
return false;
}
transaction->dispose();
return true;
}
void Replicator::releaseTransaction(Transaction* transaction)
{
try
{
MutexLockGuard guard(m_mutex, FB_FUNCTION);
auto& txnData = transaction->getData();
m_manager->releaseBuffer(txnData.buffer);
FB_SIZE_T pos;
if (m_transactions.find(transaction, pos))
m_transactions.remove(pos);
}
catch (const Exception& ex)
{} // no-op
}
bool Replicator::startSavepoint(Transaction* transaction)
{
try

View File

@ -133,6 +133,7 @@ namespace Replication
void dispose()
{
m_replicator->releaseTransaction(this);
delete this;
}
@ -226,7 +227,11 @@ namespace Replication
bool cleanupTransactions);
// IDisposable methods
void dispose();
void dispose()
{
delete this;
}
// IReplicatedSession methods
@ -257,6 +262,7 @@ namespace Replication
bool prepareTransaction(Transaction* transaction);
bool commitTransaction(Transaction* transaction);
bool rollbackTransaction(Transaction* transaction);
void releaseTransaction(Transaction* transaction);
bool startSavepoint(Transaction* transaction);
bool releaseSavepoint(Transaction* transaction);