condasm.c

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

C
408
字号
/****************************************************************************
*
*                            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:  WASM conditional processing routines
*
****************************************************************************/


#include "asmglob.h"
#include <ctype.h>

#include "asmins.h"
#include "directiv.h"
#include "asmexpnd.h"
#include "asmdefs.h"

#include "myassert.h"

extern int              get_instruction_position( char *string );

extern int              MacroExitState;

static enum if_state CurState = ACTIVE;

// fixme char *IfSymbol;        /* save symbols in IFDEF's so they don't get expanded */
#define    MAX_NESTING  20

void prep_line_for_conditional_assembly( char *line )
/***************************************************/
{
    char        *ptr;
    char        *end;
    int         count;
    char        delim;
    char        fix;

    if( Comment( QUERY_COMMENT, 0 ) ) {
        delim = (char)Comment( QUERY_COMMENT_DELIM, 0 );
        if( strchr( line, delim ) != NULL ) {
            Comment( END_COMMENT, 0 );
        }
        *line = '\0';
        return;
    }

    if( *line == '\0' ) return;

    /* fixme */
    /* get rid of # signs */
    ptr = line + strspn( line, " \t" );
    for( end = ptr; isalnum( *end ) || *end == '#' ; ++end )
        /* nothing to do */ ;
    fix = *end;
    *end = '\0';

    if( *ptr == '#' ) {
        /* treat #if as if, remove # sign */
        *ptr =' ';
        ptr++;
    }

    count = get_instruction_position( ptr );
    *end = fix;
    if( count == EMPTY ) {
        /* if it is not in the table */
        if(( CurState == LOOKING_FOR_TRUE_COND )
            || ( CurState == DONE )
            || MacroExitState ) {
            *line = '\0';
        }
        return;
    }

    /* otherwise, see if it is a conditional assembly directive */

    switch( AsmOpTable[count].token) {
    case T_IFDEF:
    case T_IFNDEF:
    case T_IFB:
    case T_IFNB:
    case T_IF:
    case T_IF1:
    case T_IF2:
    case T_IFE:
    case T_ELSEIF:
    case T_ELSE:
    case T_ENDIF:
    case T_IFDIF:
    case T_IFDIFI:
    case T_IFIDN:
    case T_IFIDNI:
        break;
    default:
        if(( CurState == LOOKING_FOR_TRUE_COND )
            || ( CurState == DONE )
            || MacroExitState ) {
            *line = '\0';
        }
    }
    return;
}

static char check_defd( char *string )
/************************************/
{
    char                *ptr;
    char                *end;
    struct asm_sym      *sym;

    /* isolate 1st word */
    ptr = string + strspn( string, " \t" );
    end = ptr + strcspn( ptr, " \t" );
    *end = '\0';

    sym = AsmGetSymbol( ptr );
    if( sym != NULL ) {
        return( TRUE );
    } else {
        return( FALSE );
    }
}

static char check_blank( char *string )
/************************************/
{
    return( strlen( string ) == 0 ? TRUE : FALSE );
}

static char check_dif( bool sensitive, char *string, char *string2 )
/******************************************************************/
{
    if( sensitive ) {
        return( strcmp( string, string2 ) != 0 );
    } else {
        return( stricmp( string, string2 ) != 0 );
    }
}

