i86opseg.c
来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 197 行
C
197 行
/****************************************************************************
*
* Open Watcom Project
*
* Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
* ========================================================================
*
* This file contains Original Code and/or Modifications of Original
* Code as defined in and that are subject to the Sybase Open Watcom
* Public License version 1.0 (the 'License'). You may not use this file
* except in compliance with the License. BY USING THIS FILE YOU AGREE TO
* ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
* provided with the Original Code and Modifications, and is also
* available at www.sybase.com/developer/opensource.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
* ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
* NON-INFRINGEMENT. Please see the License for the specific language
* governing rights and limitations under the License.
*
* ========================================================================
*
* Description: WHEN YOU FIGURE OUT WHAT THIS FILE DOES, PLEASE
* DESCRIBE IT HERE!
*
****************************************************************************/
#include "standard.h"
#include "coderep.h"
#include "procdef.h"
#include "zeropage.h"
#include "model.h"
#include "opcodes.h"
#include "pattern.h"
#include "vergen.h"
zero_page_scheme ZPageType;
extern proc_def *CurrProc;
extern block *HeadBlock;
extern byte OptForSize;
extern void QuickSave(hw_reg_set,opcode_defs);
extern void GenRegMove(hw_reg_set,hw_reg_set);
extern void GenRegXor(hw_reg_set,hw_reg_set);
extern void GenRegNeg(hw_reg_set);
extern bool SegIsSS(name*);
extern bool CanZapBP(void);
extern int NumOperands(instruction*);
/* forward declarations */
static int Overs( name *op );
static int CountSegOvers( void );
extern void InitZeroPage( void ) {
/*******************************
Decide what type of "zeropage" scheme we should use (if it's worthwhile).
This is sort of a misleading name, carried over from the days of the 6809.
What it is is cheap addressing to the stack segment. We'll set BP to zero
and use _X[BP], or, if that's not possible, set DI or SI to -BP and
use _x[bp+di] or _x[bp+si]. This generates the code to initialize
the "zeropage" register, as well as pushing it to save it.
*/
ZPageType = ZP_USES_SS;
if( _IsTargetModel( FLOATING_DS ) && OptForSize > 50 ) {
if( !HW_COvlap( CurrProc->state.used, HW_DS ) ) {
ZPageType = ZP_USES_DS;
if( CountSegOvers() > 4 ) {
QuickSave( HW_DS, OP_PUSH );
QuickSave( HW_SS, OP_PUSH );
QuickSave( HW_DS, OP_POP );
} else {
ZPageType = ZP_USES_SS;
}
} else if( CanZapBP() && !HW_COvlap( CurrProc->state.used, HW_BP ) ) {
ZPageType = ZP_USES_BP;
if( CountSegOvers() > 4 ) {
QuickSave( HW_BP, OP_PUSH );
GenRegXor( HW_BP, HW_BP );
} else {
ZPageType = ZP_USES_SS;
}
} else if( !HW_COvlap( CurrProc->state.used, HW_DI ) ) {
ZPageType = ZP_USES_DI;
if( CountSegOvers() > 6 ) {
QuickSave( HW_DI, OP_PUSH );
GenRegMove( HW_BP, HW_DI );
GenRegNeg( HW_DI );
} else {
ZPageType = ZP_USES_SS;
}
} else if( !HW_COvlap( CurrProc->state.used, HW_SI ) ) {
ZPageType = ZP_USES_SI;
if( CountSegOvers() > 6 ) {
QuickSave( HW_SI, OP_PUSH );
GenRegMove( HW_BP, HW_SI );
GenRegNeg( HW_SI );
} else {
ZPageType = ZP_USES_SS;
}
}
}
}
extern void FiniZeroPage( void ) {
/*******************************
Pop the "zeropage" register.
*/
switch( ZPageType ) {
case ZP_USES_SS:
/* nothing*/
break;
case ZP_USES_DS:
QuickSave( HW_DS, OP_POP );
break;
case ZP_USES_BP:
QuickSave( HW_BP, OP_POP );
break;
case ZP_USES_SI:
QuickSave( HW_SI, OP_POP );
break;
case ZP_USES_DI:
QuickSave( HW_DI, OP_POP );
break;
}
ZPageType = ZP_USES_SS;
}
static int CountSegOvers( void ) {
/********************************
Count the number of SS: overrides in the routine.
*/
block *blk;
instruction *ins;
int i;
int overs;
overs = 0;
blk = HeadBlock;
while( blk != NULL ) {
ins = blk->ins.hd.next;
while( ins->head.opcode != OP_BLOCK ) {
i = NumOperands( ins );
if( i == ins->num_operands
&& ( ZPageType == ZP_USES_DS
|| ( ins->u.gen_table->generate != G_MOVAM
&& ins->u.gen_table->generate != G_MOVMA ) ) ) {
while( --i >= 0 ) {
overs += Overs( ins->operands[ i ] );
}
if( ins->result != NULL ) {
overs += Overs( ins->result );
}
}
ins = ins->head.next;
}
blk = blk->next_block;
}
return( overs );
}
static int Overs( name *op ) {
/**********************************
return the number of SS: overrides associated with "op" (0 or 1)
*/
name *base;
name *index;
if( op->n.class == N_MEMORY ) {
if( SegIsSS( op ) ) return( 1 );
return( 0 );
} else if( op->n.class == N_INDEXED ) {
base = op->i.base;
if( !HasTrueBase( op ) ) return( 0 );
if( base->n.class != N_MEMORY ) return( 0 );
if( !SegIsSS( base ) ) return( 0 );
index = op->i.index;
if( HW_CEqual( index->r.reg, HW_BX ) ) return( 0 );
if( ZPageType == ZP_USES_SI ) return( 0 );
if( ZPageType == ZP_USES_DI ) return( 0 );
return( 1 );
}
return( 0 );
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?