* harbour/Makefile
+ harbour/make_bsd.sh
+ harbour/make_drw.sh
* harbour/make_tgz.sh
+ harbour/bin/hb-func.sh
* harbour/bin/pack_src.sh
+ harbour/bin/postinst.bat
+ harbour/bin/postinst.sh
* harbour/config/dir.cf
* harbour/config/bsd/gcc.cf
* harbour/config/bsd/global.cf
+ harbour/config/darwin/dir.cf
+ harbour/config/darwin/gcc.cf
+ harbour/config/darwin/global.cf
+ harbour/config/darwin/install.cf
* harbour/config/dos/dir.cf
* harbour/config/dos/global.cf
* harbour/config/dos/install.cf
* harbour/config/dos/owatcom.cf
* harbour/config/dos/watcom.cf
* harbour/config/linux/gcc.cf
* harbour/config/linux/global.cf
* harbour/contrib/Makefile
* harbour/contrib/dot/pp_harb.ch
* harbour/contrib/libct/Makefile
* harbour/contrib/libct/bit1.c
* harbour/contrib/libct/bit2.c
* harbour/contrib/libct/bit3.c
* harbour/contrib/libct/files.c
* harbour/contrib/libct/ftoc.c
- harbour/contrib/libct/invertwin.prg
+ harbour/contrib/libct/invrtwin.prg
* harbour/contrib/libct/keyset.c
* harbour/contrib/libct/makefile.bc
* harbour/contrib/libmisc/dates2.c
* harbour/contrib/rdd_ads/ads1.c
* harbour/contrib/rdd_ads/adsfunc.c
* harbour/contrib/samples/date.c
* harbour/include/dbinfo.ch
* harbour/include/hbapi.h
* harbour/include/hbapicdp.h
* harbour/include/hbapifs.h
* harbour/include/hbapigt.h
* harbour/include/hbapiitm.h
* harbour/include/hbcomp.h
* harbour/include/hbdate.h
* harbour/include/hbdbf.h
* harbour/include/hbdbferr.h
* harbour/include/hbdefs.h
* harbour/include/hbexprb.c
* harbour/include/hbexprc.c
* harbour/include/hbexprop.h
* harbour/include/hbmacro.h
* harbour/include/hbmath.h
* harbour/include/hbpcode.h
* harbour/include/hbrddcdx.h
* harbour/include/hbrdddbf.h
* harbour/include/hbrdddbt.h
* harbour/include/hbrddfpt.h
* harbour/include/hbrddntx.h
* harbour/include/hbset.h
* harbour/include/hbsetup.h
* harbour/include/set.ch
* harbour/source/common/Makefile
* harbour/source/common/expropt1.c
* harbour/source/common/expropt2.c
- harbour/source/common/hbffind.c
* harbour/source/common/hbfsapi.c
* harbour/source/common/hbstr.c
* harbour/source/common/hbver.c
+ harbour/source/common/hbverdsp.c
* harbour/source/compiler/genc.c
* harbour/source/compiler/gencli.c
* harbour/source/compiler/harbour.c
* harbour/source/compiler/harbour.l
* harbour/source/compiler/harbour.slx
* harbour/source/compiler/harbour.sly
* harbour/source/compiler/harbour.y
* harbour/source/compiler/hbfix.c
* harbour/source/compiler/hbpcode.c
* harbour/source/compiler/hbusage.c
* harbour/source/compiler/simplex.c
* harbour/source/macro/macro.l
* harbour/source/macro/macro.slx
* harbour/source/macro/macro.y
* harbour/source/pp/ppcore.c
* harbour/source/rdd/dbcmd.c
* harbour/source/rdd/dbf1.c
* harbour/source/rdd/dbfcdx/dbfcdx1.c
- harbour/source/rdd/dbfcdx/dbfcdx1.h
* harbour/source/rdd/dbffpt/dbffpt1.c
* harbour/source/rdd/dbfntx/dbfntx1.c
* harbour/source/rtl/Makefile
* harbour/source/rtl/abs.c
* harbour/source/rtl/at.c
* harbour/source/rtl/datec.c
* harbour/source/rtl/dates.c
* harbour/source/rtl/dateshb.c
* harbour/source/rtl/datesx.c
* harbour/source/rtl/empty.c
* harbour/source/rtl/errorapi.c
* harbour/source/rtl/errorint.c
* harbour/source/rtl/filesys.c
* harbour/source/rtl/fstemp.c
+ harbour/source/rtl/hbffind.c
* harbour/source/rtl/hbrandom.c
* harbour/source/rtl/idle.c
* harbour/source/rtl/inkey.c
* harbour/source/rtl/math.c
* harbour/source/rtl/minmax.c
* harbour/source/rtl/pad.c
* harbour/source/rtl/padc.c
* harbour/source/rtl/padl.c
* harbour/source/rtl/padr.c
* harbour/source/rtl/philes.c
* harbour/source/rtl/round.c
* harbour/source/rtl/soundex.c
* harbour/source/rtl/str.c
* harbour/source/rtl/strcase.c
* harbour/source/rtl/strmatch.c
* harbour/source/rtl/strtran.c
* harbour/source/rtl/strzero.c
* harbour/source/rtl/substr.c
* harbour/source/rtl/val.c
* harbour/source/rtl/gtcrs/gtcrs.c
* harbour/source/rtl/gtcrs/kbdcrs.c
* harbour/source/rtl/gtdos/gtdos.c
* harbour/source/rtl/gtsln/gtsln.c
* harbour/source/rtl/gtsln/kbsln.c
* harbour/source/rtl/gtsln/keytrans.c
* harbour/source/vm/arrays.c
* harbour/source/vm/codebloc.c
* harbour/source/vm/estack.c
* harbour/source/vm/eval.c
* harbour/source/vm/extend.c
* harbour/source/vm/hvm.c
* harbour/source/vm/itemapi.c
* harbour/source/vm/macro.c
* harbour/source/vm/memvars.c
* harbour/source/vm/runner.c
* harbour/tests/bldtest/bldtest.c
* harbour/utils/hbtest/hbtest.prg
* harbour/utils/hbtest/rt_misc.prg
* harbour/utils/hbtest/rt_str.prg
* Sorry but it's too much modification for full description
cvs diff gives file 785982 bytes length. So I only count the
main things:
! cleand the code (no more warning messages under Linux and GCC and
DOS OpenWatcom) - some of them were real bugs
! cleaned all endian dependend code I've found - now Harbour can be
compiled on LITLE and BIG endian machines - for some other like
PDP ENDIAN it's enough to define proper macros in hbdefs.h
+ added macros for to get/put values in chosen byte order:
HB_GET_LE_[U]INT{16,24,32,64}( pPtr )
HB_GET_BE_[U]INT{16,24,32,64}( pPtr )
HB_PUT_LE_[U]INT{16,24,32,64}( pPtr, nVal )
HB_PUT_BE_[U]INT{16,24,32,64}( pPtr, nVal )
+ added macro HB_CAST_BYTE_NUMBERS_OFF which disables casting in
HB_{GET|PUT}_{LE|BE}_* macros - it's necessary for some platforms
like ALPHA DEC.
! cleaned the code for 64bit machines
* changed all parameters in hb_date* functions (day, month, year, week)
from LONG to int - it doesn't change binary compatibility for 32bit
machines but can cause troubles with compiling the old source
+ changed HB_IT_LONG type to HB_LONG which is mapped to long long
by default for 32 bit machines.
+ change HB_IT_INTEGER to be real 'int' C type not 'short int'
+ added HB_IS_NUMINT() macro
+ added hb_parnll, hb_stornll, hb_retnll, hbretnlllen, hb_itemPutNLL,
hb_itemPutNLLLen, hb_itemGetNLL which operates on LONGLONG
+ added hb_parnint, hb_stornint, hb_retnint, hb_retnintlen,
hb_itemPutNInt, hb_itemPutNIntLen, hb_itemGetNInt which operates on
HB_LONG
+ added HB_PUSHLONGLONG pcode
+ changed compiler and optimizer to use HB_LONG numbers and reduce
conversion from to double which may damage the 64bit number.
+ common functions for string to number conversions for compiler, RTL
and RDD to reduce problems with differ FL values for the same number:
hb_compStrToNum(), hb_valStrnToNum(), hb_strToNum(), hb_strnToNum()
+ common function hb_numRoun() which uses exactly the same algorithms
as string to number conversion for the same reason - please keep
this functions together.
+ hack inside hb_numRound() similar to the one used by CL5.3
+ hb_numInt() which uses uses the same hack as hb_numRound()
+ rewritten number to string conversion
+ some new string manipulation functions hb_strncpy(), hb_strncat(), ...
They works differ the the C one - always set 0 at the end, the buffer
has to be n+1 bytes length, the n is total size of buffer not the
left free space.
! cleaned some code which operates on ASCIIZ string to avoid potential
buffer overflow
+ updated RDD code - it's the first part - in few days I plan to change
workarea structure in both projects - it will break any 3rd party RDDs
so they have to be updated. I want to add SUPERTABLE into workarea
to allow creating new RDD on-line.
! cleaned the bugs with negating integers - on most machines (like x86)
the integers are not 0 symmetric - it means that x = -x does not work
for {INT,LONG,LONGLONG}_MIN (hb_vmNegate, ABS())
+ cleaned error messages to be Clipper compatible.
+ updated build process for .DEB packages - now hb* scripts and shared
libs are created by standard make install
* added new .prg #defines: __PLATFORM__<cPlatfrom>,
__ARCH{16|32|64}BIT__, __LITTLE_ENDIAN__|__BIG_ENDIAN__|__PDP_ENDIAN__
!!! cPlatfrom can have lower letters (for xHarbour compatibility)
If you do not like it please change it.
* others ...
910 lines
24 KiB
C
910 lines
24 KiB
C
/*
|
|
* $Id$
|
|
*/
|
|
|
|
/*
|
|
* Harbour Project source code:
|
|
* Compiler Expression Optimizer - common expressions
|
|
*
|
|
* Copyright 1999 Ryszard Glab
|
|
* www - http://www.harbour-project.org
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
* any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software; see the file COPYING. If not, write to
|
|
* the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
|
* Boston, MA 02111-1307 USA (or visit the web site http://www.gnu.org/).
|
|
*
|
|
* As a special exception, the Harbour Project gives permission for
|
|
* additional uses of the text contained in its release of Harbour.
|
|
*
|
|
* The exception is that, if you link the Harbour libraries with other
|
|
* files to produce an executable, this does not by itself cause the
|
|
* resulting executable to be covered by the GNU General Public License.
|
|
* Your use of that executable is in no way restricted on account of
|
|
* linking the Harbour library code into it.
|
|
*
|
|
* This exception does not however invalidate any other reasons why
|
|
* the executable file might be covered by the GNU General Public License.
|
|
*
|
|
* This exception applies only to the code released by the Harbour
|
|
* Project under the name Harbour. If you copy code from other
|
|
* Harbour Project or Free Software Foundation releases into a copy of
|
|
* Harbour, as the General Public License permits, the exception does
|
|
* not apply to the code that you add in this way. To avoid misleading
|
|
* anyone as to the status of such modified files, you must delete
|
|
* this exception notice from them.
|
|
*
|
|
* If you write modifications of your own for Harbour, it is your choice
|
|
* whether to permit this exception to apply to your modifications.
|
|
* If you do not wish that, delete this exception notice.
|
|
*
|
|
*/
|
|
|
|
/* TODO:
|
|
* - Correct post- and pre- operations to correctly handle the following code
|
|
* a[ i++ ]++
|
|
* Notice: in current implementation (an in Clipper too) 'i++' is evaluated
|
|
* two times! This causes that the new value (after incrementation) is
|
|
* stored in next element of the array.
|
|
*/
|
|
|
|
/* NOTE: This must be the first definition
|
|
* This is a common code shared by macro and standalone compiler
|
|
*/
|
|
#define HB_MACRO_SUPPORT
|
|
|
|
#include <math.h>
|
|
#include "hbmacro.h"
|
|
#include "hbcomp.h"
|
|
|
|
/* memory allocation
|
|
*/
|
|
#define HB_XGRAB( size ) hb_xgrab( (size) )
|
|
#define HB_XFREE( pPtr ) hb_xfree( (void *)(pPtr) )
|
|
|
|
static char * s_OperTable[] = {
|
|
"",
|
|
"NIL",
|
|
"Numeric",
|
|
"String",
|
|
"Codeblock",
|
|
"Logical",
|
|
"SELF",
|
|
"Array",
|
|
"@",
|
|
"@",
|
|
"@",
|
|
"IIF",
|
|
",",
|
|
",",
|
|
"[",
|
|
"&",
|
|
"()",
|
|
"->",
|
|
"->",
|
|
":",
|
|
"", /* symbol */
|
|
"", /* alias */
|
|
"", /* RunTime variable */
|
|
"", /* variable */
|
|
"++", /* post-operators -> lowest precedence */
|
|
"--",
|
|
":=", /* assigments */
|
|
"+=",
|
|
"-=",
|
|
"*=",
|
|
"/=",
|
|
"%=",
|
|
"^=",
|
|
".OR.", /* logical operators */
|
|
".AND.",
|
|
".NOT.",
|
|
"=", /* relational operators */
|
|
"==",
|
|
"<",
|
|
">",
|
|
"<=",
|
|
">=",
|
|
"!=",
|
|
"$",
|
|
"+", /* addition */
|
|
"-",
|
|
"*", /* multiple */
|
|
"/",
|
|
"%",
|
|
"^",
|
|
"-", /* sign operator */
|
|
"++",
|
|
"--"
|
|
};
|
|
|
|
/* ************************************************************************* */
|
|
|
|
/* Increase a reference counter (this allows to share the same expression
|
|
* in more then one context)
|
|
*/
|
|
HB_EXPR_PTR hb_compExprClone( HB_EXPR_PTR pSrc )
|
|
{
|
|
pSrc->Counter++;
|
|
return pSrc;
|
|
}
|
|
|
|
char * hb_compExprDescription( HB_EXPR_PTR pExpr )
|
|
{
|
|
if( pExpr )
|
|
return s_OperTable[ pExpr->ExprType ];
|
|
else
|
|
return s_OperTable[ 0 ];
|
|
}
|
|
|
|
int hb_compExprType( HB_EXPR_PTR pExpr )
|
|
{
|
|
return ( int ) pExpr->ExprType;
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
|
|
HB_EXPR_PTR hb_compExprNewEmpty( void )
|
|
{
|
|
return hb_compExprNew( HB_ET_NONE );
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewDouble( double dValue, BYTE ucWidth, BYTE ucDec )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewDouble(%f, %i)", dValue, ucDec));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_NUMERIC );
|
|
|
|
pExpr->value.asNum.dVal = dValue;
|
|
pExpr->value.asNum.bWidth = ucWidth;
|
|
pExpr->value.asNum.bDec = ucDec;
|
|
pExpr->value.asNum.NumType = HB_ET_DOUBLE;
|
|
pExpr->ValType = HB_EV_NUMERIC;
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewLong( HB_LONG lValue )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewLong(%" PFHL "d)", lValue));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_NUMERIC );
|
|
|
|
pExpr->value.asNum.lVal = lValue;
|
|
pExpr->value.asNum.bDec = 0;
|
|
pExpr->value.asNum.NumType = HB_ET_LONG;
|
|
pExpr->ValType = HB_EV_NUMERIC;
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
int hb_compExprIsInteger( HB_EXPR_PTR pExpr )
|
|
{
|
|
return ( pExpr->ExprType == HB_ET_NUMERIC && pExpr->value.asNum.NumType == HB_ET_LONG &&
|
|
HB_LIM_INT16( pExpr->value.asNum.lVal ) );
|
|
}
|
|
|
|
int hb_compExprAsInteger( HB_EXPR_PTR pExpr )
|
|
{
|
|
if( pExpr->ExprType == HB_ET_NUMERIC && pExpr->value.asNum.NumType == HB_ET_LONG )
|
|
return ( int ) pExpr->value.asNum.lVal;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
char *hb_compExprAsString( HB_EXPR_PTR pExpr )
|
|
{
|
|
switch( pExpr->ExprType )
|
|
{
|
|
case HB_ET_VARIABLE:
|
|
return pExpr->value.asSymbol ;
|
|
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewCodeBlock( char *string, BOOL isMacro, BOOL lateEval )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewCodeBlock(%s,%u,%u)",string,isMacro,lateEval));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_CODEBLOCK );
|
|
|
|
pExpr->value.asCodeblock.pExprList = NULL;
|
|
pExpr->value.asCodeblock.pLocals = NULL; /* this will hold local variables declarations */
|
|
pExpr->ValType = HB_EV_CODEBLOCK;
|
|
pExpr->value.asCodeblock.string = string;
|
|
pExpr->value.asCodeblock.isMacro = isMacro;
|
|
pExpr->value.asCodeblock.lateEval = lateEval;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprAddCodeblockExpr( HB_EXPR_PTR pList, HB_EXPR_PTR pNewItem )
|
|
{
|
|
if( pList->value.asCodeblock.pExprList )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
/* add new item to the end of the list */
|
|
pExpr = pList->value.asCodeblock.pExprList;
|
|
while( pExpr->pNext )
|
|
pExpr = pExpr->pNext;
|
|
pExpr->pNext = pNewItem;
|
|
}
|
|
else
|
|
pList->value.asCodeblock.pExprList = pNewItem;
|
|
|
|
return pList;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewLogical( int iValue )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewLogical(%i)", iValue));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_LOGICAL );
|
|
|
|
pExpr->value.asLogical = iValue;
|
|
pExpr->ValType = HB_EV_LOGICAL;
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
|
|
HB_EXPR_PTR hb_compExprNewNil( void )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewNil()"));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_NIL );
|
|
|
|
pExpr->ValType = HB_EV_NIL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewSelf( void )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewSelf()"));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_SELF );
|
|
|
|
pExpr->ValType = HB_EV_OBJECT;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewVarRef( char * szVarName )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewVarRef(%s)", szVarName));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_VARREF );
|
|
|
|
pExpr->value.asSymbol = szVarName;
|
|
pExpr->ValType = HB_EV_VARREF;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewFunRef( char * szFunName )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewFunRef(%s)", szFunName));
|
|
|
|
pExpr =hb_compExprNew( HB_ET_FUNREF );
|
|
|
|
pExpr->value.asSymbol = szFunName;
|
|
pExpr->ValType = HB_EV_FUNREF;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewRef( HB_EXPR_PTR pRefer )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewRef(%p)", pRefer));
|
|
|
|
pExpr =hb_compExprNew( HB_ET_REFERENCE );
|
|
|
|
pExpr->value.asReference = pRefer;
|
|
pExpr->ValType = HB_EV_VARREF;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Creates a new literal array { item1, item2, ... itemN }
|
|
* 'pArrList' is a list of array elements
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewArray( HB_EXPR_PTR pArrList )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewArray()"));
|
|
|
|
pArrList->ExprType = HB_ET_ARRAY; /* change type from ET_LIST */
|
|
pArrList->ValType = HB_EV_ARRAY;
|
|
pArrList->ulLength = 0;
|
|
|
|
pExpr = pArrList->value.asList.pExprList; /* get first element on the list */
|
|
/* Now we need to replace all EO_NONE expressions with ET_NIL expressions
|
|
* If EO_NONE is the first expression and there is no more expressions
|
|
* then it is an empty array {} and ET_NIL cannot be used
|
|
*/
|
|
if( pExpr->ExprType == HB_ET_NONE && pExpr->pNext == NULL )
|
|
{
|
|
pArrList->value.asList.pExprList = NULL;
|
|
}
|
|
else
|
|
{
|
|
/* there are at least one non-empty element specified
|
|
*/
|
|
while( pExpr )
|
|
{
|
|
/* if empty element was specified replace it with NIL value */
|
|
if( pExpr->ExprType == HB_ET_NONE )
|
|
pExpr->ExprType = HB_ET_NIL;
|
|
pExpr = pExpr->pNext;
|
|
++pArrList->ulLength;
|
|
}
|
|
}
|
|
pArrList->value.asList.pIndex = NULL;
|
|
return pArrList;
|
|
}
|
|
|
|
/* Creates new macro expression
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewMacro( HB_EXPR_PTR pMacroExpr, unsigned char cMacroOp, char * szName )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
if( szName )
|
|
{
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewMacro(%s)", szName));
|
|
|
|
/* Macro variable is used: &identifier
|
|
* or macro text: [text]&variable[more_macro_text]
|
|
*/
|
|
/*
|
|
* NOTE: Clipper assumes that all variables used in macro expressions
|
|
* are memvar variables
|
|
* NOTE: Clipper pushes the complete macro expression converted
|
|
* to string in case complex expression is used, e.g.
|
|
* My&var.1
|
|
* is pushed as:
|
|
* "MY&VAR.1"
|
|
*/
|
|
pExpr = hb_compExprNew( HB_ET_MACRO );
|
|
pExpr->value.asMacro.cMacroOp = cMacroOp; /* '&' if variable or 0 if text */
|
|
pExpr->value.asMacro.szMacro = szName; /* variable name or macro text */
|
|
pExpr->value.asMacro.pExprList = NULL; /* this is not a parenthesized expressions */
|
|
pExpr->value.asMacro.SubType = HB_ET_MACRO_VAR;
|
|
|
|
if( cMacroOp == 0 )
|
|
{
|
|
/* check if variable with valid scope is used in macro text
|
|
* (local, static and field variables are not allowed)
|
|
* e.g.
|
|
* LOCAL var
|
|
* ? &var // this is OK
|
|
* ? &var.ext // this is invalid
|
|
*/
|
|
char *szDupl;
|
|
BOOL bUseTextSubst;
|
|
|
|
szDupl = hb_strupr( hb_strdup( szName ) );
|
|
if( ! hb_compExprIsValidMacro( szDupl, &bUseTextSubst, NULL ) )
|
|
{
|
|
hb_compErrorMacro( szName );
|
|
}
|
|
hb_xfree( szDupl );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewMacro(&)"));
|
|
|
|
/* Macro expression: &( expression_list )
|
|
*/
|
|
pExpr = hb_compExprNew( HB_ET_MACRO );
|
|
pExpr->value.asMacro.pExprList = pMacroExpr;
|
|
pExpr->value.asMacro.szMacro = NULL; /* this is used to distinguish &(...) from &ident */
|
|
pExpr->value.asMacro.SubType = HB_ET_MACRO_EXPR;
|
|
}
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
/* Creates new aliased variable
|
|
* aliasexpr -> identifier
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewAliasVar( HB_EXPR_PTR pAlias, HB_EXPR_PTR pVariable )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewAliasVar()"));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_ALIASVAR );
|
|
|
|
pExpr->value.asAlias.pAlias = pAlias;
|
|
pExpr->value.asAlias.pVar = pVariable;
|
|
pExpr->value.asAlias.pExpList = NULL;
|
|
|
|
/* macro expressions in alias context require a special handling
|
|
*/
|
|
if( pAlias->ExprType == HB_ET_MACRO )
|
|
pAlias->value.asMacro.SubType = HB_ET_MACRO_ALIASED;
|
|
if( pVariable->ExprType == HB_ET_MACRO )
|
|
pVariable->value.asMacro.SubType = HB_ET_MACRO_ALIASED;
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
/* Creates new aliased expression
|
|
* alias_expr -> ( expression )
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewAliasExpr( HB_EXPR_PTR pAlias, HB_EXPR_PTR pExpList )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewAliasExpr()"));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_ALIASEXPR );
|
|
|
|
pExpr->value.asAlias.pAlias = pAlias;
|
|
pExpr->value.asAlias.pExpList = pExpList;
|
|
pExpr->value.asAlias.pVar = NULL;
|
|
|
|
if( pAlias->ExprType == HB_ET_MACRO )
|
|
{
|
|
/* Is it a special case &variable->( expressionList ) */
|
|
if( pAlias->value.asMacro.SubType == HB_ET_MACRO_VAR ||
|
|
pAlias->value.asMacro.SubType == HB_ET_MACRO_EXPR )
|
|
pAlias->value.asMacro.SubType = HB_ET_MACRO_ALIASED;
|
|
}
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
/* Creates new send expression
|
|
* pObject : szMessage
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewSend( HB_EXPR_PTR pObject, char * szMessage )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewSend(%p, %s)", pObject, szMessage));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_SEND );
|
|
pExpr->value.asMessage.szMessage = szMessage;
|
|
pExpr->value.asMessage.pObject = pObject;
|
|
pExpr->value.asMessage.pParms = NULL;
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
/* Creates new method call
|
|
* pObject : identifier ( pArgList )
|
|
*
|
|
* pObject = is an expression returned by hb_compExprNewSend
|
|
* pArgList = list of passed arguments - it will be HB_ET_NONE if no arguments
|
|
* are passed
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewMethodCall( HB_EXPR_PTR pObject, HB_EXPR_PTR pArgList )
|
|
{
|
|
pObject->value.asMessage.pParms = pArgList;
|
|
|
|
return pObject;
|
|
}
|
|
|
|
/* Creates a list - all elements will be used
|
|
* This list can be used to create an array or function's call arguments
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewList( HB_EXPR_PTR pFirstItem )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewList()"));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_LIST );
|
|
pExpr->value.asList.pExprList = pFirstItem;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Creates a list of function call arguments
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewArgList( HB_EXPR_PTR pFirstItem )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewArgList()"));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_ARGLIST );
|
|
pExpr->value.asList.pExprList = pFirstItem;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Adds new element to the list
|
|
*/
|
|
HB_EXPR_PTR hb_compExprAddListExpr( HB_EXPR_PTR pList, HB_EXPR_PTR pNewItem )
|
|
{
|
|
if( pList->value.asList.pExprList )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
/* add new item to the end of the list */
|
|
pExpr = pList->value.asList.pExprList;
|
|
while( pExpr->pNext )
|
|
pExpr = pExpr->pNext;
|
|
pExpr->pNext = pNewItem;
|
|
}
|
|
else
|
|
pList->value.asList.pExprList = pNewItem;
|
|
|
|
return pList;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewVar( char * szName )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewVar(%s)", szName));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_VARIABLE );
|
|
pExpr->value.asSymbol = szName;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Create a new declaration of PUBLIC or PRIVATE variable.
|
|
*
|
|
* szName is a string with variable name if 'PUBLIC varname' context
|
|
* pMacroVar is a macro expression if 'PUBLIC &varname' context
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewRTVar( char * szName, HB_EXPR_PTR pMacroVar )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewRTVar(%s, %p)", szName, pMacroVar));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_RTVAR );
|
|
pExpr->value.asRTVar.szName = szName;
|
|
pExpr->value.asRTVar.pMacro = pMacroVar;
|
|
if( pMacroVar )
|
|
pMacroVar->value.asMacro.SubType = HB_ET_MACRO_SYMBOL;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Create a new symbol used in function calls
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewFunName( char * szName )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewFunName(%s)", szName));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_FUNNAME );
|
|
pExpr->value.asSymbol = szName;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Create a new symbol used in an alias expressions
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewAlias( char * szName )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprNewAlias(%s)", szName));
|
|
|
|
pExpr = hb_compExprNew( HB_ET_ALIAS );
|
|
pExpr->value.asSymbol = szName;
|
|
return pExpr;
|
|
}
|
|
|
|
|
|
/* ************************************************************************* */
|
|
|
|
HB_EXPR_PTR hb_compExprNewEqual( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_EQUAL );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPlus( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_PLUS );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewMinus( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_MINUS );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewMult( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_MULT );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewDiv( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_DIV );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewMod( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_MOD );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPower( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_POWER );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPostInc( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_POSTINC );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPostDec( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_POSTDEC );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPreInc( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_PREINC );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPreDec( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_PREDEC );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewPlusEq( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_PLUSEQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewMinusEq( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_MINUSEQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewMultEq( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_MULTEQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewDivEq( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_DIVEQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewModEq( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_MODEQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewExpEq( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_EXPEQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewAnd( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_AND );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewOr( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_OR );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewNot( HB_EXPR_PTR pNotExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
if( pNotExpr->ExprType == HB_ET_LOGICAL )
|
|
{
|
|
pNotExpr->value.asLogical = ! pNotExpr->value.asLogical;
|
|
pExpr = pNotExpr;
|
|
}
|
|
else
|
|
{
|
|
pExpr = hb_compExprNew( HB_EO_NOT );
|
|
pExpr->value.asOperator.pLeft = pNotExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
}
|
|
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewEQ( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_EQ );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewLT( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_LT );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewGT( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_GT );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewLE( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_LE );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewGE( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_GE );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewNE( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_NE );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
HB_EXPR_PTR hb_compExprNewIN( HB_EXPR_PTR pLeftExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr = hb_compExprNew( HB_EO_IN );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
return pExpr;
|
|
}
|
|
|
|
/* NOTE: all invalid cases are handled by yacc rules
|
|
*/
|
|
HB_EXPR_PTR hb_compExprNewNegate( HB_EXPR_PTR pNegExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
if( pNegExpr->ExprType == HB_ET_NUMERIC )
|
|
{
|
|
if( pNegExpr->value.asNum.NumType == HB_ET_DOUBLE )
|
|
{
|
|
pNegExpr->value.asNum.dVal = - pNegExpr->value.asNum.dVal;
|
|
pNegExpr->value.asNum.bWidth = HB_DBL_LENGTH( pNegExpr->value.asNum.dVal );
|
|
}
|
|
else
|
|
{
|
|
pNegExpr->value.asNum.lVal = - pNegExpr->value.asNum.lVal;
|
|
pNegExpr->value.asNum.bWidth = HB_LONG_LENGTH( pNegExpr->value.asNum.lVal );
|
|
}
|
|
pExpr = pNegExpr;
|
|
}
|
|
else
|
|
{
|
|
pExpr = hb_compExprNew( HB_EO_NEGATE );
|
|
pExpr->value.asOperator.pLeft = pNegExpr;
|
|
pExpr->value.asOperator.pRight = NULL;
|
|
}
|
|
return pExpr;
|
|
}
|
|
|
|
/* ************************************************************************* */
|
|
|
|
/* Handles (expression := expression) syntax
|
|
*/
|
|
HB_EXPR_PTR hb_compExprAssign( HB_EXPR_PTR pLeftExpr, HB_EXPR_PTR pRightExpr )
|
|
{
|
|
HB_EXPR_PTR pExpr;
|
|
|
|
HB_TRACE(HB_TR_DEBUG, ("hb_compExprAssign()"));
|
|
|
|
pExpr = hb_compExprNew( HB_EO_ASSIGN );
|
|
pExpr->value.asOperator.pLeft = pLeftExpr;
|
|
pExpr->value.asOperator.pRight = pRightExpr;
|
|
return pExpr;
|
|
}
|
|
|
|
/* Return a number of elements on the linked list
|
|
*/
|
|
ULONG hb_compExprListLen( HB_EXPR_PTR pExpr )
|
|
{
|
|
ULONG ulLen = 0;
|
|
|
|
pExpr = pExpr->value.asList.pExprList;
|
|
while( pExpr )
|
|
{
|
|
pExpr = pExpr->pNext;
|
|
++ulLen;
|
|
}
|
|
|
|
return ulLen;
|
|
}
|