i87wait.c

来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 191 行

C
191
字号
/****************************************************************************
*
*                            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:  Insert FWAIT instructions as necessary.
*
****************************************************************************/


#include "standard.h"
#include "opcodes.h"
#include "coderep.h"
#include "pattern.h"
#include "vergen.h"
#include "gen8087.h"
#include "model.h"

extern    block *HeadBlock;
extern    bool  BlockByBlock;

extern  void            PrefixIns(instruction*,instruction*);
extern  instruction     *MakeWait(void);
extern  void            SuffixIns(instruction*,instruction*);
extern  bool            SameThing(name*,name*);
extern  bool            DoesSomething( instruction *ins );

static  bool    OpsCollide( name *op_87, name *op_86 ) {
/****************************************************/

    if( op_87->n.class == N_INDEXED ) {
        if( op_86 == NULL ) return( FALSE );
        if( op_86->n.class == N_INDEXED ) {
            if( op_87->i.base == NULL ) return( TRUE );
            if( op_86->i.base == NULL ) return( TRUE );
            return( SameThing( op_87->i.base, op_86->i.base ) );
        }
        if( op_86->n.class != N_MEMORY && op_86->n.class != N_TEMP ) {
            return( FALSE );
        }
        if( op_86->v.usage & ( NEEDS_MEMORY | USE_ADDRESS ) ) {
            if( op_87->i.base != NULL ) {
                return( SameThing( op_87->i.base, op_86 ) );
            }
            return( TRUE );
        }
    } else if( SameThing( op_87, op_86 ) ) {
        return( TRUE );
    }
    return( FALSE );
}


static  bool    CallNeedsWait( name *op, name *res ) {
/****************************************************/

    if( op == NULL ) {
        op = res;
    }
    if( op == NULL ) return( FALSE );
    if( op->n.class == N_REGISTER ) return( FALSE );
    if( op->n.class == N_CONSTANT ) return( FALSE );
    if( op->n.class == N_INDEXED ) return( TRUE );
    if( op->v.usage & (USE_ADDRESS | NEEDS_MEMORY) ) return( TRUE );
    return( FALSE );
}


static  bool    NeedWait( name *op, name *res, instruction *ins_86 ) {
/********************************************************************/

    bool        need_wait;
    int         i;

    if( op != NULL ) {
        need_wait = OpsCollide( op, ins_86->result );
    } else {
        if( !DoesSomething( ins_86 ) ) return( FALSE );
        need_wait = OpsCollide( res, ins_86->result );
        i = ins_86->num_operands;
        while( --i >= 0 ) {
            need_wait |= OpsCollide( res, ins_86->operands[ i ] );
        }
    }
    return( need_wait );
}


extern  void    Wait8087( void ) {
/**************************/

    block       *blk;
    instruction *ins;
    instruction *last_non_fpins = NULL;
    instruction *last_fpins;
    gentype     gen;
    name        *last_fpop = NULL;
    name        *last_fpres = NULL;
    bool        past_jump;

    /* Prior to 386, FWAITs are always needed. On 386 and up, only when
     * user explicitly asked for them.
     */
    if( _CPULevel( CPU_386 ) && _IsntTargetModel( GEN_FWAIT_386 ) ) return;

    blk = HeadBlock;
    while( blk != NULL ) {
        ins = blk->ins.hd.next;
        last_fpins = NULL;
        past_jump = FALSE;
        for(;;) {
            for(;;) {
                if( ins->head.opcode == OP_BLOCK ) break;
                if( _OpIsCall( ins->head.opcode ) ) break;
                gen = ins->u.gen_table->generate;
                if( _GenIs8087( gen ) ) {
                    last_non_fpins = NULL;
                    switch( gen ) {
                    case G_MFLD:
                    case G_MNFBIN:
                    case G_MRFBIN:
                        last_fpop = ins->operands[ 0 ];
                        last_fpres = NULL;
                        last_fpins = ins;
                        break;
                    case G_MFSTNP:
                    case G_MFST:
                    case G_MFSTRND:
                        last_fpins = ins;
                        last_fpres = ins->result;
                        last_fpop = NULL;
                        break;
                    default:
                        last_fpins = NULL;
                        break;
                    }
                } else {
                    if( past_jump || _OpIsJump( ins->head.opcode ) ) {
                        past_jump = TRUE;
                    } else {
                        last_non_fpins = ins;
                        if( last_fpins != NULL
                         && NeedWait( last_fpop, last_fpres, ins ) ) {
                            PrefixIns( ins, MakeWait() );
                            last_fpins = NULL;
                        }
                    }
                }
                ins = ins->head.next;
            }
            if( ins->head.opcode == OP_BLOCK ) break;
            if( last_fpins != NULL
             && CallNeedsWait( last_fpop, last_fpres ) ) {
                PrefixIns( ins, MakeWait() );
                last_fpins = NULL;
            }
            ins = ins->head.next;
        }
        if( last_fpins != NULL
        && ( ( last_fpres != NULL ) || ( last_fpop != NULL ) ) ) {
            if( last_non_fpins != NULL ) {
                SuffixIns( last_non_fpins, MakeWait() );
            } else {
                SuffixIns( last_fpins, MakeWait() );
            }
        }
        blk = blk->next_block;
    }
}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?