fis-gtm/sr_port/t_write_map.c

126 lines
4.6 KiB
C

/****************************************************************
* *
* Copyright 2001, 2012 Fidelity Information Services, Inc *
* *
* This source code contains the intellectual property *
* of its copyright holder(s), and is made available *
* under a license. If you do not know the terms of *
* the license, please stop and do not read further. *
* *
****************************************************************/
#include "mdef.h"
#include "gdsroot.h"
#include "gdskill.h"
#include "gdsblk.h"
#include "gdsbt.h"
#include "gtm_facility.h"
#include "fileinfo.h"
#include "gdsfhead.h"
#include "gdscc.h"
#include "gdsbml.h"
#include "filestruct.h"
#include "jnl.h"
#include "buddy_list.h" /* needed for tp.h */
#include "hashtab_int4.h" /* needed for tp.h */
#include "tp.h"
#include "t_write_map.h"
#include "min_max.h"
#include "jnl_get_checksum.h"
GBLREF cw_set_element cw_set[];
GBLREF unsigned char cw_set_depth;
GBLREF sgmnt_addrs *cs_addrs;
GBLREF sgm_info *sgm_info_ptr;
GBLREF uint4 dollar_tlevel;
#ifdef GTM_TRUNCATE
GBLREF unsigned int t_tries;
#endif
void t_write_map (
srch_blk_status *blkhist, /* Search History of the block to be written. Currently the
* following members in this structure are used by "t_write_map"
* "blk_num" --> Block number being modified
* "buffaddr" --> Address of before image of the block
* "cr" --> cache-record that holds the block (BG only)
* "cycle" --> cycle when block was read by t_qread (BG only)
* "cr->ondsk_blkver" --> Actual block version on disk
*/
unsigned char *upd_addr, /* Address of the update array containing list of blocks to be cleared in bitmap */
trans_num tn, /* Transaction Number when this block was read. Used for cdb_sc_blkmod validation */
int4 reference_cnt) /* Same meaning as cse->reference_cnt (see gdscc.h for comments) */
{
cw_set_element *cs;
cache_rec_ptr_t cr;
jnl_buffer_ptr_t jbbp; /* jbbp is non-NULL only if before-image journaling */
sgmnt_addrs *csa;
blk_hdr_ptr_t old_block;
unsigned int bsiz;
block_id blkid;
uint4 *updptr;
csa = cs_addrs;
if (!dollar_tlevel)
{
assert(cw_set_depth < CDB_CW_SET_SIZE);
cs = &cw_set[cw_set_depth];
cs->mode = gds_t_noop; /* initialize it to a value that is not "gds_t_committed" before incrementing
* cw_set_depth as secshr_db_clnup relies on it */
cw_set_depth++;
} else
{
tp_cw_list(&cs);
sgm_info_ptr->cw_set_depth++;
}
cs->mode = gds_t_writemap;
cs->blk_checksum = 0;
cs->blk = blkhist->blk_num;
assert((cs->blk < csa->ti->total_blks) GTM_TRUNCATE_ONLY(|| (CDB_STAGNATE > t_tries)));
cs->old_block = blkhist->buffaddr;
cs->was_free = FALSE; /* t_write_map operates on BUSY blocks and hence cs->was_free is set to FALSE unconditionally */
old_block = (blk_hdr_ptr_t)cs->old_block;
assert(NULL != old_block);
jbbp = (JNL_ENABLED(csa) && csa->jnl_before_image) ? csa->jnl->jnl_buff : NULL;
if ((NULL != jbbp) && (old_block->tn < jbbp->epoch_tn))
{ /* Pre-compute CHECKSUM. Since we dont necessarily hold crit at this point, ensure we never try to
* access the buffer more than the db blk_size.
*/
bsiz = MIN(old_block->bsiz, csa->hdr->blk_size);
cs->blk_checksum = jnl_get_checksum((uint4*)old_block, csa, bsiz);
}
cs->cycle = blkhist->cycle;
cr = blkhist->cr;
cs->cr = cr;
/* the buffer in shared memory holding the GDS block contents currently does not have in its block header the
* on-disk format of that block. if it had, we could have easily copied that over to the cw-set-element.
* until then, we have to use the cache-record's field "ondsk_blkver". but the cache-record is available only in BG.
* thankfully, in MM, we do not allow GDSV4 type blocks, so we can safely assign GDSV5 (or GDSVCURR) to this field.
*/
assert((NULL != cr) || (dba_mm == csa->hdr->acc_meth));
cs->ondsk_blkver = (NULL == cr) ? GDSVCURR : cr->ondsk_blkver;
cs->ins_off = 0;
cs->index = 0;
assert(reference_cnt < csa->hdr->bplmap); /* Cannot allocate more blocks than a bitmap holds */
assert(reference_cnt > -csa->hdr->bplmap); /* Cannot free more blocks than a bitmap holds */
cs->reference_cnt = reference_cnt;
cs->jnl_freeaddr = 0; /* reset jnl_freeaddr that previous transaction might have filled in */
cs->upd_addr = upd_addr;
DEBUG_ONLY(
if (reference_cnt < 0)
reference_cnt = -reference_cnt;
/* Check that all block numbers are relative to the bitmap block number (i.e. bit number) */
updptr = (uint4 *)upd_addr;
while (reference_cnt--)
{
blkid = *updptr;
assert((int4)blkid < csa->hdr->bplmap);
updptr++;
}
)
cs->tn = tn;
cs->level = LCL_MAP_LEVL;
cs->done = FALSE;
cs->write_type = GDS_WRITE_PLAIN;
}