int conditional_error_directive( int i )
/**************************************/
{
    uint_16         direct;

    direct = AsmBuffer[i]->value;

    /* expand any constants if necessary */
    switch( direct ) {
    case T_DOT_ERRE:
    case T_DOT_ERRNZ:
    case T_DOT_ERRDIF:
    case T_DOT_ERRDIFI:
    case T_DOT_ERRIDN:
    case T_DOT_ERRIDNI:
        ExpandTheWorld( i+1, FALSE, TRUE );
    }

    switch( direct ) {
    case T_DOT_ERR:
        AsmErr( FORCED );
        return( ERROR );
    case T_DOT_ERRNZ:
        if( AsmBuffer[i+1]->token == T_NUM && AsmBuffer[i+1]->value ) {
            AsmErr( FORCED_NOT_ZERO, AsmBuffer[i+1]->value );
            return( ERROR );
        }
        break;
    case T_DOT_ERRE:
        if( AsmBuffer[i+1]->token == T_NUM && !AsmBuffer[i+1]->value ) {
            AsmErr( FORCED_EQUAL, AsmBuffer[i+1]->value );
            return( ERROR );
        }
        break;
    case T_DOT_ERRDEF:
        if( check_defd( AsmBuffer[i+1]->string_ptr ) ) {
            AsmErr( FORCED_DEF, AsmBuffer[i+1]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRNDEF:
        if( !check_defd( AsmBuffer[i+1]->string_ptr ) ) {
            AsmErr( FORCED_NOT_DEF, AsmBuffer[i+1]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRB:
        if( AsmBuffer[i+1]->token == T_STRING &&
            check_blank( AsmBuffer[i+1]->string_ptr ) ) {
            AsmErr( FORCED_BLANK, AsmBuffer[i+1]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRNB:
        if( AsmBuffer[i+1]->token != T_STRING ||
            !check_blank( AsmBuffer[i+1]->string_ptr ) ) {
            AsmErr( FORCED_NOT_BLANK, AsmBuffer[i+1]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRDIF:
        if( check_dif( TRUE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ) {
            AsmErr( FORCED_DIF, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRDIFI:
        if( check_dif( FALSE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ) {
            AsmErr( FORCED_DIF, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRIDN:
        if( !check_dif( TRUE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ) {
            AsmErr( FORCED_IDN, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr );
            return( ERROR );
        }
        break;
    case T_DOT_ERRIDNI:
        if( !check_dif( FALSE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ) {
            AsmErr( FORCED_IDN, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr );
            return( ERROR );
        }
        break;
    }
    return( NOT_ERROR );
}

static int_8    NestLevel = 0;

int conditional_assembly_directive( int i )
/*****************************************/
{
    uint_16         direct;
    static int_8    falseblocknestlevel = 0;

    direct = AsmBuffer[i]->value;

    switch( CurState ) {
    case ACTIVE:
        switch( direct ) {
        case T_IF:
        case T_IF1:
        case T_IF2:
        case T_IFE:
        case T_IFDEF:
        case T_IFNDEF:
        case T_IFB:
        case T_IFNB:
        case T_IFDIF:
        case T_IFDIFI:
        case T_IFIDN:
        case T_IFIDNI:
            NestLevel++;
            if( NestLevel > MAX_NESTING ) {
                /*fixme */
                AsmError( NESTING_LEVEL_TOO_DEEP );
                return( ERROR );
            }
        }
        break;
    case LOOKING_FOR_TRUE_COND:
        if( direct == T_ELSE || direct == T_ELSEIF ) {
            if( falseblocknestlevel == 0 ) {
                break;
            }
        }
        /* fall through for other cases */
    case DONE:
        switch( direct ) {
        case T_IF:
        case T_IF1:
        case T_IF2:
        case T_IFE:
        case T_IFDEF:
        case T_IFNDEF:
        case T_IFB:
        case T_IFNB:
        case T_IFDIF:
        case T_IFDIFI:
        case T_IFIDN:
        case T_IFIDNI:
            falseblocknestlevel++;
            return( NOT_ERROR );
            break;
        case T_ENDIF:
            if( falseblocknestlevel > 0 ) {
                falseblocknestlevel--;
                return( NOT_ERROR );
            }
            break;
        case T_ELSE:
        case T_ELSEIF:
            return( NOT_ERROR );
        default:
            /**/myassert( 0 );
            return( ERROR );
        }
    }

    /* expand any constants if necessary */
    switch( direct ) {
    case T_IF:
    case T_IFE:
    case T_ELSEIF:
    case T_IFDIF:
    case T_IFDIFI:
    case T_IFIDN:
    case T_IFIDNI:
        ExpandTheWorld( i+1, FALSE, TRUE );
    }

    switch( direct ) {
    case T_IF1:
        CurState = Parse_Pass == PASS_1 ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IF2:
        CurState = Parse_Pass == PASS_1 ? LOOKING_FOR_TRUE_COND : ACTIVE;
        break;
    case T_IF:
        CurState = ( AsmBuffer[i+1]->token == T_NUM && AsmBuffer[i+1]->value )
                   ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFE:
        CurState = ( AsmBuffer[i+1]->token == T_NUM && !AsmBuffer[i+1]->value )
                   ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFDEF:
        CurState = check_defd( AsmBuffer[i+1]->string_ptr )  ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFNDEF:
        CurState = !check_defd( AsmBuffer[i+1]->string_ptr )  ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFB:
        CurState = check_blank( AsmBuffer[i+1]->string_ptr ) ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFNB:
        CurState = !check_blank( AsmBuffer[i+1]->string_ptr ) ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFDIF:
        CurState = check_dif( TRUE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFDIFI:
        CurState = check_dif( FALSE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFIDN:
        CurState = !check_dif( TRUE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_IFIDNI:
        CurState = !check_dif( FALSE, AsmBuffer[i+1]->string_ptr, AsmBuffer[i+3]->string_ptr ) ? ACTIVE : LOOKING_FOR_TRUE_COND;
        break;
    case T_ELSE:
    case T_ELSEIF:
        switch( CurState ) {
        case ACTIVE:
            CurState = DONE;
            break;
        case LOOKING_FOR_TRUE_COND:
            if( direct == T_ELSE ) {
                CurState = ACTIVE;
            } else {
                CurState = ( AsmBuffer[i+1]->token == T_NUM &&
                             AsmBuffer[i+1]->value )
                           ? ACTIVE : LOOKING_FOR_TRUE_COND;
            }
            break;
        }
        break;
    case T_ENDIF:
        NestLevel--;
        if( NestLevel < 0 ) {
            NestLevel = 0;
            AsmError( BLOCK_NESTING_ERROR );
            return( ERROR );
        }
        CurState = ACTIVE;
    }
    return( NOT_ERROR );
}

void CheckForOpenConditionals( void )
{
    if( NestLevel > 0 ) {
        AsmErr( UNCLOSED_CONDITIONALS, NestLevel );
    }
}

⌨️ 快捷键说明

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