126 lines
4.6 KiB
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;
|
|
}
|