2012-02-05 11:35:58 -05:00
|
|
|
/****************************************************************
|
|
|
|
* *
|
2012-03-24 14:06:46 -04:00
|
|
|
* Copyright 2003, 2011 Fidelity Information Services, Inc *
|
2012-02-05 11:35:58 -05:00
|
|
|
* *
|
|
|
|
* 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 "gtm_string.h"
|
|
|
|
#if defined(UNIX)
|
|
|
|
#include "gtm_aio.h"
|
|
|
|
#elif defined(VMS)
|
|
|
|
#include "iosb_disk.h"
|
|
|
|
#include "iosp.h" /* for SS_NORMAL */
|
|
|
|
#endif
|
|
|
|
#include "gdsroot.h"
|
|
|
|
#include "gdsbt.h"
|
|
|
|
#include "gtm_facility.h"
|
|
|
|
#include "fileinfo.h"
|
|
|
|
#include "gdsblk.h"
|
|
|
|
#include "gdsfhead.h"
|
|
|
|
#include "filestruct.h"
|
|
|
|
#include "jnl.h"
|
|
|
|
#include "buddy_list.h"
|
|
|
|
#include "hashtab_int4.h" /* needed for muprec.h */
|
|
|
|
#include "hashtab_int8.h" /* needed for muprec.h */
|
|
|
|
#include "hashtab_mname.h" /* needed for muprec.h */
|
|
|
|
#include "hashtab.h"
|
|
|
|
#include "muprec.h"
|
|
|
|
#include "mur_read_file.h"
|
|
|
|
|
|
|
|
GBLREF jnl_gbls_t jgbl;
|
|
|
|
GBLREF mur_gbls_t murgbl;
|
|
|
|
GBLREF void (*call_on_signal)();
|
|
|
|
GBLREF unsigned char t_fail_hist[CDB_MAX_TRIES];
|
2012-03-24 14:06:46 -04:00
|
|
|
GBLREF boolean_t gv_play_duplicate_kills;
|
2012-02-05 11:35:58 -05:00
|
|
|
|
|
|
|
#define FREE_AND_NULLIFY_BUDDY_LIST(LIST) \
|
|
|
|
{ \
|
|
|
|
assert(NULL != LIST); \
|
|
|
|
if (NULL != LIST) \
|
|
|
|
{ \
|
|
|
|
FREEUP_BUDDY_LIST(LIST); \
|
|
|
|
LIST = NULL; \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define FREE_AND_NULLIFY(PTR) \
|
|
|
|
{ \
|
|
|
|
assert(NULL != PTR); \
|
|
|
|
if (NULL != PTR) \
|
|
|
|
{ \
|
|
|
|
free(PTR); \
|
|
|
|
PTR = NULL; \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************************
|
|
|
|
* Function Name: mur_init() *
|
|
|
|
* Input: None *
|
|
|
|
* Output: None *
|
2012-03-24 14:06:46 -04:00
|
|
|
****************************************************************************************/
|
2012-02-05 11:35:58 -05:00
|
|
|
|
|
|
|
void mur_init(void)
|
|
|
|
{
|
|
|
|
int index;
|
|
|
|
|
|
|
|
murgbl.multi_list = (buddy_list *)malloc(SIZEOF(buddy_list));
|
|
|
|
initialize_list(murgbl.multi_list, SIZEOF(multi_struct), MUR_MULTI_LIST_INIT_ALLOC);
|
|
|
|
murgbl.forw_multi_list = (buddy_list *)malloc(SIZEOF(buddy_list));
|
|
|
|
initialize_list(murgbl.forw_multi_list, SIZEOF(forw_multi_struct), MUR_MULTI_LIST_INIT_ALLOC);
|
|
|
|
init_hashtab_int8(&murgbl.token_table, MUR_MULTI_HASHTABLE_INIT_ELEMS, HASHTAB_COMPACT, HASHTAB_SPARE_TABLE);
|
|
|
|
init_hashtab_int8(&murgbl.forw_token_table, MUR_MULTI_HASHTABLE_INIT_ELEMS, HASHTAB_COMPACT, HASHTAB_SPARE_TABLE);
|
|
|
|
murgbl.pini_buddy_list = (buddy_list *)malloc(SIZEOF(buddy_list));
|
|
|
|
initialize_list(murgbl.pini_buddy_list, SIZEOF(pini_list_struct), MUR_PINI_LIST_INIT_ELEMS);
|
|
|
|
/* pini_list hash table of a jnl_ctl_list is initialized in mur_fopen */
|
|
|
|
/* FAB of a jnl_ctl_list is initialized in mur_fopen */
|
|
|
|
murgbl.resync_seqno = 0; /* interrupted rollback set this to non-zero value later */
|
|
|
|
jgbl.mupip_journal = TRUE; /* this is a MUPIP JOURNAL command */
|
2012-03-24 14:06:46 -04:00
|
|
|
/* Write journal records for KILL even if node does not exist in this database. This way if any process had
|
|
|
|
* written a KILL record for a no-op kill (a kill of a node that does not exist; this is currently possible in
|
|
|
|
* the update process), recovery does the exact same thing and thereby keeps the db/jnl in sync.
|
2012-02-05 11:35:58 -05:00
|
|
|
*/
|
2012-03-24 14:06:46 -04:00
|
|
|
gv_play_duplicate_kills = TRUE;
|
2012-02-05 11:35:58 -05:00
|
|
|
/* Because journal recovery operates with no one else touching the database, t_tries is going to be
|
|
|
|
* directly set to CDB_STAGNATE for every TP transaction (in op_tstart). So initialize
|
|
|
|
* t_fail_hist[0] through t_fail_hist[CDB_STAGNATE-1] to '0' for display purposes.
|
|
|
|
*/
|
|
|
|
for (index = 0; index < CDB_STAGNATE; index++)
|
|
|
|
t_fail_hist[index] = '0';
|
|
|
|
call_on_signal = mur_close_files;
|
|
|
|
DEBUG_ONLY(assert_jrec_member_offsets();)
|
|
|
|
}
|
|
|
|
|
|
|
|
void mur_free(void)
|
|
|
|
{
|
|
|
|
FREE_AND_NULLIFY_BUDDY_LIST(murgbl.multi_list);
|
|
|
|
FREE_AND_NULLIFY_BUDDY_LIST(murgbl.forw_multi_list);
|
|
|
|
FREE_AND_NULLIFY_BUDDY_LIST(murgbl.pini_buddy_list);
|
|
|
|
/* Note : In addition, there is a lot of mallocs done in mur_open_files (of rctls and jctls) that need to be freed.
|
|
|
|
* It is not considered as important since MUPIP is a one-command image activation so no memory leak issues.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
void mur_rctl_desc_alloc(reg_ctl_list *rctl)
|
|
|
|
{
|
|
|
|
mur_read_desc_t *mur_desc;
|
|
|
|
|
|
|
|
mur_desc = (mur_read_desc_t *)malloc(SIZEOF(mur_read_desc_t));
|
|
|
|
/*
|
|
|
|
* Layout of rctl->mur_desc buffers (consecutive buffers are contiguous)
|
|
|
|
*
|
|
|
|
* |<--MUR_BUFF_SIZE-->|<--MUR_BUFF_SIZE-->|<--MUR_BUFF_SIZE-->|<--MUR_BUFF_SIZE-->|<--MUR_BUFF_SIZE-->|
|
|
|
|
* | random_buff | aux_buff1 | seq_buff[0] | seq_buff[1] | aux_buff2 |
|
|
|
|
*/
|
|
|
|
assert(MUR_BUFF_SIZE > MAX_JNL_REC_SIZE); /* in order for a journal record to fit in one buffer */
|
|
|
|
mur_desc->alloc_base = malloc(5 * MUR_BUFF_SIZE); /* two for double buffering, two for auxiliary, one for random */
|
|
|
|
mur_desc->random_buff.base = mur_desc->alloc_base;
|
|
|
|
mur_desc->random_buff.top = mur_desc->aux_buff1 = mur_desc->alloc_base + MUR_BUFF_SIZE;
|
|
|
|
mur_desc->seq_buff[0].base = mur_desc->aux_buff1 + MUR_BUFF_SIZE;
|
|
|
|
mur_desc->seq_buff[0].top = mur_desc->seq_buff[1].base = mur_desc->seq_buff[0].base + MUR_BUFF_SIZE;
|
|
|
|
mur_desc->seq_buff[1].top = mur_desc->aux_buff2.base = mur_desc->seq_buff[1].base + MUR_BUFF_SIZE;
|
|
|
|
mur_desc->aux_buff2.top = mur_desc->seq_buff[1].top + MUR_BUFF_SIZE;
|
|
|
|
mur_desc->random_buff.read_in_progress = FALSE;
|
|
|
|
mur_desc->seq_buff[0].read_in_progress = FALSE;
|
|
|
|
mur_desc->seq_buff[1].read_in_progress = FALSE;
|
|
|
|
mur_desc->aux_buff2.read_in_progress = FALSE;
|
|
|
|
# if defined(UNIX) && defined(MUR_USE_AIO)
|
|
|
|
mur_desc->seq_buff[0].aiocbp = (struct aiocb *)malloc(SIZEOF(struct aiocb));
|
|
|
|
mur_desc->seq_buff[1].aiocbp = (struct aiocb *)malloc(SIZEOF(struct aiocb));
|
|
|
|
memset((char *)mur_desc->seq_buff[0].aiocbp, 0, SIZEOF(struct aiocb));
|
|
|
|
memset((char *)mur_desc->seq_buff[1].aiocbp, 0, SIZEOF(struct aiocb));
|
|
|
|
mur_desc->aux_buff2.aiocbp = (struct aiocb *)NULL; /* no aio for this buffer */
|
|
|
|
mur_desc->random_buff.aiocbp = (struct aiocb *)NULL; /* no aio for this buffer */
|
|
|
|
# elif defined(VMS)
|
|
|
|
mur_desc->random_buff.iosb.cond = SS_NORMAL;
|
|
|
|
mur_desc->seq_buff[0].iosb.cond = SS_NORMAL;
|
|
|
|
mur_desc->seq_buff[1].iosb.cond = SS_NORMAL;
|
|
|
|
# endif
|
|
|
|
rctl->mur_desc = mur_desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mur_rctl_desc_free(reg_ctl_list *rctl)
|
|
|
|
{
|
|
|
|
int index;
|
|
|
|
mur_read_desc_t *mur_desc;
|
|
|
|
|
|
|
|
mur_desc = rctl->mur_desc;
|
|
|
|
FREE_AND_NULLIFY(mur_desc->alloc_base);
|
|
|
|
# if defined(UNIX) && defined(MUR_USE_AIO)
|
|
|
|
for (index = 0; index <= 1; index++)
|
|
|
|
FREE_AND_NULLIFY(mur_desc->seq_buff[index].aiocbp);
|
|
|
|
# endif
|
|
|
|
FREE_AND_NULLIFY(rctl->mur_desc); /* cannot use mur_desc since we want rctl->mur_desc (not mur_desc) to be NULLified */
|
|
|
|
}
|