From ed3287c70e8d4d469a7656a75ec000ce8929f211 Mon Sep 17 00:00:00 2001 From: Ilya Eremin Date: Tue, 29 Aug 2023 15:34:55 +0300 Subject: [PATCH] Fix cases where the precedence relationship between a record page and a blob page is not set 1. While performing the intermediate garbage collection. 2. While modifying a blob field of a record. --- src/jrd/blb.cpp | 1 + src/jrd/vio.cpp | 25 ++++++++++++++----------- 2 files changed, 15 insertions(+), 11 deletions(-) diff --git a/src/jrd/blb.cpp b/src/jrd/blb.cpp index dbb461aa4b..c0c247c8c8 100644 --- a/src/jrd/blb.cpp +++ b/src/jrd/blb.cpp @@ -448,6 +448,7 @@ void BLB_garbage_collect(thread_db* tdbb, **************************************/ SET_TDBB(tdbb); + fb_assert(prior_page > 0); RecordBitmap bmGoing; ULONG cntGoing = 0; diff --git a/src/jrd/vio.cpp b/src/jrd/vio.cpp index 93e7f5822a..fbb719e2df 100644 --- a/src/jrd/vio.cpp +++ b/src/jrd/vio.cpp @@ -2485,25 +2485,26 @@ static void delete_version_chain(thread_db* tdbb, record_param* rpb, bool delete ULONG prior_page = 0; - if (!delete_head) + // Note that the page number of the oldest version in the chain should + // be stored in rpb->rpb_page before exiting this function because + // VIO_intermediate_gc will use it as a prior page number. + while (rpb->rpb_b_page != 0 || delete_head) { - prior_page = rpb->rpb_page; - rpb->rpb_page = rpb->rpb_b_page; - rpb->rpb_line = rpb->rpb_b_line; - } + if (!delete_head) + { + prior_page = rpb->rpb_page; + rpb->rpb_page = rpb->rpb_b_page; + rpb->rpb_line = rpb->rpb_b_line; + } + else + delete_head = false; - while (rpb->rpb_page != 0) - { if (!DPM_fetch(tdbb, rpb, LCK_write)) BUGCHECK(291); // msg 291 cannot find record back version record_param temp_rpb = *rpb; DPM_delete(tdbb, &temp_rpb, prior_page); delete_tail(tdbb, &temp_rpb, temp_rpb.rpb_page); - - prior_page = rpb->rpb_page; - rpb->rpb_page = rpb->rpb_b_page; - rpb->rpb_line = rpb->rpb_b_line; } } @@ -3678,6 +3679,8 @@ bool VIO_modify(thread_db* tdbb, record_param* org_rpb, record_param* new_rpb, j org_rpb->rpb_flags &= ~(rpb_delta | rpb_uk_modified); org_rpb->rpb_flags |= new_rpb->rpb_flags & (rpb_delta | rpb_uk_modified); + stack.merge(new_rpb->rpb_record->getPrecedence()); + replace_record(tdbb, org_rpb, &stack, transaction); if (!(transaction->tra_flags & TRA_system) &&