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 + -
显示快捷键?