160 lines
5.8 KiB
C
160 lines
5.8 KiB
C
|
/****************************************************************
|
||
|
* *
|
||
|
* Copyright 2001, 2011 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 "rtnhdr.h"
|
||
|
#include "mv_stent.h"
|
||
|
#include "stack_frame.h"
|
||
|
#include "tp_frame.h"
|
||
|
#include "gtm_string.h"
|
||
|
#include "gtm_newintrinsic.h"
|
||
|
#include "op.h"
|
||
|
|
||
|
GBLREF mv_stent *mv_chain;
|
||
|
GBLREF unsigned char *stackbase, *stacktop, *msp, *stackwarn;
|
||
|
GBLREF stack_frame *frame_pointer;
|
||
|
GBLREF tp_frame *tp_pointer;
|
||
|
GBLREF symval *curr_symval;
|
||
|
GBLREF uint4 dollar_tlevel;
|
||
|
#ifdef GTM_TRIGGER
|
||
|
GBLREF mval dollar_ztwormhole;
|
||
|
#endif
|
||
|
|
||
|
/* Note this module follows the basic pattern of op_newvar which handles the same
|
||
|
function except for local vars instead of intrinsic vars. */
|
||
|
void gtm_newintrinsic(mval *intrinsic)
|
||
|
{
|
||
|
mv_stent *mv_st_ent, *mvst_tmp, *mvst_prev;
|
||
|
stack_frame *fp, *fp_prev, *fp_fix;
|
||
|
tp_frame *tpp;
|
||
|
unsigned char *old_sp, *top;
|
||
|
int indx;
|
||
|
int4 shift_size;
|
||
|
|
||
|
error_def(ERR_STACKOFLOW);
|
||
|
error_def(ERR_STACKCRIT);
|
||
|
|
||
|
assert(intrinsic);
|
||
|
if (frame_pointer->type & SFT_COUNT)
|
||
|
{ /* Current (youngest) frame is NOT an indirect frame.
|
||
|
Create restore entry for intrinsic var.
|
||
|
*/
|
||
|
PUSH_MV_STENT(MVST_MSAV);
|
||
|
mv_chain->mv_st_cont.mvs_msav.v = *intrinsic;
|
||
|
mv_chain->mv_st_cont.mvs_msav.addr = intrinsic;
|
||
|
} else
|
||
|
{ /* Current (youngest) frame IS an indirect frame.
|
||
|
The situation is more complex because this is not a true stackframe.
|
||
|
It has full access to the base "counted" frame's vars and any new
|
||
|
done here must behave as if it were done in the base/counted frame.
|
||
|
To accomplish this, we actually find the base frame we are executing
|
||
|
in, then shift all frames younger than that up by the size of the mvstent
|
||
|
entry we need to save/restore the value being new'd and then go into
|
||
|
each frame modified and fixup all the addresses.
|
||
|
*/
|
||
|
fp = frame_pointer;
|
||
|
fp_prev = fp->old_frame_pointer;
|
||
|
assert(fp_prev);
|
||
|
/* Find relevant base (counted) frame */
|
||
|
while (!(fp_prev->type & SFT_COUNT))
|
||
|
{
|
||
|
fp = fp_prev;
|
||
|
fp_prev = fp->old_frame_pointer;
|
||
|
assert(fp_prev);
|
||
|
}
|
||
|
/* top is beginning of earliest indirect stackframe before counted base frame.
|
||
|
It is the point where we will shift to make room to insert an mv_stent into
|
||
|
the base frame.
|
||
|
*/
|
||
|
top = (unsigned char *)(fp + 1);
|
||
|
old_sp = msp;
|
||
|
shift_size = mvs_size[MVST_MSAV];
|
||
|
msp -= shift_size;
|
||
|
if (msp <= stackwarn)
|
||
|
{
|
||
|
if (msp <= stacktop)
|
||
|
{
|
||
|
msp = old_sp;
|
||
|
rts_error(VARLSTCNT(1) ERR_STACKOFLOW);
|
||
|
}
|
||
|
else
|
||
|
rts_error(VARLSTCNT(1) ERR_STACKCRIT);
|
||
|
}
|
||
|
/* Ready, set, shift the younger indirect frames to make room for mv_stent */
|
||
|
memmove(msp, old_sp, top - (unsigned char *)old_sp);
|
||
|
mv_st_ent = (mv_stent *)(top - shift_size);
|
||
|
mv_st_ent->mv_st_type = MVST_MSAV;
|
||
|
ADJUST_FRAME_POINTER(frame_pointer, shift_size);
|
||
|
/* adjust all the pointers in all the stackframes that were moved */
|
||
|
for (fp_fix = frame_pointer; fp_fix != fp_prev; fp_fix = fp_fix->old_frame_pointer)
|
||
|
{
|
||
|
if ((unsigned char *)fp_fix->l_symtab < top && (unsigned char *)fp_fix->l_symtab > stacktop)
|
||
|
fp_fix->l_symtab = (ht_ent_mname **)((char *)fp_fix->l_symtab - shift_size);
|
||
|
if (fp_fix->temps_ptr < top && fp_fix->temps_ptr > stacktop)
|
||
|
fp_fix->temps_ptr -= shift_size;
|
||
|
if (fp_fix->vartab_ptr < (char *)top && fp_fix->vartab_ptr > (char *)stacktop)
|
||
|
fp_fix->vartab_ptr -= shift_size;
|
||
|
if ((unsigned char *)fp_fix->old_frame_pointer < top && (char *)fp_fix->old_frame_pointer
|
||
|
> (char *)stacktop)
|
||
|
{
|
||
|
ADJUST_FRAME_POINTER(fp_fix->old_frame_pointer, shift_size);
|
||
|
}
|
||
|
}
|
||
|
/* Adjust stackframe and mvstent pointers in relevant tp_frame blocks */
|
||
|
assert(((NULL == tp_pointer) && !dollar_tlevel) || ((NULL != tp_pointer) && dollar_tlevel));
|
||
|
for (tpp = tp_pointer; (tpp && ((unsigned char *)tpp->fp < top)); tpp = tpp->old_tp_frame)
|
||
|
{
|
||
|
if ((unsigned char *)tpp->fp > stacktop)
|
||
|
tpp->fp = (struct stack_frame_struct *)((char *)tpp->fp - shift_size);
|
||
|
/* Note low check for < top may be superfluous here but without a test case to verify, I
|
||
|
feel better leaving it in. SE 8/2001 */
|
||
|
if ((unsigned char *)tpp->mvc < top && (unsigned char *)tpp->mvc > stacktop)
|
||
|
tpp->mvc = (struct mv_stent_struct *)((char *)tpp->mvc - shift_size);
|
||
|
}
|
||
|
/* Put new mvstent entry on (into) the mvstent chain */
|
||
|
if ((unsigned char *)mv_chain >= top)
|
||
|
{ /* Just put new entry on end of chain which preceeds our base frame */
|
||
|
mv_st_ent->mv_st_next = (unsigned int)((char *)mv_chain - (char *)mv_st_ent);
|
||
|
mv_chain = mv_st_ent;
|
||
|
} else
|
||
|
{ /* One of the indirect frames has mv_stents associated with it so we have to find
|
||
|
the appropriate insertion point for this frame.
|
||
|
*/
|
||
|
fp = (stack_frame *)((char *)fp - shift_size);
|
||
|
mv_chain = (mv_stent *)((char *)mv_chain - shift_size);
|
||
|
mvst_tmp = mv_chain;
|
||
|
mvst_prev = (mv_stent *)((char *)mvst_tmp + mvst_tmp->mv_st_next);
|
||
|
while (mvst_prev < (mv_stent *)fp)
|
||
|
{
|
||
|
mvst_tmp = mvst_prev;
|
||
|
mvst_prev = (mv_stent *)((char *)mvst_tmp + mvst_tmp->mv_st_next);
|
||
|
}
|
||
|
mvst_tmp->mv_st_next = (unsigned int)((char *)mv_st_ent - (char *)mvst_tmp);
|
||
|
mv_st_ent->mv_st_next = (unsigned int)((char *)mvst_prev - (char *)mv_st_ent + shift_size);
|
||
|
}
|
||
|
/* Save current values of intrinsic var in the inserted mv_stent */
|
||
|
mv_st_ent->mv_st_cont.mvs_msav.v = *intrinsic;
|
||
|
mv_st_ent->mv_st_cont.mvs_msav.addr = intrinsic;
|
||
|
}
|
||
|
/* New the intrinsic var's current value if not $ZTWORMHOLE */
|
||
|
# ifdef GTM_TRIGGER
|
||
|
if (&dollar_ztwormhole != intrinsic)
|
||
|
{
|
||
|
# endif
|
||
|
intrinsic->mvtype = MV_STR;
|
||
|
intrinsic->str.len = 0;
|
||
|
# ifdef GTM_TRIGGER
|
||
|
}
|
||
|
# endif
|
||
|
return;
|
||
|
}
|