2017-10-21 18:40:19 -05:00
|
|
|
{$optimize 1}
|
|
|
|
{---------------------------------------------------------------}
|
|
|
|
{ }
|
|
|
|
{ Expression }
|
|
|
|
{ }
|
|
|
|
{ Evaluate expressions }
|
|
|
|
{ }
|
|
|
|
{ Note: The expression evaluator uses the scanner to fetch }
|
|
|
|
{ tokens, but IT IS ALSO USED BY THE SCANNER to evaluate }
|
|
|
|
{ expressions in preprocessor commands. This circular }
|
|
|
|
{ dependency is handle by defining all of the expression }
|
|
|
|
{ evaluator's external types, constants, and variables in the }
|
|
|
|
{ CCOMMON module. The only procedure from this module used by }
|
|
|
|
{ the scanner is Expression, which is declared as an external }
|
|
|
|
{ procedure in the scanner. }
|
|
|
|
{ }
|
|
|
|
{ External Variables: }
|
|
|
|
{ }
|
|
|
|
{ startExpression - tokens that may start an expression }
|
|
|
|
{ bitDisp,bitSize - bit field disp, size }
|
|
|
|
{ unsigned - is the bit field unsigned? }
|
|
|
|
{ isBitField - is the field a bit field? }
|
|
|
|
{ }
|
|
|
|
{ External Subroutines: }
|
|
|
|
{ }
|
|
|
|
{ AssignmentConversion - do type checking and conversions for }
|
|
|
|
{ assignment statements }
|
|
|
|
{ CompareToZero - Compare the result on tos to zero. }
|
|
|
|
{ DisposeTree - dispose of an expression tree }
|
|
|
|
{ DoSelection - Find the displacement & type for a }
|
|
|
|
{ selection operation }
|
|
|
|
{ Expression - handle an expression }
|
|
|
|
{ FreeTemp - place a temporary label in the available label }
|
|
|
|
{ list }
|
|
|
|
{ GenerateCode - generate code from a fully formed expression }
|
|
|
|
{ tree }
|
|
|
|
{ GetTemp - find a temporary work variable }
|
2020-01-29 17:09:52 -06:00
|
|
|
{ InitExpression - initialize the expression handler }
|
2017-10-21 18:40:19 -05:00
|
|
|
{ UsualBinaryConversions - performs the usual binary }
|
|
|
|
{ conversions }
|
|
|
|
{ UsualUnaryConversions - performs the usual unary conversions }
|
|
|
|
{ }
|
|
|
|
{---------------------------------------------------------------}
|
|
|
|
|
|
|
|
unit Expression;
|
|
|
|
|
|
|
|
{$LibPrefix '0/obj/'}
|
|
|
|
|
|
|
|
interface
|
|
|
|
|
2018-03-23 21:51:27 -04:00
|
|
|
uses CCommon, Table, CGI, Scanner, Symbol, MM, Printf;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2017-06-18 22:07:32 -05:00
|
|
|
{$segment 'EXP'}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
var
|
|
|
|
startExpression: tokenSet; {tokens that can start an expression}
|
|
|
|
|
|
|
|
{set by DoSelection}
|
|
|
|
{------------------}
|
|
|
|
bitDisp,bitSize: integer; {bit field disp, size}
|
|
|
|
unsigned: boolean; {is the bit field unsigned?}
|
|
|
|
isBitField: boolean; {is the field a bit field?}
|
|
|
|
|
|
|
|
{misc}
|
|
|
|
{----}
|
|
|
|
lastwasconst: boolean; {did the last GenerateCode result in an integer constant?}
|
|
|
|
lastconst: longint; {last integer constant from GenerateCode}
|
2022-06-23 22:04:03 -05:00
|
|
|
lastWasNullPtrConst: boolean; {did last GenerateCode give a null ptr const?}
|
2017-10-21 18:40:19 -05:00
|
|
|
{---------------------------------------------------------------}
|
|
|
|
|
|
|
|
procedure AssignmentConversion (t1, t2: typePtr; isConstant: boolean;
|
|
|
|
value: longint; genCode, checkConst: boolean);
|
|
|
|
|
|
|
|
{ TOS is of type t2, and is about to be stored to a variable of }
|
|
|
|
{ type t1 by an assignment or a return statement. Make sure }
|
|
|
|
{ this is legal, and do any necessary type conversions on t2, }
|
|
|
|
{ which is on the top of the evaluation stack. Flag an error }
|
|
|
|
{ if the conversion is illegal. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ t1 - type of the variable }
|
|
|
|
{ t2 - type of the expression }
|
|
|
|
{ isConstant - is the rhs a constant? }
|
|
|
|
{ value - if isConstant = true, then this is the value }
|
|
|
|
{ genCode - should conversion code be generated? }
|
|
|
|
{ checkConst - check for assignments to constants? }
|
|
|
|
|
|
|
|
|
|
|
|
procedure CompareToZero(op: pcodes);
|
|
|
|
|
|
|
|
{ Compare the result on tos to zero. }
|
|
|
|
{ }
|
|
|
|
{ This procedure is used by the logical statements to compare }
|
|
|
|
{ _any_ scalar result to zero, giving a boolean result. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ op - operation to use on the compare }
|
|
|
|
|
|
|
|
|
|
|
|
procedure DisposeTree (tree: tokenPtr);
|
|
|
|
|
|
|
|
{ dispose of an expression tree }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - head of the expression tree to dispose of }
|
|
|
|
|
|
|
|
|
|
|
|
procedure DoSelection (lType: typePtr; tree: tokenPtr; var size: longint);
|
|
|
|
|
|
|
|
{ Find the displacement & type for a selection operation }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ lType - structure/union type }
|
|
|
|
{ id - tag field name }
|
|
|
|
{ size - disp into the structure/union }
|
|
|
|
{ }
|
|
|
|
{ returned in non-local variables: }
|
|
|
|
{ bitDisp - displacement to bit field }
|
|
|
|
{ bitSize - size of bit field }
|
|
|
|
{ unsigned - is the bit field unsigned? }
|
|
|
|
{ isBitField - is the field a bit field? }
|
|
|
|
{ }
|
2020-01-29 17:09:52 -06:00
|
|
|
{ variables: }
|
2017-10-21 18:40:19 -05:00
|
|
|
{ expressionType - set to the type of the field }
|
|
|
|
|
|
|
|
|
|
|
|
procedure Expression (kind: expressionKind; stopSym: tokenSet);
|
|
|
|
|
|
|
|
{ handle an expression }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ kind - Kind of expression; determines what operations }
|
|
|
|
{ and what kind of operands are allowed. }
|
|
|
|
{ stopSym - Set of symbols that can mark the end of an }
|
|
|
|
{ expression; used to skip tokens after syntax }
|
|
|
|
{ errors and to block certain operations. For }
|
|
|
|
{ example, the comma operator is not allowed in }
|
|
|
|
{ an expression when evaluating a function }
|
|
|
|
{ parameter list. }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ realExpressionValue - value of a real constant }
|
|
|
|
{ expression }
|
|
|
|
{ expressionValue - value of a constant expression }
|
|
|
|
{ expressionType - type of the constant expression }
|
|
|
|
|
|
|
|
|
|
|
|
procedure FreeTemp(labelNum, size: integer);
|
|
|
|
|
|
|
|
{ place a temporary label in the available label list }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ labelNum - number of the label to free }
|
|
|
|
{ size - size of the variable }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ tempList - list of free labels }
|
|
|
|
|
|
|
|
|
|
|
|
procedure GenerateCode (tree: tokenPtr);
|
|
|
|
|
|
|
|
{ generate code from a fully formed expression tree }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - top of the expression tree to generate code from }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ expressionType - result type of the expression }
|
|
|
|
|
|
|
|
|
|
|
|
function GetTemp(size: integer): integer;
|
|
|
|
|
|
|
|
{ find a temporary work variable }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ size - size of the variable }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ tempList - list of free labels }
|
|
|
|
{ }
|
|
|
|
{ Returns the label number. }
|
|
|
|
|
|
|
|
|
|
|
|
procedure InitExpression;
|
|
|
|
|
|
|
|
{ initialize the expression handler }
|
|
|
|
|
|
|
|
|
|
|
|
function UsualBinaryConversions (lType: typePtr): baseTypeEnum;
|
|
|
|
|
|
|
|
{ performs the usual binary conversions }
|
|
|
|
{ }
|
|
|
|
{ inputs: }
|
|
|
|
{ lType - type of the left operand }
|
|
|
|
{ expressionType - type of the right operand }
|
|
|
|
{ }
|
|
|
|
{ result: }
|
|
|
|
{ The base type of the operation to perform is }
|
|
|
|
{ returned. Any conversion code necessary has been }
|
|
|
|
{ generated. }
|
|
|
|
{ }
|
|
|
|
{ outputs: }
|
|
|
|
{ expressionType - set to result type }
|
|
|
|
|
|
|
|
|
|
|
|
function UsualUnaryConversions: baseTypeEnum;
|
|
|
|
|
|
|
|
{ performs the usual unary conversions }
|
|
|
|
{ }
|
|
|
|
{ inputs: }
|
|
|
|
{ expressionType - type of the operand }
|
|
|
|
{ }
|
|
|
|
{ result: }
|
|
|
|
{ The base type of the operation to perform is returned. }
|
|
|
|
{ Any conversion code necessary has been generated. }
|
|
|
|
{ }
|
|
|
|
{ outputs: }
|
|
|
|
{ expressionType - set to result type }
|
|
|
|
|
2021-02-17 19:41:46 -06:00
|
|
|
procedure GetLLExpressionValue (var val: longlong);
|
|
|
|
|
|
|
|
{ get the value of the last integer constant expression as a }
|
|
|
|
{ long long (whether it had long long type or not). }
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
{---------------------------------------------------------------}
|
|
|
|
|
|
|
|
implementation
|
|
|
|
|
|
|
|
const
|
|
|
|
{notAnOperation is also used in TABLE.ASM}
|
|
|
|
notAnOperation = 200; {used as the icp for non-operation tokens}
|
|
|
|
|
|
|
|
var
|
|
|
|
{structured constants}
|
|
|
|
{--------------------}
|
|
|
|
startTerm: tokenSet; {tokens that can start a term}
|
|
|
|
|
|
|
|
{misc}
|
|
|
|
{----}
|
|
|
|
errorFound: boolean; {was there are error during generation?}
|
|
|
|
|
|
|
|
{-- Procedures imported from the parser ------------------------}
|
|
|
|
|
|
|
|
procedure Match (kind: tokenEnum; err: integer); extern;
|
|
|
|
|
|
|
|
{ insure that the next token is of the specified type }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ kind - expected token kind }
|
|
|
|
{ err - error number if the expected token is not found }
|
|
|
|
|
|
|
|
|
2022-10-01 21:28:16 -05:00
|
|
|
function TypeName: typePtr; extern;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2020-01-05 21:45:03 -06:00
|
|
|
{ process a type name (used for casts and sizeof/_Alignof) }
|
2017-10-21 18:40:19 -05:00
|
|
|
{ }
|
2022-10-01 21:28:16 -05:00
|
|
|
{ returns: a pointer to the type }
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2021-03-02 20:01:32 -06:00
|
|
|
|
|
|
|
function MakeFuncIdentifier: identPtr; extern;
|
|
|
|
|
|
|
|
{ Make the predefined identifier __func__. }
|
|
|
|
{ }
|
|
|
|
{ It is inserted in the symbol table as if the following }
|
|
|
|
{ declaration appeared at the beginning of the function body: }
|
|
|
|
{ }
|
|
|
|
{ static const char __func__[] = "function-name"; }
|
|
|
|
{ }
|
|
|
|
{ This must only be called within a function body. }
|
|
|
|
|
2021-09-15 22:22:16 -05:00
|
|
|
|
|
|
|
function MakeCompoundLiteral(tp: typePtr): identPtr; extern;
|
|
|
|
|
|
|
|
{ Make the identifier for a compound literal. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tp - the type of the compound literal }
|
|
|
|
|
2022-06-08 20:58:52 -05:00
|
|
|
|
2022-10-22 18:54:46 -05:00
|
|
|
procedure AutoInit (variable: identPtr; line: longint;
|
2022-06-08 20:58:52 -05:00
|
|
|
isCompoundLiteral: boolean); extern;
|
|
|
|
|
|
|
|
{ generate code to initialize an auto variable }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ variable - the variable to initialize }
|
|
|
|
{ line - line number (used for debugging) }
|
|
|
|
{ isCompoundLiteral - initializing a compound literal? }
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
{-- External unsigned math routines ----------------------------}
|
|
|
|
|
|
|
|
function lshr (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function udiv (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function uge (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function ugt (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function ule (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function ult (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function umod (x,y: longint): longint; extern;
|
|
|
|
|
|
|
|
function umul (x,y: longint): longint; extern;
|
|
|
|
|
2021-02-14 20:39:22 -06:00
|
|
|
{-- External 64-bit math routines ------------------------------}
|
|
|
|
{ Procedures for arithmetic and shifts compute "x := x OP y". }
|
|
|
|
|
|
|
|
procedure umul64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure udiv64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure div64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure umod64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure rem64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure add64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure sub64 (var x: longlong; y: longlong); extern;
|
|
|
|
|
|
|
|
procedure shl64 (var x: longlong; y: integer); extern;
|
|
|
|
|
|
|
|
procedure ashr64 (var x: longlong; y: integer); extern;
|
|
|
|
|
|
|
|
procedure lshr64 (var x: longlong; y: integer); extern;
|
|
|
|
|
2021-02-16 23:11:41 -06:00
|
|
|
function ult64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function uge64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function ule64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function ugt64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function slt64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function sge64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function sle64(a,b: longlong): integer; extern;
|
|
|
|
|
|
|
|
function sgt64(a,b: longlong): integer; extern;
|
|
|
|
|
2021-02-19 21:57:31 -06:00
|
|
|
{-- External conversion functions; imported from CGC.pas -------}
|
|
|
|
|
|
|
|
procedure CnvXLL (var result: longlong; val: extended); extern;
|
|
|
|
|
|
|
|
procedure CnvXULL (var result: longlong; val: extended); extern;
|
|
|
|
|
|
|
|
function CnvLLX (val: longlong): extended; extern;
|
|
|
|
|
|
|
|
function CnvULLX (val: longlong): extended; extern;
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
{---------------------------------------------------------------}
|
|
|
|
|
|
|
|
function Unary(tp: baseTypeEnum): baseTypeEnum;
|
|
|
|
|
|
|
|
{ usual unary conversions }
|
|
|
|
{ }
|
|
|
|
{ This function returns the base type actually loaded on the }
|
|
|
|
{ stack for a particular data type. This corresponds to C's }
|
|
|
|
{ usual unary conversions. }
|
|
|
|
{ }
|
|
|
|
{ parameter: }
|
|
|
|
{ tp - data type }
|
|
|
|
{ }
|
|
|
|
{ result: }
|
|
|
|
{ Stack type. }
|
|
|
|
|
|
|
|
begin {Unary}
|
2021-03-06 23:54:55 -06:00
|
|
|
if tp in [cgByte,cgUByte] then
|
|
|
|
tp := cgWord;
|
2017-10-21 18:40:19 -05:00
|
|
|
Unary := tp;
|
|
|
|
end; {Unary}
|
|
|
|
|
|
|
|
|
|
|
|
function UsualBinaryConversions {lType: typePtr): baseTypeEnum};
|
|
|
|
|
|
|
|
{ performs the usual binary conversions }
|
|
|
|
{ }
|
|
|
|
{ inputs: }
|
|
|
|
{ lType - type of the left operand }
|
|
|
|
{ expressionType - type of the right operand }
|
|
|
|
{ }
|
|
|
|
{ result: }
|
|
|
|
{ The base type of the operation to perform is }
|
|
|
|
{ returned. Any conversion code necessary has been }
|
|
|
|
{ generated. }
|
|
|
|
{ }
|
|
|
|
{ outputs: }
|
|
|
|
{ expressionType - set to result type }
|
|
|
|
|
|
|
|
var
|
|
|
|
rType: typePtr; {right type}
|
|
|
|
lt,rt: baseTypeEnum; {work variables}
|
|
|
|
|
2021-03-06 23:54:55 -06:00
|
|
|
|
|
|
|
function CommonRealType (lt, rt: baseTypeEnum): baseTypeEnum;
|
|
|
|
|
|
|
|
{ Compute the common real type of two types, where at least }
|
|
|
|
{ one of the types is a real type. }
|
|
|
|
{ }
|
|
|
|
{ inputs: }
|
|
|
|
{ lt, rt - the two operand types }
|
|
|
|
{ }
|
|
|
|
{ outputs: }
|
|
|
|
{ expressionType - set to result type }
|
|
|
|
|
|
|
|
begin {CommonRealType}
|
|
|
|
if (lt = cgComp) and (rt = cgComp) then
|
|
|
|
lt := cgComp
|
|
|
|
else if (lt in [cgExtended,cgComp]) or (rt in [cgExtended,cgComp]) then
|
|
|
|
lt := cgExtended
|
|
|
|
else if (lt = cgDouble) or (rt = cgDouble) then
|
|
|
|
lt := cgDouble
|
|
|
|
else
|
|
|
|
lt := cgReal;
|
|
|
|
CommonRealType := lt;
|
|
|
|
case lt of
|
|
|
|
cgReal: expressionType := floatPtr;
|
|
|
|
cgDouble: expressionType := doublePtr;
|
|
|
|
cgExtended: expressionType := extendedPtr;
|
|
|
|
cgComp: expressionType := compPtr;
|
|
|
|
end; {case}
|
|
|
|
end; {CommonRealType}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
begin {UsualBinaryConversions}
|
|
|
|
UsualBinaryConversions := cgULong;
|
|
|
|
if lType^.kind = pointerType then
|
|
|
|
lType := uLongPtr
|
|
|
|
else if lType^.kind = scalarType then
|
2016-10-14 20:37:10 -05:00
|
|
|
if lType^.baseType = cgVoid then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := uLongPtr;
|
2016-10-14 20:37:10 -05:00
|
|
|
Error(66);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
rType := expressionType;
|
|
|
|
if rType^.kind = pointerType then
|
|
|
|
rType := uLongPtr
|
|
|
|
else if rType^.kind = scalarType then
|
2016-10-14 20:37:10 -05:00
|
|
|
if rType^.baseType = cgVoid then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
rType := uLongPtr;
|
2016-10-14 20:37:10 -05:00
|
|
|
Error(66);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
if (lType^.kind = scalarType) and (rType^.kind = scalarType) then begin
|
|
|
|
lt := Unary(lType^.baseType);
|
|
|
|
rt := Unary(rType^.baseType);
|
|
|
|
if lt <> rt then begin
|
2021-03-06 23:54:55 -06:00
|
|
|
if lt in [cgReal,cgDouble,cgExtended,cgComp] then begin
|
2021-01-29 23:25:21 -06:00
|
|
|
if rt in [cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen2(pc_cnv, ord(rt), ord(cgExtended));
|
2021-03-06 23:54:55 -06:00
|
|
|
UsualBinaryConversions := CommonRealType(lt, rt);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
2021-03-06 23:54:55 -06:00
|
|
|
else if rt in [cgReal,cgDouble,cgExtended,cgComp] then begin
|
2021-01-29 23:25:21 -06:00
|
|
|
if lt in [cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen2(pc_cnn, ord(lt), ord(cgExtended));
|
2021-03-06 23:54:55 -06:00
|
|
|
UsualBinaryConversions := CommonRealType(lt, rt);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
2021-01-29 23:25:21 -06:00
|
|
|
else if lt = cgUQuad then begin
|
|
|
|
if rt in [cgWord,cgUWord,cgLong,cgULong] then
|
|
|
|
Gen2(pc_cnv, ord(rt), ord(cgUQuad));
|
|
|
|
UsualBinaryConversions := cgUQuad;
|
|
|
|
expressionType := uLongLongPtr;
|
|
|
|
end {else if}
|
|
|
|
else if rt = cgUQuad then begin
|
|
|
|
if lt in [cgWord,cgUWord,cgLong,cgULong] then
|
|
|
|
Gen2(pc_cnn, ord(lt), ord(cgUQuad));
|
|
|
|
UsualBinaryConversions := cgUQuad;
|
|
|
|
expressionType := uLongLongPtr;
|
|
|
|
end {else if}
|
|
|
|
else if lt = cgQuad then begin
|
|
|
|
if rt in [cgWord,cgUWord,cgLong,cgULong] then
|
|
|
|
Gen2(pc_cnv, ord(rt), ord(cgQuad));
|
|
|
|
UsualBinaryConversions := cgQuad;
|
|
|
|
expressionType := longLongPtr;
|
|
|
|
end {else if}
|
|
|
|
else if rt = cgQuad then begin
|
|
|
|
if lt in [cgWord,cgUWord,cgLong,cgULong] then
|
|
|
|
Gen2(pc_cnn, ord(lt), ord(cgQuad));
|
|
|
|
UsualBinaryConversions := cgQuad;
|
|
|
|
expressionType := longLongPtr;
|
|
|
|
end {else if}
|
2017-10-21 18:40:19 -05:00
|
|
|
else if lt = cgULong then begin
|
|
|
|
if rt in [cgWord,cgUWord] then
|
|
|
|
Gen2(pc_cnv, ord(rt), ord(cgULong));
|
|
|
|
UsualBinaryConversions := cgULong;
|
|
|
|
expressionType := uLongPtr;
|
|
|
|
end {else if}
|
|
|
|
else if rt = cgULong then begin
|
|
|
|
if lt in [cgWord,cgUWord] then
|
|
|
|
Gen2(pc_cnn, ord(lt), ord(cgULong));
|
|
|
|
UsualBinaryConversions := cgULong;
|
|
|
|
expressionType := uLongPtr;
|
|
|
|
end {else if}
|
|
|
|
else if lt = cgLong then begin
|
|
|
|
if rt in [cgWord,cgUWord] then
|
|
|
|
Gen2(pc_cnv, ord(rt), ord(cgLong));
|
|
|
|
UsualBinaryConversions := cgLong;
|
|
|
|
expressionType := longPtr;
|
|
|
|
end {else if}
|
|
|
|
else if rt = cgLong then begin
|
|
|
|
if lt in [cgWord,cgUWord] then
|
|
|
|
Gen2(pc_cnn, ord(lt), ord(cgLong));
|
|
|
|
UsualBinaryConversions := cgLong;
|
|
|
|
expressionType := longPtr;
|
|
|
|
end {else if}
|
|
|
|
else {one operand is unsigned in and the other is int} begin
|
|
|
|
UsualBinaryConversions := cgUWord;
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := uIntPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
|
|
|
end {if}
|
2016-11-20 19:25:26 -06:00
|
|
|
else begin {types are the same}
|
2017-10-21 18:40:19 -05:00
|
|
|
UsualBinaryConversions := lt;
|
2016-11-20 19:25:26 -06:00
|
|
|
if lt = cgWord then {update types that may have changed}
|
2021-03-06 23:54:55 -06:00
|
|
|
expressionType := intPtr;
|
2016-11-20 19:25:26 -06:00
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
end; {UsualBinaryConversions}
|
|
|
|
|
|
|
|
|
|
|
|
function UsualUnaryConversions{: baseTypeEnum};
|
|
|
|
|
|
|
|
{ performs the usual unary conversions }
|
|
|
|
{ }
|
|
|
|
{ inputs: }
|
|
|
|
{ expressionType - type of the operand }
|
|
|
|
{ }
|
|
|
|
{ result: }
|
|
|
|
{ The base type of the operation to perform is returned. }
|
|
|
|
{ Any conversion code necessary has been generated. }
|
|
|
|
{ }
|
|
|
|
{ outputs: }
|
|
|
|
{ expressionType - set to result type }
|
|
|
|
|
|
|
|
var
|
2016-11-20 19:25:26 -06:00
|
|
|
et: baseTypeEnum; {work variables}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
begin {UsualUnaryConversions}
|
|
|
|
UsualUnaryConversions := cgULong;
|
2016-11-20 19:25:26 -06:00
|
|
|
if expressionType^.kind = scalarType then begin
|
|
|
|
et := Unary(expressionType^.baseType);
|
|
|
|
UsualUnaryConversions := et;
|
|
|
|
if et = cgWord then {update types that may have changed}
|
2021-03-06 23:54:55 -06:00
|
|
|
expressionType := intPtr;
|
2016-11-20 19:25:26 -06:00
|
|
|
end {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
{else if expressionType^.kind in [arrayType,pointerType] then
|
|
|
|
UsualUnaryConversions := cgULong};
|
|
|
|
end; {UsualUnaryConversions}
|
|
|
|
|
|
|
|
|
|
|
|
procedure DisposeTree {tree: tokenPtr};
|
|
|
|
|
|
|
|
{ dispose of an expression tree }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - head of the expression tree to dispose of }
|
|
|
|
|
|
|
|
begin {DisposeTree}
|
|
|
|
if tree <> nil then begin
|
|
|
|
DisposeTree(tree^.left);
|
|
|
|
DisposeTree(tree^.middle);
|
|
|
|
DisposeTree(tree^.right);
|
|
|
|
dispose(tree);
|
|
|
|
end; {if}
|
|
|
|
end; {DisposeTree}
|
|
|
|
|
|
|
|
|
2022-06-23 22:04:03 -05:00
|
|
|
procedure ValueExpressionConversions;
|
|
|
|
|
|
|
|
{ Perform type conversions applicable to an expression used }
|
|
|
|
{ for its value. These include lvalue conversion (removing }
|
|
|
|
{ qualifiers), array-to-pointer conversion, and }
|
|
|
|
{ function-to-pointer conversion. See C17 section 6.3.2.1. }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ expressionType - set to type after conversions }
|
|
|
|
|
|
|
|
begin {ValueExpressionConversions}
|
|
|
|
expressionType := Unqualify(expressionType);
|
|
|
|
if expressionType^.kind = arrayType then
|
|
|
|
expressionType := MakePointerTo(expressionType^.aType)
|
|
|
|
else if expressionType^.kind = functionType then
|
|
|
|
expressionType := MakePointerTo(expressionType);
|
|
|
|
end; {ValueExpressionConversions}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
procedure AssignmentConversion {t1, t2: typePtr; isConstant: boolean;
|
|
|
|
value: longint; genCode, checkConst: boolean};
|
|
|
|
|
|
|
|
{ TOS is of type t2, and is about to be stored to a variable of }
|
|
|
|
{ type t1 by an assignment or a return statement. Make sure }
|
|
|
|
{ this is legal, and do any necessary type conversions on t2, }
|
|
|
|
{ which is on the top of the evaluation stack. Flag an error }
|
|
|
|
{ if the conversion is illegal. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ t1 - type of the variable }
|
|
|
|
{ t2 - type of the expression }
|
|
|
|
{ isConstant - is the rhs a constant? }
|
|
|
|
{ value - if isConstant = true, then this is the value }
|
|
|
|
{ genCode - should conversion code be generated? }
|
|
|
|
{ checkConst - check for assignments to constants? }
|
|
|
|
|
|
|
|
var
|
|
|
|
baseType1,baseType2: baseTypeEnum; {temp variables (for speed)}
|
|
|
|
kind1,kind2: typeKind; {temp variables (for speed)}
|
|
|
|
|
2023-01-03 18:56:46 -06:00
|
|
|
|
|
|
|
procedure CheckConstantRange(t1: typePtr; value: longint);
|
|
|
|
|
|
|
|
{ Check for situations where an implicit conversion will }
|
|
|
|
{ change the value of a constant. }
|
|
|
|
{ }
|
|
|
|
{ Note: This currently only addresses conversions to 8-bit }
|
|
|
|
{ or 16-bit integer types, and intentionally does not }
|
|
|
|
{ distinguish between signed and unsigned types. }
|
|
|
|
|
|
|
|
var
|
|
|
|
min,max: longint; {min/max allowed values}
|
|
|
|
|
|
|
|
begin {CheckConstantRange}
|
|
|
|
if t1^.cType = ctBool then begin
|
|
|
|
min := 0;
|
|
|
|
max := 1;
|
|
|
|
end {if}
|
|
|
|
else if t1^.baseType in [cgByte,cgUByte] then begin
|
|
|
|
min := -128;
|
|
|
|
max := 255;
|
|
|
|
end {else if}
|
|
|
|
else if t1^.baseType in [cgWord,cgUWord] then begin
|
|
|
|
min := -32768;
|
|
|
|
max := 65536;
|
|
|
|
end {else if}
|
|
|
|
else begin
|
|
|
|
min := -maxint4-1;
|
|
|
|
max := maxint4;
|
|
|
|
end; {else}
|
|
|
|
if (value < min) or (value > max) then
|
|
|
|
Error(186);
|
|
|
|
end; {CheckConstantRange}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
begin {AssignmentConversion}
|
|
|
|
kind1 := t1^.kind;
|
|
|
|
kind2 := t2^.kind;
|
2021-09-11 18:12:58 -05:00
|
|
|
if genCode then
|
|
|
|
if checkConst then
|
|
|
|
if kind2 <> definedType then
|
|
|
|
if tqConst in t1^.qualifiers then
|
|
|
|
Error(93)
|
|
|
|
else if kind1 in [structType,unionType] then
|
|
|
|
if t1^.constMember then
|
|
|
|
Error(93);
|
2017-10-21 18:40:19 -05:00
|
|
|
if kind2 = definedType then
|
|
|
|
AssignmentConversion(t1, t2^.dType, false, 0, genCode, checkConst)
|
|
|
|
else if kind1 = definedType then
|
|
|
|
AssignmentConversion(t1^.dType, t2, false, 0, genCode, checkConst)
|
|
|
|
else if kind2 in
|
|
|
|
[scalarType,pointerType,enumType,structType,unionType,arrayType,functionType] then
|
|
|
|
case kind1 of
|
|
|
|
|
|
|
|
scalarType: begin
|
2023-01-03 18:56:46 -06:00
|
|
|
if ((lint & lintConstantRange) <> 0) then
|
|
|
|
if isConstant then
|
|
|
|
CheckConstantRange(t1, value);
|
2017-10-21 18:40:19 -05:00
|
|
|
baseType1 := t1^.baseType;
|
2021-03-06 22:28:39 -06:00
|
|
|
if baseType1 in [cgReal,cgDouble,cgComp] then
|
|
|
|
baseType1 := cgExtended;
|
2017-10-21 18:40:19 -05:00
|
|
|
if baseType1 = cgString then
|
|
|
|
Error(64)
|
|
|
|
else if baseType1 = cgVoid then
|
|
|
|
Error(65)
|
|
|
|
else if kind2 = enumType then begin
|
|
|
|
if genCode then
|
|
|
|
Gen2(pc_cnv, ord(cgWord), ord(baseType1));
|
|
|
|
end {else if}
|
|
|
|
else if kind2 = scalarType then begin
|
|
|
|
baseType2 := t2^.baseType;
|
|
|
|
if baseType2 in [cgString,cgVoid] then
|
|
|
|
Error(47)
|
2021-01-25 21:22:58 -06:00
|
|
|
else if genCode then begin
|
|
|
|
if t1^.cType = ctBool then begin
|
|
|
|
expressionType := t2;
|
|
|
|
CompareToZero(pc_neq);
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Gen2(pc_cnv, ord(baseType2), ord(baseType1));
|
|
|
|
end {else if}
|
|
|
|
end {else if}
|
|
|
|
else if (t1^.cType = ctBool)
|
|
|
|
and (kind2 in [pointerType,arrayType]) then begin
|
|
|
|
if genCode then begin
|
|
|
|
expressionType := t2;
|
|
|
|
CompareToZero(pc_neq);
|
|
|
|
end {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
|
|
|
else
|
|
|
|
Error(47);
|
|
|
|
end;
|
|
|
|
|
|
|
|
arrayType: ;
|
|
|
|
{any errors are handled elsewhere}
|
|
|
|
|
|
|
|
functionType,enumConst:
|
|
|
|
Error(47);
|
|
|
|
|
|
|
|
pointerType: begin
|
|
|
|
if kind2 = pointerType then begin
|
|
|
|
if not CompTypes(t1, t2) then
|
2021-09-09 21:39:29 -05:00
|
|
|
Error(47)
|
|
|
|
else if not looseTypeChecks then
|
|
|
|
if not (t1^.ptype^.qualifiers >= t2^.ptype^.qualifiers) then
|
|
|
|
Error(163);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else if kind2 = arrayType then begin
|
2021-09-09 21:39:29 -05:00
|
|
|
if not CompTypes(t1^.ptype, t2^.atype) and
|
|
|
|
(t1^.ptype^.baseType <> cgVoid) then
|
|
|
|
Error(47)
|
|
|
|
else if not looseTypeChecks then
|
|
|
|
if not (t1^.ptype^.qualifiers >= t2^.atype^.qualifiers) then
|
|
|
|
Error(163);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else if kind2 = scalarType then begin
|
|
|
|
if isConstant and (value = 0) then begin
|
|
|
|
if genCode then
|
|
|
|
Gen2(pc_cnv, ord(t2^.baseType), ord(cgULong));
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(47);
|
|
|
|
end {else if}
|
|
|
|
else
|
|
|
|
Error(47);
|
|
|
|
end;
|
|
|
|
|
|
|
|
enumType: begin
|
|
|
|
if kind2 = scalarType then begin
|
2023-01-03 18:56:46 -06:00
|
|
|
if ((lint & lintConstantRange) <> 0) then
|
|
|
|
if isConstant then
|
|
|
|
CheckConstantRange(intPtr, value);
|
2017-10-21 18:40:19 -05:00
|
|
|
baseType2 := t2^.baseType;
|
|
|
|
if baseType2 in [cgString,cgVoid] then
|
|
|
|
Error(47)
|
|
|
|
else if genCode then
|
|
|
|
Gen2(pc_cnv, ord(baseType2), ord(cgWord));
|
|
|
|
end {if}
|
|
|
|
else if kind2 <> enumType then
|
|
|
|
Error(47);
|
|
|
|
end;
|
|
|
|
|
|
|
|
definedType:
|
|
|
|
AssignmentConversion(t1^.dType, t2, isConstant, value, genCode,
|
|
|
|
checkConst);
|
|
|
|
|
|
|
|
structType,unionType:
|
|
|
|
if not CompTypes(t1, t2) then
|
|
|
|
Error(47);
|
|
|
|
|
|
|
|
otherwise: Error(57);
|
|
|
|
|
|
|
|
end; {case T1^.kind}
|
|
|
|
|
|
|
|
expressionType := t1; {set the type of the expression}
|
|
|
|
end; {AssignmentConversion}
|
|
|
|
|
|
|
|
|
|
|
|
function ExpressionTree (kind: expressionKind; stopSym: tokenSet): tokenPtr;
|
|
|
|
|
|
|
|
{ generate an expression tree }
|
|
|
|
{ }
|
|
|
|
{ Returns a pointer to the generated tree. The pointer is }
|
|
|
|
{ nil, and the variable errorFound is set to true, if an }
|
|
|
|
{ error is found. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ kind - Kind of expression; determines what operations }
|
|
|
|
{ and what kind of operands are allowed. }
|
|
|
|
{ stopSym - Set of symbols that can mark the end of an }
|
|
|
|
{ expression; used to skip tokens after syntax }
|
|
|
|
{ errors and to block certain operations. For }
|
|
|
|
{ example, the comma operator is not allowed in }
|
|
|
|
{ an expression when evaluating a function }
|
|
|
|
{ parameter list. }
|
|
|
|
|
2016-01-01 16:23:30 -06:00
|
|
|
label 1,2,3;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
var
|
|
|
|
done,done2: boolean; {for loop termination}
|
|
|
|
doingSizeof: boolean; {used to test for a sizeof operator}
|
2020-01-05 18:09:26 -06:00
|
|
|
doingAlignof: boolean; {used to test for an _Alignof operator}
|
2017-10-21 18:40:19 -05:00
|
|
|
expectingTerm: boolean; {should the next token be a term?}
|
|
|
|
opStack: tokenPtr; {operation stack}
|
|
|
|
parenCount: integer; {# of open parenthesis}
|
|
|
|
stack: tokenPtr; {operand stack}
|
2022-10-01 21:28:16 -05:00
|
|
|
tType: typePtr; {type for cast/sizeof/etc.}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
op,sp: tokenPtr; {work pointers}
|
|
|
|
|
|
|
|
|
|
|
|
procedure ComplexTerm;
|
|
|
|
|
|
|
|
{ handle complex terms }
|
|
|
|
|
|
|
|
var
|
|
|
|
done: boolean; {for loop termination}
|
|
|
|
namePtr: stringPtr; {name of struct/union fields}
|
|
|
|
sp,tp,tm: tokenPtr; {work pointers}
|
|
|
|
|
|
|
|
begin {ComplexTerm}
|
|
|
|
while token.kind in
|
|
|
|
[lbrackch,lparench,dotch,minusgtop,plusplusop,minusminusop] do begin
|
|
|
|
case token.kind of
|
|
|
|
|
|
|
|
lbrackch: begin {subscripting}
|
|
|
|
NextToken; {skip the '['}
|
|
|
|
new(sp); {evaluate the subscript}
|
|
|
|
sp^.token.kind := plusch;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := stack;
|
|
|
|
stack := stack^.next;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := ExpressionTree(normalExpression, [rbrackch]);
|
|
|
|
sp^.next := stack;
|
|
|
|
stack := sp;
|
|
|
|
Match(rbrackch,24); {skip the ']'}
|
|
|
|
new(sp); {resolve the pointer}
|
|
|
|
sp^.token.kind := uasterisk;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := stack;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
sp^.next := stack^.next;
|
|
|
|
stack := sp;
|
|
|
|
end;
|
|
|
|
|
|
|
|
lparench: begin {function call}
|
|
|
|
NextToken;
|
|
|
|
new(sp); {create a parameter list terminator}
|
|
|
|
sp^.token.kind := parameteroper;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
sp^.next := stack;
|
|
|
|
stack := sp;
|
|
|
|
if token.kind <> rparench {evaluate the parameters}
|
|
|
|
then begin
|
|
|
|
done := false;
|
|
|
|
repeat
|
|
|
|
if token.kind in [rparench,eofsy] then begin
|
|
|
|
done := true;
|
|
|
|
Error(35);
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
new(sp);
|
|
|
|
sp^.token.kind := parameteroper;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle :=
|
|
|
|
ExpressionTree(normalExpression, [rparench,commach]);
|
|
|
|
sp^.right := stack;
|
|
|
|
sp^.next := stack^.next;
|
|
|
|
stack := sp;
|
|
|
|
if token.kind = commach then
|
|
|
|
NextToken
|
|
|
|
else
|
|
|
|
done := true;
|
|
|
|
end; {else}
|
|
|
|
until done;
|
|
|
|
end; {if}
|
|
|
|
sp := stack;
|
|
|
|
stack := sp^.next;
|
|
|
|
sp^.left := stack;
|
|
|
|
sp^.next := stack^.next;
|
|
|
|
stack := sp;
|
|
|
|
Match(rparench,12);
|
|
|
|
end;
|
|
|
|
|
|
|
|
dotch,minusgtop: begin {direct and indirect selection}
|
|
|
|
if token.kind = minusgtop then begin
|
|
|
|
new(sp); {e->name == (*e).name}
|
|
|
|
sp^.token.kind := uasterisk;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := stack;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
sp^.next := stack^.next;
|
|
|
|
stack := sp;
|
|
|
|
token.kind := dotch;
|
|
|
|
token.class := reservedSymbol;
|
|
|
|
end; {if}
|
|
|
|
new(sp); {create a record for the selection operator}
|
|
|
|
sp^.token := token;
|
|
|
|
sp^.left := stack;
|
|
|
|
stack := stack^.next;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
sp^.next := stack;
|
|
|
|
stack := sp;
|
|
|
|
NextToken; {skip the operator}
|
|
|
|
if token.kind in [ident,typedef] then begin
|
|
|
|
namePtr := token.name; {record the name}
|
|
|
|
new(sp); {record the selection field}
|
|
|
|
sp^.token := token;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
stack^.right := sp; {this becomes the right opnd}
|
|
|
|
NextToken; {skip the field name}
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(9);
|
|
|
|
end;
|
|
|
|
|
|
|
|
plusplusop: begin {postfix ++}
|
|
|
|
NextToken;
|
|
|
|
new(sp);
|
|
|
|
sp^.token.kind := opplusplus;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := stack;
|
|
|
|
stack := stack^.next;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
sp^.next := stack;
|
|
|
|
stack := sp;
|
|
|
|
end;
|
|
|
|
|
|
|
|
minusminusop: begin {postfix --}
|
|
|
|
NextToken;
|
|
|
|
new(sp);
|
|
|
|
sp^.token.kind := opminusminus;
|
|
|
|
sp^.token.class := reservedSymbol;
|
|
|
|
sp^.left := stack;
|
|
|
|
stack := stack^.next;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
sp^.next := stack;
|
|
|
|
stack := sp;
|
|
|
|
end;
|
|
|
|
|
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
|
|
|
end; {while}
|
|
|
|
end; {ComplexTerm}
|
|
|
|
|
|
|
|
|
|
|
|
procedure DoOperand;
|
|
|
|
|
|
|
|
{ process an operand }
|
|
|
|
|
|
|
|
label 1,2;
|
|
|
|
|
|
|
|
var
|
|
|
|
fnPtr: typePtr; {for defining functions on the fly}
|
|
|
|
fToken: tokenType; {used to save function name token}
|
|
|
|
id: identPtr; {pointer to an id's symbol table entry}
|
|
|
|
np: stringPtr; {for forming global names}
|
|
|
|
sp: tokenPtr; {work pointer}
|
|
|
|
|
|
|
|
begin {DoOperand}
|
|
|
|
{create an operand on the stack}
|
|
|
|
new(sp);
|
|
|
|
sp^.token := token;
|
|
|
|
sp^.next := stack;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
stack := sp;
|
|
|
|
|
|
|
|
{handle the preprocessor 'defined' function}
|
|
|
|
if kind = preprocessorExpression then
|
|
|
|
if token.name^ = 'defined' then begin
|
|
|
|
expandMacros := false;
|
|
|
|
NextToken;
|
|
|
|
sp^.token.kind := intconst;
|
|
|
|
sp^.token.class := intConstant;
|
|
|
|
if token.kind in [ident,typedef] then begin
|
|
|
|
sp^.token.ival := ord(IsDefined(token.name));
|
|
|
|
NextToken;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
Match(lparench, 13);
|
|
|
|
if token.kind in [ident,typedef] then begin
|
|
|
|
sp^.token.ival := ord(IsDefined(token.name));
|
|
|
|
NextToken;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
Error(9);
|
|
|
|
sp^.token.ival := 0;
|
|
|
|
end; {else}
|
|
|
|
Match(rparench, 12);
|
|
|
|
end; {else}
|
|
|
|
expandMacros := true;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
|
|
|
|
{check for illegal use}
|
|
|
|
id := FindSymbol(token, variableSpace, false, true);
|
|
|
|
if not (kind in
|
|
|
|
[normalExpression,initializerExpression,autoInitializerExpression])
|
|
|
|
then begin
|
|
|
|
if id <> nil then
|
|
|
|
if id^.itype^.kind = enumConst then
|
|
|
|
goto 2;
|
|
|
|
if kind <> preprocessorExpression then begin
|
|
|
|
op := opStack;
|
|
|
|
while op <> nil do begin
|
|
|
|
if op^.token.kind = sizeofsy then
|
|
|
|
goto 2;
|
|
|
|
op := op^.next;
|
|
|
|
end; {while}
|
|
|
|
Error(41);
|
|
|
|
errorFound := true;
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
|
|
|
2:
|
|
|
|
{skip the name}
|
|
|
|
fToken := token;
|
|
|
|
NextToken;
|
|
|
|
|
2020-02-05 12:47:30 -06:00
|
|
|
{in the preprocessor, all identifiers (post macro replacement) become 0}
|
|
|
|
if kind = preprocessorExpression then begin
|
2021-02-25 21:42:54 -06:00
|
|
|
stack^.token.class := longlongConstant;
|
|
|
|
stack^.token.kind := longlongconst;
|
|
|
|
stack^.token.qval := longlong0;
|
2022-12-12 21:47:32 -06:00
|
|
|
id := nil;
|
2020-02-05 12:47:30 -06:00
|
|
|
end {if}
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
{if the id is not declared, create a function returning integer}
|
2020-02-05 12:47:30 -06:00
|
|
|
else if id = nil then begin
|
2021-03-02 20:01:32 -06:00
|
|
|
if (fToken.name^ = '__func__') and (functionTable <> nil) then
|
|
|
|
id := MakeFuncIdentifier
|
|
|
|
else if token.kind = lparench then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
fnPtr := pointer(GCalloc(sizeof(typeRecord)));
|
|
|
|
{fnPtr^.size := 0;}
|
|
|
|
{fnPtr^.saveDisp := 0;}
|
Record volatile and restrict qualifiers in types.
These are needed to correctly distinguish pointer types in _Generic. They should also be used for type compatibility checks in other contexts, but currently are not.
This also fixes a couple small problems related to type qualifiers:
*restrict was not allowed to appear after * in type-names
*volatile status was not properly recorded in sym files
Here is an example of using _Generic to distinguish pointer types based on the qualifiers of the pointed-to type:
#include <stdio.h>
#define f(e) _Generic((e),\
int * restrict *: 1,\
int * volatile const *: 2,\
int **: 3,\
default: 0)
#define g(e) _Generic((e),\
int *: 1,\
const int *: 2,\
volatile int *: 3,\
default: 0)
int main(void) {
int * restrict * p1;
int * volatile const * p2;
int * const * p3;
// should print "1 2 0 1"
printf("%i %i %i %i\n", f(p1), f(p2), f(p3), f((int * restrict *)0));
int *q1;
const int *q2;
volatile int *q3;
const volatile int *q4;
// should print "1 2 3 0"
printf("%i %i %i %i\n", g(q1), g(q2), g(q3), g(q4));
}
Here is an example of a problem resulting from volatile not being recorded in sym files (if a sym file was present, the read of x was lifted out of the loop):
#pragma optimize -1
static volatile int x;
#include <stdio.h>
int main(void) {
int y;
for (unsigned i = 0; i < 100; i++) {
y = x*2 + 7;
}
}
2021-08-29 21:10:20 -05:00
|
|
|
{fnPtr^.qualifiers := [];}
|
2017-10-21 18:40:19 -05:00
|
|
|
fnPtr^.kind := functionType;
|
2020-02-29 22:43:29 -06:00
|
|
|
fnPtr^.fType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
{fnPtr^.varargs := false;}
|
|
|
|
{fnPtr^.prototyped := false;}
|
|
|
|
{fnPtr^.overrideKR := false;}
|
|
|
|
{fnPtr^.parameterList := nil;}
|
|
|
|
{fnPtr^.isPascal := false;}
|
|
|
|
{fnPtr^.toolNum := 0;}
|
|
|
|
{fnPtr^.dispatcher := 0;}
|
|
|
|
np := pointer(GMalloc(length(fToken.name^)+1));
|
|
|
|
CopyString(pointer(np), pointer(fToken.name));
|
2022-11-19 23:02:50 -06:00
|
|
|
id := NewSymbol(np, fnPtr, ident, variableSpace, declared, false);
|
2020-01-29 18:33:19 -06:00
|
|
|
if ((lint & lintUndefFn) <> 0) or ((lint & lintC99Syntax) <> 0) then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(51);
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
Error(31);
|
|
|
|
errorFound := true;
|
|
|
|
end; {else}
|
2020-01-29 17:09:52 -06:00
|
|
|
end {if id = nil}
|
2017-10-21 18:40:19 -05:00
|
|
|
else if id^.itype^.kind = enumConst then begin
|
2019-03-31 18:40:14 -05:00
|
|
|
stack^.token.class := intConstant;
|
2017-10-21 18:40:19 -05:00
|
|
|
stack^.token.kind := intconst;
|
|
|
|
stack^.token.ival := id^.itype^.eval;
|
|
|
|
end; {else if}
|
2022-12-12 21:47:32 -06:00
|
|
|
|
|
|
|
if id <> nil then
|
|
|
|
id^.used := true;
|
2017-10-21 18:40:19 -05:00
|
|
|
stack^.id := id; {save the identifier}
|
|
|
|
ComplexTerm; {handle subscripts, selection, etc.}
|
|
|
|
1:
|
|
|
|
end; {DoOperand}
|
|
|
|
|
|
|
|
|
|
|
|
procedure Operation;
|
|
|
|
|
|
|
|
{ do an operation }
|
|
|
|
|
2021-03-06 00:57:13 -06:00
|
|
|
label 1,2,3,4;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
var
|
|
|
|
baseType: baseTypeEnum; {base type of value to cast}
|
|
|
|
class: tokenClass; {class of cast token}
|
|
|
|
ekind: tokenEnum; {kind of constant expression}
|
|
|
|
kindLeft, kindRight: tokenEnum; {kinds of operands}
|
|
|
|
lCodeGeneration: boolean; {local copy of codeGeneration}
|
|
|
|
op: tokenPtr; {work pointer}
|
|
|
|
op1,op2: longint; {for evaluating constant expressions}
|
2021-03-04 23:52:41 -06:00
|
|
|
rop1,rop2: extended; {for evaluating fp expressions}
|
2021-02-13 15:07:16 -06:00
|
|
|
llop1, llop2: longlong; {for evaluating long long expressions}
|
2017-10-21 18:40:19 -05:00
|
|
|
tp: typePtr; {cast type}
|
2016-12-19 00:24:47 -06:00
|
|
|
unsigned: boolean; {is the term unsigned?}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
|
|
|
|
function Pop: tokenPtr;
|
|
|
|
|
|
|
|
{ pop an operand, returning its pointer }
|
|
|
|
|
|
|
|
begin {Pop}
|
|
|
|
if stack = nil then begin
|
|
|
|
Error(36);
|
|
|
|
errorFound := true;
|
2019-12-23 14:12:44 -06:00
|
|
|
new(stack); {synthesize the missing token}
|
|
|
|
stack^.token.class := intConstant;
|
|
|
|
stack^.token.kind := intconst;
|
|
|
|
stack^.token.ival := 0;
|
|
|
|
stack^.next := nil;
|
|
|
|
stack^.left := nil;
|
|
|
|
stack^.middle := nil;
|
|
|
|
stack^.right := nil;
|
|
|
|
end; {if}
|
|
|
|
Pop := stack;
|
|
|
|
stack := stack^.next;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {Pop}
|
|
|
|
|
|
|
|
|
2021-02-21 18:43:53 -06:00
|
|
|
function RealVal (token: tokenType): extended;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{ convert an operand to a real value }
|
|
|
|
|
|
|
|
begin {RealVal}
|
2021-03-07 13:38:21 -06:00
|
|
|
if token.kind in [intconst,charconst,scharconst,ucharconst] then
|
2017-10-21 18:40:19 -05:00
|
|
|
RealVal := token.ival
|
2021-10-11 20:54:37 -05:00
|
|
|
else if token.kind in [uintconst,ushortconst] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
if token.ival < 0 then
|
|
|
|
RealVal := (token.ival & $7FFF) + 32768.0
|
|
|
|
else
|
|
|
|
RealVal := token.ival;
|
|
|
|
end {else if}
|
|
|
|
else if token.kind = longconst then
|
|
|
|
RealVal := token.lval
|
|
|
|
else if token.kind = ulongconst then begin
|
|
|
|
if token.lval < 0 then
|
|
|
|
RealVal := (token.lval & $7FFFFFFF) + 2147483648.0
|
|
|
|
else
|
|
|
|
RealVal := token.lval;
|
|
|
|
end {else if}
|
2021-02-21 18:43:53 -06:00
|
|
|
else if token.kind = longlongconst then
|
|
|
|
RealVal := CnvLLX(token.qval)
|
|
|
|
else if token.kind = ulonglongconst then
|
|
|
|
RealVal := CnvULLX(token.qval)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
RealVal := token.rval;
|
|
|
|
end; {RealVal}
|
|
|
|
|
|
|
|
|
|
|
|
function IntVal (token: tokenType): longint;
|
|
|
|
|
|
|
|
{ convert an operand to a longint value }
|
|
|
|
|
|
|
|
begin {IntVal}
|
2021-03-07 13:38:21 -06:00
|
|
|
if token.kind in [intconst,charconst,scharconst,ucharconst] then
|
2017-10-21 18:40:19 -05:00
|
|
|
IntVal := token.ival
|
2021-10-11 20:54:37 -05:00
|
|
|
else if token.kind in [uintconst,ushortconst] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
IntVal := token.ival & $0000FFFF;
|
|
|
|
end {else if}
|
2016-12-19 00:24:47 -06:00
|
|
|
else {if token.kind in [longconst,ulongconst] then} begin
|
2017-10-21 18:40:19 -05:00
|
|
|
IntVal := token.lval;
|
|
|
|
end; {else}
|
|
|
|
end; {IntVal}
|
|
|
|
|
|
|
|
|
2021-02-13 15:07:16 -06:00
|
|
|
procedure GetLongLongVal (var result: longlong; token: tokenType);
|
|
|
|
|
|
|
|
{ convert an operand to a long long value }
|
|
|
|
|
|
|
|
begin {LongLongVal}
|
2021-03-07 13:38:21 -06:00
|
|
|
if token.kind in [intconst,charconst,scharconst,ucharconst] then begin
|
2021-02-13 15:07:16 -06:00
|
|
|
result.lo := token.ival;
|
|
|
|
if result.lo < 0 then
|
|
|
|
result.hi := -1
|
|
|
|
else
|
|
|
|
result.hi := 0;
|
|
|
|
end {if}
|
2021-10-11 20:54:37 -05:00
|
|
|
else if token.kind in [uintconst,ushortconst] then begin
|
2021-02-13 15:07:16 -06:00
|
|
|
result.lo := token.ival & $0000FFFF;
|
|
|
|
result.hi := 0;
|
|
|
|
end {else if}
|
|
|
|
else if token.kind = longconst then begin
|
|
|
|
result.lo := token.lval;
|
|
|
|
if result.lo < 0 then
|
|
|
|
result.hi := -1
|
|
|
|
else
|
|
|
|
result.hi := 0;
|
|
|
|
end {else if}
|
|
|
|
else if token.kind = ulongconst then begin
|
|
|
|
result.lo := token.lval;
|
|
|
|
result.hi := 0;
|
|
|
|
end {else if}
|
|
|
|
else {if token.kind in [longlongconst,ulonglongconst] then} begin
|
|
|
|
result := token.qval;
|
|
|
|
end; {else}
|
|
|
|
end; {LongLongVal}
|
|
|
|
|
|
|
|
|
2016-12-20 20:30:05 -06:00
|
|
|
function PPKind (token: tokenType): tokenEnum;
|
|
|
|
|
|
|
|
{ adjust kind of token for use in preprocessor expression }
|
|
|
|
|
|
|
|
begin {PPKind}
|
2021-02-16 23:45:59 -06:00
|
|
|
if token.kind in [intconst,longconst] then
|
|
|
|
PPKind := longlongconst
|
2021-10-11 20:54:37 -05:00
|
|
|
else if token.kind in [uintconst,ushortconst,ulongconst] then
|
2021-02-16 23:45:59 -06:00
|
|
|
PPKind := ulonglongconst
|
2016-12-20 20:30:05 -06:00
|
|
|
else
|
|
|
|
PPKind := token.kind;
|
|
|
|
end; {PPKind}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
begin {Operation}
|
|
|
|
op := opStack; {pop the operation}
|
|
|
|
opStack := op^.next;
|
|
|
|
case op^.token.kind of
|
|
|
|
|
|
|
|
commach: begin {,}
|
|
|
|
op^.right := Pop;
|
|
|
|
op^.left := Pop;
|
|
|
|
end;
|
|
|
|
|
|
|
|
eqch, {=}
|
|
|
|
pluseqop, {+=}
|
|
|
|
minuseqop, {-=}
|
|
|
|
asteriskeqop, {*=}
|
|
|
|
slasheqop, {/=}
|
|
|
|
percenteqop, {%=}
|
|
|
|
ltlteqop, {<<=}
|
|
|
|
gtgteqop, {>>=}
|
|
|
|
andeqop, {&=}
|
|
|
|
caroteqop, {^=}
|
|
|
|
bareqop: begin {|=}
|
|
|
|
op^.right := Pop;
|
|
|
|
op^.left := Pop;
|
|
|
|
end;
|
|
|
|
|
|
|
|
colonch: begin {? :}
|
|
|
|
op^.right := Pop;
|
|
|
|
op^.middle := Pop;
|
|
|
|
op^.left := Pop;
|
2021-10-11 20:54:37 -05:00
|
|
|
if op^.right^.token.kind in [intconst,uintconst,ushortconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
longconst,ulongconst,longlongconst,ulonglongconst,
|
|
|
|
charconst,scharconst,ucharconst] then
|
2021-10-11 20:54:37 -05:00
|
|
|
if op^.left^.token.kind in [intconst,uintconst,ushortconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
longconst,ulongconst,longlongconst,ulonglongconst,
|
|
|
|
charconst,scharconst,ucharconst] then
|
2021-10-11 20:54:37 -05:00
|
|
|
if op^.middle^.token.kind in [intconst,uintconst,ushortconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
longconst,ulongconst,longlongconst,ulonglongconst,
|
|
|
|
charconst,scharconst,ucharconst] then begin
|
2022-07-04 22:30:25 -05:00
|
|
|
|
|
|
|
kindLeft := op^.middle^.token.kind;
|
|
|
|
kindRight := op^.right^.token.kind;
|
|
|
|
|
|
|
|
{do the usual binary conversions}
|
|
|
|
if (kindRight = ulonglongconst) or (kindLeft = ulonglongconst) then
|
|
|
|
ekind := ulonglongconst
|
|
|
|
else if (kindRight = longlongconst) or (kindLeft = longlongconst) then
|
|
|
|
ekind := longlongconst
|
|
|
|
else if (kindRight = ulongconst) or (kindLeft = ulongconst) then
|
|
|
|
ekind := ulongconst
|
|
|
|
else if (kindRight = longconst) or (kindLeft = longconst) then
|
|
|
|
ekind := longconst
|
|
|
|
else if (kindRight = uintconst) or (kindLeft = uintconst)
|
|
|
|
or (kindRight = ushortconst) or (kindLeft = ushortconst) then
|
|
|
|
ekind := uintconst
|
|
|
|
else
|
|
|
|
ekind := intconst;
|
|
|
|
|
2021-02-19 23:46:57 -06:00
|
|
|
GetLongLongVal(llop1, op^.left^.token);
|
|
|
|
if (llop1.lo <> 0) or (llop1.hi <> 0) then
|
2022-07-04 22:30:25 -05:00
|
|
|
GetLongLongVal(llop2, op^.middle^.token)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
2022-07-04 22:30:25 -05:00
|
|
|
GetLongLongVal(llop2, op^.right^.token);
|
|
|
|
op^.token.kind := ekind;
|
|
|
|
if ekind in [longlongconst,ulonglongconst] then begin
|
|
|
|
op^.token.qval := llop2;
|
|
|
|
op^.token.class := longlongConstant;
|
|
|
|
end {if}
|
|
|
|
else if ekind in [longconst,ulongconst] then begin
|
|
|
|
op^.token.lval := llop2.lo;
|
|
|
|
op^.token.class := longConstant;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
op^.token.ival := long(llop2.lo).lsw;
|
|
|
|
op^.token.class := intConstant;
|
|
|
|
end; {else}
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
dispose(op^.left);
|
|
|
|
dispose(op^.right);
|
|
|
|
dispose(op^.middle);
|
|
|
|
op^.left := nil;
|
|
|
|
op^.right := nil;
|
|
|
|
op^.middle := nil;
|
|
|
|
end; {if}
|
|
|
|
end;
|
|
|
|
|
|
|
|
questionch: begin {error -> ? should not be unmatched}
|
|
|
|
Error(29);
|
|
|
|
errorFound := true;
|
|
|
|
end;
|
|
|
|
|
|
|
|
barbarop, {||}
|
|
|
|
andandop, {&&}
|
|
|
|
carotch, {^}
|
|
|
|
barch, {|}
|
|
|
|
andch, {&}
|
|
|
|
eqeqop, {==}
|
|
|
|
exceqop, {!=}
|
|
|
|
ltch, {<}
|
|
|
|
gtch, {>}
|
|
|
|
lteqop, {<=}
|
|
|
|
gteqop, {>=}
|
|
|
|
ltltop, {<<}
|
|
|
|
gtgtop, {>>}
|
|
|
|
plusch, {+}
|
|
|
|
minusch, {-}
|
|
|
|
asteriskch, {*}
|
|
|
|
slashch, {/}
|
|
|
|
percentch: begin {%}
|
|
|
|
op^.right := Pop;
|
|
|
|
op^.left := Pop;
|
|
|
|
kindRight := op^.right^.token.kind;
|
|
|
|
kindLeft := op^.left^.token.kind;
|
2021-10-11 20:54:37 -05:00
|
|
|
if kindRight in [intconst,uintconst,ushortconst,longconst,ulongconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
charconst,scharconst,ucharconst] then begin
|
2021-10-11 20:54:37 -05:00
|
|
|
if kindLeft in [intconst,uintconst,ushortconst,longconst,ulongconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
charconst,scharconst,ucharconst] then begin
|
2021-02-16 23:45:59 -06:00
|
|
|
if kind = preprocessorExpression then
|
|
|
|
goto 2;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{do the usual binary conversions}
|
|
|
|
if (kindRight = ulongconst) or (kindLeft = ulongconst) then
|
|
|
|
ekind := ulongconst
|
|
|
|
else if (kindRight = longconst) or (kindLeft = longconst) then
|
|
|
|
ekind := longconst
|
2021-10-11 20:54:37 -05:00
|
|
|
else if (kindRight = uintconst) or (kindLeft = uintconst)
|
|
|
|
or (kindRight = ushortconst) or (kindLeft = ushortconst) then
|
2017-10-21 18:40:19 -05:00
|
|
|
ekind := uintconst
|
|
|
|
else
|
|
|
|
ekind := intconst;
|
|
|
|
|
|
|
|
{evaluate a constant operation}
|
2016-12-19 00:24:47 -06:00
|
|
|
unsigned := ekind in [uintconst,ulongconst];
|
2017-10-21 18:40:19 -05:00
|
|
|
op1 := IntVal(op^.left^.token);
|
|
|
|
op2 := IntVal(op^.right^.token);
|
|
|
|
dispose(op^.right);
|
|
|
|
op^.right := nil;
|
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
|
|
|
case op^.token.kind of
|
2016-12-20 22:40:24 -06:00
|
|
|
barbarop : begin {||}
|
|
|
|
op1 := ord((op1 <> 0) or (op2 <> 0));
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
andandop : begin {&&}
|
|
|
|
op1 := ord((op1 <> 0) and (op2 <> 0));
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
2017-10-21 18:40:19 -05:00
|
|
|
carotch : op1 := op1 ! op2; {^}
|
|
|
|
barch : op1 := op1 | op2; {|}
|
|
|
|
andch : op1 := op1 & op2; {&}
|
|
|
|
eqeqop : begin {==}
|
|
|
|
op1 := ord(op1 = op2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
exceqop : begin {!=}
|
|
|
|
op1 := ord(op1 <> op2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
ltch : begin {<}
|
|
|
|
if unsigned then
|
|
|
|
op1 := ult(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := ord(op1 < op2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
gtch : begin {>}
|
|
|
|
if unsigned then
|
|
|
|
op1 := ugt(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := ord(op1 > op2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
lteqop : begin {<=}
|
|
|
|
if unsigned then
|
|
|
|
op1 := ule(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := ord(op1 <= op2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
gteqop : begin {>=}
|
|
|
|
if unsigned then
|
|
|
|
op1 := uge(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := ord(op1 >= op2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
2016-11-07 18:57:40 -06:00
|
|
|
ltltop : begin {<<}
|
|
|
|
op1 := op1 << op2;
|
|
|
|
ekind := kindLeft;
|
|
|
|
end;
|
|
|
|
gtgtop : begin {>>}
|
2021-10-11 20:54:37 -05:00
|
|
|
if kindLeft in [uintconst,ushortconst,ulongconst]
|
|
|
|
then
|
2017-10-21 18:40:19 -05:00
|
|
|
op1 := lshr(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := op1 >> op2;
|
2016-11-07 18:57:40 -06:00
|
|
|
ekind := kindLeft;
|
|
|
|
end;
|
2017-10-21 18:40:19 -05:00
|
|
|
plusch : op1 := op1 + op2; {+}
|
|
|
|
minusch : op1 := op1 - op2; {-}
|
|
|
|
asteriskch : if unsigned then {*}
|
|
|
|
op1 := umul(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := op1 * op2;
|
|
|
|
slashch : begin {/}
|
|
|
|
if op2 = 0 then begin
|
2024-02-02 19:52:36 -06:00
|
|
|
if not (kind in [normalExpression,
|
|
|
|
autoInitializerExpression]) then
|
|
|
|
Error(109)
|
|
|
|
else if ((lint & lintOverflow) <> 0) then
|
|
|
|
Error(129);
|
2017-10-21 18:40:19 -05:00
|
|
|
op2 := 1;
|
|
|
|
end; {if}
|
|
|
|
if unsigned then
|
|
|
|
op1 := udiv(op1,op2)
|
|
|
|
else
|
|
|
|
op1 := op1 div op2;
|
|
|
|
end;
|
|
|
|
percentch : begin {%}
|
Make % operator give proper remainders even if one or both operands are negative.
Per the C standards, the % operator should give a remainder after division, such that (a/b)*b + a%b equals a (provided that a/b is representable). As such, the operation of % is defined for cases where either or both of the operands are negative. Since division truncates toward 0, a%b should give a negative result (or 0) in cases where a is negative.
Previously, the % operator was essentially behaving like the "mod" operator in Pascal, which is equivalent for positive operands but not if either operand is negative. It would generally give incorrect results in those cases, or in some cases give compile-time or run-time errors.
This patch addresses both 16-bit and 32-bit signed computations at run time, and operations in constant expressions. The approach at run time is to call existing division routines, which return the correct remainder, except always as a positive number. The generated code checks the sign of the first operand, and if it is negative negates the remainder.
The code generated is somewhat large (especially for the 32-bit case), so it might be sensible to put it in a library function and call that, but for now it's just generated in-line. This avoids introducing a dependency on a new library function, so the generated code remains compatible with older versions of ORCALib (e.g. the GNO one).
Fixes #10.
2018-09-10 17:26:20 -05:00
|
|
|
if op2 = 0 then begin
|
2024-02-02 19:52:36 -06:00
|
|
|
if not (kind in [normalExpression,
|
|
|
|
autoInitializerExpression]) then
|
|
|
|
Error(109)
|
|
|
|
else if ((lint & lintOverflow) <> 0) then
|
|
|
|
Error(129);
|
Make % operator give proper remainders even if one or both operands are negative.
Per the C standards, the % operator should give a remainder after division, such that (a/b)*b + a%b equals a (provided that a/b is representable). As such, the operation of % is defined for cases where either or both of the operands are negative. Since division truncates toward 0, a%b should give a negative result (or 0) in cases where a is negative.
Previously, the % operator was essentially behaving like the "mod" operator in Pascal, which is equivalent for positive operands but not if either operand is negative. It would generally give incorrect results in those cases, or in some cases give compile-time or run-time errors.
This patch addresses both 16-bit and 32-bit signed computations at run time, and operations in constant expressions. The approach at run time is to call existing division routines, which return the correct remainder, except always as a positive number. The generated code checks the sign of the first operand, and if it is negative negates the remainder.
The code generated is somewhat large (especially for the 32-bit case), so it might be sensible to put it in a library function and call that, but for now it's just generated in-line. This avoids introducing a dependency on a new library function, so the generated code remains compatible with older versions of ORCALib (e.g. the GNO one).
Fixes #10.
2018-09-10 17:26:20 -05:00
|
|
|
op2 := 1;
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
if unsigned then
|
|
|
|
op1 := umod(op1,op2)
|
|
|
|
else
|
Make % operator give proper remainders even if one or both operands are negative.
Per the C standards, the % operator should give a remainder after division, such that (a/b)*b + a%b equals a (provided that a/b is representable). As such, the operation of % is defined for cases where either or both of the operands are negative. Since division truncates toward 0, a%b should give a negative result (or 0) in cases where a is negative.
Previously, the % operator was essentially behaving like the "mod" operator in Pascal, which is equivalent for positive operands but not if either operand is negative. It would generally give incorrect results in those cases, or in some cases give compile-time or run-time errors.
This patch addresses both 16-bit and 32-bit signed computations at run time, and operations in constant expressions. The approach at run time is to call existing division routines, which return the correct remainder, except always as a positive number. The generated code checks the sign of the first operand, and if it is negative negates the remainder.
The code generated is somewhat large (especially for the 32-bit case), so it might be sensible to put it in a library function and call that, but for now it's just generated in-line. This avoids introducing a dependency on a new library function, so the generated code remains compatible with older versions of ORCALib (e.g. the GNO one).
Fixes #10.
2018-09-10 17:26:20 -05:00
|
|
|
op1 := op1 - (op1 div op2) * op2;
|
2017-10-21 18:40:19 -05:00
|
|
|
end;
|
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((lint & lintOverflow) <> 0) then begin
|
|
|
|
if op^.token.kind in [plusch,minusch,asteriskch,slashch] then
|
|
|
|
if ekind = intConst then
|
|
|
|
if op1 <> long(op1).lsw then
|
|
|
|
Error(128);
|
|
|
|
if op^.token.kind in [ltltop,gtgtop] then begin
|
|
|
|
if ekind in [intConst,uintConst] then
|
|
|
|
if (op2 < 0) or (op2 > 15) then
|
|
|
|
Error(130);
|
|
|
|
if ekind in [longConst,ulongConst] then
|
|
|
|
if (op2 < 0) or (op2 > 31) then
|
|
|
|
Error(130);
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.kind := ekind;
|
|
|
|
if ekind in [longconst,ulongconst] then begin
|
|
|
|
op^.token.lval := op1;
|
|
|
|
op^.token.class := longConstant;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
op^.token.ival := long(op1).lsw;
|
|
|
|
op^.token.class := intConstant;
|
|
|
|
end; {else}
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
2021-02-16 23:45:59 -06:00
|
|
|
2:
|
2021-10-11 20:54:37 -05:00
|
|
|
if kindRight in [intconst,uintconst,ushortconst,longconst,ulongconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
longlongconst,ulonglongconst,charconst,scharconst,ucharconst]
|
|
|
|
then begin
|
2021-10-11 20:54:37 -05:00
|
|
|
if kindLeft in [intconst,uintconst,ushortconst,longconst,ulongconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
longlongconst,ulonglongconst,charconst,scharconst,ucharconst]
|
|
|
|
then begin
|
2021-02-13 15:07:16 -06:00
|
|
|
|
|
|
|
if kind = preprocessorExpression then begin
|
|
|
|
kindLeft := PPKind(op^.left^.token);
|
|
|
|
kindRight := PPKind(op^.right^.token);
|
|
|
|
end; {if}
|
|
|
|
|
|
|
|
{do the usual binary conversions}
|
|
|
|
if (kindRight = ulonglongconst) or (kindLeft = ulonglongconst) then
|
|
|
|
ekind := ulonglongconst
|
|
|
|
else
|
|
|
|
ekind := longlongconst;
|
|
|
|
|
2021-02-14 20:39:22 -06:00
|
|
|
unsigned := ekind = ulonglongconst;
|
2021-02-13 15:07:16 -06:00
|
|
|
GetLongLongVal(llop1, op^.left^.token);
|
|
|
|
GetLongLongVal(llop2, op^.right^.token);
|
|
|
|
|
|
|
|
case op^.token.kind of
|
|
|
|
barbarop : begin {||}
|
2021-02-14 20:39:22 -06:00
|
|
|
llop1.lo :=
|
|
|
|
ord((llop1.lo <> 0) or (llop1.hi <> 0) or
|
|
|
|
(llop2.lo <> 0) or (llop2.hi <> 0));
|
2021-03-01 18:13:16 -05:00
|
|
|
llop1.hi := 0;
|
2021-02-13 15:07:16 -06:00
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
andandop : begin {&&}
|
2021-02-14 20:39:22 -06:00
|
|
|
llop1.lo :=
|
|
|
|
ord(((llop1.lo <> 0) or (llop1.hi <> 0)) and
|
|
|
|
((llop2.lo <> 0) or (llop2.hi <> 0)));
|
2021-03-01 18:13:16 -05:00
|
|
|
llop1.hi := 0;
|
2021-02-13 15:07:16 -06:00
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
carotch : begin {^}
|
|
|
|
llop1.lo := llop1.lo ! llop2.lo;
|
|
|
|
llop1.hi := llop1.hi ! llop2.hi;
|
|
|
|
end;
|
|
|
|
barch : begin {|}
|
|
|
|
llop1.lo := llop1.lo | llop2.lo;
|
|
|
|
llop1.hi := llop1.hi | llop2.hi;
|
|
|
|
end;
|
|
|
|
andch : begin {&}
|
|
|
|
llop1.lo := llop1.lo & llop2.lo;
|
|
|
|
llop1.hi := llop1.hi & llop2.hi;
|
|
|
|
end;
|
|
|
|
eqeqop : begin {==}
|
2021-02-14 20:39:22 -06:00
|
|
|
llop1.lo := ord((llop1.lo = llop2.lo) and
|
|
|
|
(llop1.hi = llop2.hi));
|
2021-03-01 18:13:16 -05:00
|
|
|
llop1.hi := 0;
|
2021-02-13 15:07:16 -06:00
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
exceqop : begin {!=}
|
2021-02-14 20:39:22 -06:00
|
|
|
llop1.lo := ord((llop1.lo <> llop2.lo) or
|
|
|
|
(llop1.hi <> llop2.hi));
|
2021-03-01 18:13:16 -05:00
|
|
|
llop1.hi := 0;
|
2021-02-13 15:07:16 -06:00
|
|
|
ekind := intconst;
|
|
|
|
end;
|
2021-02-16 23:11:41 -06:00
|
|
|
ltch : begin {<}
|
|
|
|
if unsigned then
|
|
|
|
llop1.lo := ult64(llop1, llop2)
|
|
|
|
else
|
|
|
|
llop1.lo := slt64(llop1, llop2);
|
|
|
|
llop1.hi := 0;
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
gtch : begin {>}
|
|
|
|
if unsigned then
|
|
|
|
llop1.lo := ugt64(llop1, llop2)
|
|
|
|
else
|
|
|
|
llop1.lo := sgt64(llop1, llop2);
|
|
|
|
llop1.hi := 0;
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
lteqop : begin {<=}
|
|
|
|
if unsigned then
|
|
|
|
llop1.lo := ule64(llop1, llop2)
|
|
|
|
else
|
|
|
|
llop1.lo := sle64(llop1, llop2);
|
|
|
|
llop1.hi := 0;
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
2021-02-14 20:39:22 -06:00
|
|
|
gteqop : begin {>=}
|
2021-02-16 23:11:41 -06:00
|
|
|
if unsigned then
|
|
|
|
llop1.lo := uge64(llop1, llop2)
|
|
|
|
else
|
|
|
|
llop1.lo := sge64(llop1, llop2);
|
|
|
|
llop1.hi := 0;
|
|
|
|
ekind := intconst;
|
2021-02-14 20:39:22 -06:00
|
|
|
end;
|
|
|
|
ltltop : begin {<<}
|
|
|
|
shl64(llop1, long(llop2.lo).lsw);
|
|
|
|
ekind := kindLeft;
|
|
|
|
end;
|
|
|
|
gtgtop : begin {>>}
|
|
|
|
if kindleft = ulonglongconst then
|
|
|
|
lshr64(llop1, long(llop2.lo).lsw)
|
|
|
|
else
|
|
|
|
ashr64(llop1, long(llop2.lo).lsw);
|
|
|
|
ekind := kindLeft;
|
|
|
|
end;
|
|
|
|
plusch : add64(llop1, llop2); {+}
|
|
|
|
minusch : sub64(llop1, llop2); {-}
|
|
|
|
asteriskch : umul64(llop1, llop2); {*}
|
|
|
|
slashch : begin {/}
|
|
|
|
if (llop2.lo = 0) and (llop2.hi = 0) then begin
|
2024-02-02 19:52:36 -06:00
|
|
|
if not (kind in [normalExpression,
|
|
|
|
autoInitializerExpression]) then
|
|
|
|
Error(109)
|
|
|
|
else if ((lint & lintOverflow) <> 0) then
|
|
|
|
Error(129);
|
2021-02-14 20:39:22 -06:00
|
|
|
llop2 := longlong1;
|
|
|
|
end; {if}
|
|
|
|
if unsigned then
|
|
|
|
udiv64(llop1, llop2)
|
|
|
|
else
|
|
|
|
div64(llop1, llop2);
|
|
|
|
end;
|
|
|
|
percentch : begin {%}
|
|
|
|
if (llop2.lo = 0) and (llop2.hi = 0) then begin
|
2024-02-02 19:52:36 -06:00
|
|
|
if not (kind in [normalExpression,
|
|
|
|
autoInitializerExpression]) then
|
|
|
|
Error(109)
|
|
|
|
else if ((lint & lintOverflow) <> 0) then
|
|
|
|
Error(129);
|
2021-02-14 20:39:22 -06:00
|
|
|
llop2 := longlong1;
|
|
|
|
end; {if}
|
|
|
|
if unsigned then
|
|
|
|
umod64(llop1, llop2)
|
|
|
|
else
|
|
|
|
rem64(llop1, llop2);
|
|
|
|
end;
|
2021-02-13 15:07:16 -06:00
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
2021-02-13 21:14:26 -06:00
|
|
|
|
|
|
|
dispose(op^.right);
|
|
|
|
op^.right := nil;
|
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
2021-02-13 15:07:16 -06:00
|
|
|
op^.token.kind := ekind;
|
|
|
|
if ekind in [longlongconst,ulonglongconst] then begin
|
|
|
|
op^.token.qval := llop1;
|
|
|
|
op^.token.class := longlongConstant;
|
|
|
|
end {if}
|
2021-02-14 20:39:22 -06:00
|
|
|
else if ekind in [longconst,ulongconst] then begin
|
|
|
|
op^.token.lval := llop1.lo;
|
|
|
|
op^.token.class := longConstant;
|
|
|
|
end {if}
|
2021-02-13 15:07:16 -06:00
|
|
|
else begin
|
2021-02-14 20:39:22 -06:00
|
|
|
op^.token.ival := long(llop1.lo).lsw;
|
2021-02-13 15:07:16 -06:00
|
|
|
op^.token.class := intConstant;
|
|
|
|
end; {else}
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
2021-02-13 21:14:26 -06:00
|
|
|
|
2021-02-21 18:43:53 -06:00
|
|
|
if op^.right^.token.kind in [intconst,uintconst,longconst,ulongconst,
|
2021-03-07 00:48:51 -06:00
|
|
|
longlongconst,ulonglongconst,floatconst,doubleconst,extendedconst,
|
2021-10-11 20:54:37 -05:00
|
|
|
compconst,charconst,scharconst,ucharconst,ushortconst] then
|
2021-02-21 18:43:53 -06:00
|
|
|
if op^.left^.token.kind in [intconst,uintconst,longconst,ulongconst,
|
2021-03-07 00:48:51 -06:00
|
|
|
longlongconst,ulonglongconst,floatconst,doubleconst,extendedconst,
|
2021-10-11 20:54:37 -05:00
|
|
|
compconst,charconst,scharconst,ucharconst,ushortconst] then
|
2017-10-21 18:40:19 -05:00
|
|
|
begin
|
2021-03-06 00:57:13 -06:00
|
|
|
if fenvAccess then
|
|
|
|
if kind in [normalExpression, autoInitializerExpression] then
|
|
|
|
goto 1;
|
2021-03-07 00:48:51 -06:00
|
|
|
if (op^.right^.token.kind = compConst)
|
|
|
|
and (op^.left^.token.kind = compConst) then
|
|
|
|
ekind := compconst
|
|
|
|
else if (op^.right^.token.kind in [extendedConst,compConst])
|
|
|
|
or (op^.left^.token.kind in [extendedConst,compConst]) then
|
|
|
|
ekind := extendedconst
|
|
|
|
else if (op^.right^.token.kind = doubleConst)
|
|
|
|
or (op^.left^.token.kind = doubleConst) then
|
|
|
|
ekind := doubleconst
|
|
|
|
else
|
|
|
|
ekind := floatconst;
|
2021-03-04 23:52:41 -06:00
|
|
|
rop1 := RealVal(op^.left^.token);
|
|
|
|
rop2 := RealVal(op^.right^.token);
|
2017-10-21 18:40:19 -05:00
|
|
|
dispose(op^.right);
|
|
|
|
op^.right := nil;
|
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
|
|
|
case op^.token.kind of
|
2016-12-20 22:40:24 -06:00
|
|
|
barbarop : begin {||}
|
|
|
|
op1 := ord((rop1 <> 0.0) or (rop2 <> 0.0));
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
andandop : begin {&&}
|
|
|
|
op1 := ord((rop1 <> 0.0) and (rop2 <> 0.0));
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
2017-10-21 18:40:19 -05:00
|
|
|
eqeqop : begin {==}
|
|
|
|
op1 := ord(rop1 = rop2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
exceqop : begin {!=}
|
|
|
|
op1 := ord(rop1 <> rop2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
ltch : begin {<}
|
|
|
|
op1 := ord(rop1 < rop2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
gtch : begin {>}
|
|
|
|
op1 := ord(rop1 > rop2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
lteqop : begin {<=}
|
|
|
|
op1 := ord(rop1 <= rop2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
gteqop : begin {>=}
|
|
|
|
op1 := ord(rop1 >= rop2);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
plusch : rop1 := rop1 + rop2; {+}
|
|
|
|
minusch : rop1 := rop1 - rop2; {-}
|
|
|
|
asteriskch : rop1 := rop1 * rop2; {*}
|
2019-03-10 16:47:18 -04:00
|
|
|
slashch : rop1 := rop1 / rop2; {/}
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise : Error(66); {illegal operation}
|
|
|
|
end; {case}
|
|
|
|
if ekind = intconst then begin
|
|
|
|
op^.token.ival := long(op1).lsw;
|
|
|
|
op^.token.class := intConstant;
|
|
|
|
op^.token.kind := intConst;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
op^.token.rval := rop1;
|
2021-03-04 23:52:41 -06:00
|
|
|
op^.token.class := realConstant;
|
2021-03-07 00:48:51 -06:00
|
|
|
op^.token.kind := ekind;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
|
|
|
end; {if}
|
|
|
|
1:
|
|
|
|
end;
|
|
|
|
|
|
|
|
plusplusop, {prefix ++}
|
|
|
|
minusminusop, {prefix --}
|
|
|
|
opplusplus, {postfix ++}
|
|
|
|
opminusminus, {postfix --}
|
|
|
|
sizeofsy, {sizeof}
|
2020-01-05 18:09:26 -06:00
|
|
|
_Alignofsy, {_Alignof (erroneous uses)}
|
2024-08-29 13:16:01 -05:00
|
|
|
alignofsy,
|
2017-10-21 18:40:19 -05:00
|
|
|
castoper, {(type)}
|
|
|
|
typedef, {(type-name)}
|
|
|
|
tildech, {~}
|
|
|
|
excch, {!}
|
|
|
|
uminus, {unary -}
|
2022-12-09 19:03:38 -06:00
|
|
|
uplus, {unary +}
|
2017-10-21 18:40:19 -05:00
|
|
|
uand, {unary &}
|
|
|
|
uasterisk: begin {unary *}
|
|
|
|
op^.left := Pop;
|
|
|
|
|
|
|
|
if op^.token.kind = sizeofsy then begin
|
2016-11-05 00:24:22 -05:00
|
|
|
op^.token.kind := ulongConst;
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.class := longConstant;
|
2023-05-07 18:28:31 -05:00
|
|
|
kindLeft := op^.left^.token.kind;
|
|
|
|
if kindLeft = stringConst then
|
2021-10-11 20:54:37 -05:00
|
|
|
op^.token.lval := op^.left^.token.sval^.length
|
2017-10-21 18:40:19 -05:00
|
|
|
else begin
|
|
|
|
lCodeGeneration := codeGeneration;
|
|
|
|
codeGeneration := false;
|
|
|
|
GenerateCode(op^.left);
|
2023-05-07 18:28:31 -05:00
|
|
|
if kindLeft = dotch then
|
|
|
|
if isBitfield then
|
|
|
|
Error(49);
|
2017-10-21 18:40:19 -05:00
|
|
|
codeGeneration := lCodeGeneration and (numErrors = 0);
|
|
|
|
op^.token.lval := expressionType^.size;
|
|
|
|
with expressionType^ do
|
2016-11-05 00:47:35 -05:00
|
|
|
if (size = 0) or ((kind = arrayType) and (elements = 0)) then
|
|
|
|
Error(49);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
|
|
|
op^.left := nil;
|
|
|
|
end {if sizeofsy}
|
2020-01-05 18:09:26 -06:00
|
|
|
|
2024-08-29 13:16:01 -05:00
|
|
|
else if op^.token.kind in [_Alignofsy,alignofsy] then begin
|
2020-01-05 18:09:26 -06:00
|
|
|
{error case: operand of _Alignof is not a parenthesized type-name}
|
|
|
|
Error(36);
|
|
|
|
op^.token.kind := ulongConst;
|
|
|
|
op^.token.class := longConstant;
|
|
|
|
op^.token.lval := 1;
|
|
|
|
dispose(op^.left);
|
|
|
|
end {else if _Alignofsy}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
else if op^.token.kind = castoper then begin
|
|
|
|
class := op^.left^.token.class;
|
2021-02-19 21:57:31 -06:00
|
|
|
if class in [intConstant,longConstant,longlongconstant,
|
2021-03-04 23:52:41 -06:00
|
|
|
realConstant] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
tp := op^.castType;
|
|
|
|
while tp^.kind = definedType do
|
|
|
|
tp := tp^.dType;
|
|
|
|
if tp^.kind = scalarType then begin
|
|
|
|
baseType := tp^.baseType;
|
2021-03-06 00:57:13 -06:00
|
|
|
if fenvAccess then
|
|
|
|
if kind in [normalExpression, autoInitializerExpression] then
|
|
|
|
if (baseType in [cgReal,cgDouble,cgComp,cgExtended])
|
|
|
|
or (class = realConstant) then
|
|
|
|
goto 3;
|
2021-02-19 21:57:31 -06:00
|
|
|
if (baseType < cgString) or (baseType in [cgQuad,cgUQuad])
|
|
|
|
then begin
|
2021-03-04 23:52:41 -06:00
|
|
|
if class = realConstant then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
rop1 := RealVal(op^.left^.token);
|
2021-02-19 21:57:31 -06:00
|
|
|
if baseType = cgUQuad then
|
|
|
|
CnvXULL(llop1, rop1)
|
|
|
|
else
|
|
|
|
CnvXLL(llop1, rop1);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
2021-02-19 21:57:31 -06:00
|
|
|
else begin {handle integer constants}
|
|
|
|
GetLongLongVal(llop1, op^.left^.token);
|
|
|
|
if op^.left^.token.kind = ulonglongconst then
|
2021-03-04 23:52:41 -06:00
|
|
|
rop1 := CnvULLX(llop1)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
2021-03-04 23:52:41 -06:00
|
|
|
rop1 := CnvLLX(llop1);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else if}
|
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
|
|
|
if baseType in [cgByte,cgWord] then begin
|
2021-03-07 13:38:21 -06:00
|
|
|
if baseType = cgByte then
|
|
|
|
op^.token.kind := scharConst
|
|
|
|
else
|
|
|
|
op^.token.kind := intConst;
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.class := intConstant;
|
2021-01-25 21:22:58 -06:00
|
|
|
if tp^.cType = ctBool then
|
|
|
|
op^.token.ival := ord(rop1 <> 0.0)
|
|
|
|
else
|
2021-02-19 21:57:31 -06:00
|
|
|
op^.token.ival := long(llop1.lo).lsw;
|
2017-10-21 18:40:19 -05:00
|
|
|
if baseType = cgByte then
|
|
|
|
with op^.token do begin
|
|
|
|
ival := ival & $00FF;
|
|
|
|
if (ival & $0080) <> 0 then
|
|
|
|
ival := ival | $FF00;
|
|
|
|
end; {with}
|
|
|
|
end {if}
|
2018-09-05 12:49:02 -05:00
|
|
|
else if baseType = cgUWord then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.kind := uintConst;
|
|
|
|
op^.token.class := intConstant;
|
2021-02-19 21:57:31 -06:00
|
|
|
op^.token.ival := long(llop1.lo).lsw;
|
2018-09-05 12:49:02 -05:00
|
|
|
end {else if}
|
|
|
|
else if baseType = cgUByte then begin
|
2021-03-07 13:38:21 -06:00
|
|
|
if tp^.cType = ctUChar then
|
|
|
|
op^.token.kind := ucharConst
|
|
|
|
else
|
|
|
|
op^.token.kind := charConst;
|
2018-09-05 12:49:02 -05:00
|
|
|
op^.token.class := intConstant;
|
2021-02-19 21:57:31 -06:00
|
|
|
op^.token.ival := long(llop1.lo).lsw;
|
2018-09-05 12:49:02 -05:00
|
|
|
op^.token.ival := op^.token.ival & $00FF;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
|
|
|
else if baseType = cgLong then begin
|
|
|
|
op^.token.kind := longConst;
|
|
|
|
op^.token.class := longConstant;
|
2021-02-19 21:57:31 -06:00
|
|
|
op^.token.lval := llop1.lo;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
|
|
|
else if baseType = cgULong then begin
|
|
|
|
op^.token.kind := ulongConst;
|
|
|
|
op^.token.class := longConstant;
|
2021-02-19 21:57:31 -06:00
|
|
|
op^.token.lval := llop1.lo;
|
|
|
|
end {else if}
|
|
|
|
else if baseType = cgQuad then begin
|
|
|
|
op^.token.kind := longlongConst;
|
|
|
|
op^.token.class := longlongConstant;
|
|
|
|
op^.token.qval := llop1;
|
|
|
|
end {else if}
|
|
|
|
else if baseType = cgUQuad then begin
|
|
|
|
op^.token.kind := ulonglongConst;
|
|
|
|
op^.token.class := longlongConstant;
|
|
|
|
op^.token.qval := llop1;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
|
|
|
else begin
|
2021-03-07 00:48:51 -06:00
|
|
|
case baseType of
|
|
|
|
cgReal: op^.token.kind := floatConst;
|
|
|
|
cgDouble: op^.token.kind := doubleConst;
|
|
|
|
cgExtended: op^.token.kind := extendedConst;
|
|
|
|
cgComp: op^.token.kind := compConst;
|
|
|
|
end; {case}
|
2021-03-04 23:52:41 -06:00
|
|
|
op^.token.class := realConstant;
|
2021-03-06 22:28:39 -06:00
|
|
|
LimitPrecision(rop1, baseType);
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.rval := rop1;
|
|
|
|
end; {else if}
|
|
|
|
end; {if}
|
2021-02-19 21:57:31 -06:00
|
|
|
3: end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {if}
|
|
|
|
end {else if castoper}
|
|
|
|
|
|
|
|
else if not (op^.token.kind in
|
|
|
|
[typedef,plusplusop,minusminusop,opplusplus,opminusminus,uand]) then
|
|
|
|
begin
|
2021-02-16 23:45:59 -06:00
|
|
|
if (kind <> preprocessorExpression) and (op^.left^.token.kind
|
2021-03-07 13:38:21 -06:00
|
|
|
in [intconst,uintconst,longconst,ulongconst,charconst,scharconst,
|
2021-10-11 20:54:37 -05:00
|
|
|
ucharconst,ushortconst]) then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{evaluate a constant operation}
|
|
|
|
ekind := op^.left^.token.kind;
|
2021-03-07 13:38:21 -06:00
|
|
|
if ekind in [charconst,scharconst,ucharconst] then
|
|
|
|
ekind := intconst;
|
2017-10-21 18:40:19 -05:00
|
|
|
op1 := IntVal(op^.left^.token);
|
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
|
|
|
case op^.token.kind of
|
|
|
|
tildech : op1 := ~op1; {~}
|
|
|
|
excch : begin {!}
|
|
|
|
op1 := ord(op1 = 0);
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
uminus : op1 := -op1; {unary -}
|
2022-12-09 19:03:38 -06:00
|
|
|
uplus : ; {unary +}
|
2016-12-19 22:19:36 -06:00
|
|
|
uasterisk : Error(79); {unary *}
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
|
|
|
op^.token.kind := ekind;
|
|
|
|
if ekind in [longconst,ulongconst] then begin
|
|
|
|
op^.token.class := longConstant;
|
|
|
|
op^.token.lval := op1;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
op^.token.class := intConstant;
|
|
|
|
op^.token.ival := long(op1).lsw;
|
|
|
|
end; {else}
|
|
|
|
end {if}
|
2021-02-16 23:45:59 -06:00
|
|
|
else if op^.left^.token.kind in [longlongconst,ulonglongconst,
|
2021-03-07 13:38:21 -06:00
|
|
|
intconst,uintconst,longconst,ulongconst,charconst,scharconst,
|
2021-10-11 20:54:37 -05:00
|
|
|
ucharconst,ushortconst] then begin
|
2021-02-13 15:07:16 -06:00
|
|
|
|
2021-02-16 23:45:59 -06:00
|
|
|
{evaluate a constant operation with long long operand}
|
2021-02-13 15:07:16 -06:00
|
|
|
ekind := op^.left^.token.kind;
|
2021-03-07 13:38:21 -06:00
|
|
|
if ekind in [charconst,scharconst,ucharconst] then
|
|
|
|
ekind := intconst;
|
2021-02-16 23:45:59 -06:00
|
|
|
if kind = preprocessorExpression then
|
|
|
|
ekind := PPKind(op^.left^.token);
|
|
|
|
GetLongLongVal(llop1, op^.left^.token);
|
2021-02-13 15:07:16 -06:00
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
|
|
|
case op^.token.kind of
|
|
|
|
tildech : begin {~}
|
|
|
|
llop1.lo := ~llop1.lo;
|
|
|
|
llop1.hi := ~llop1.hi;
|
|
|
|
end;
|
|
|
|
excch : begin {!}
|
|
|
|
op1 := ord((llop1.hi = 0) and (llop1.lo = 0));
|
|
|
|
ekind := intconst;
|
|
|
|
end;
|
|
|
|
uminus : begin {unary -}
|
|
|
|
llop1.lo := ~llop1.lo;
|
|
|
|
llop1.hi := ~llop1.hi;
|
|
|
|
llop1.lo := llop1.lo + 1;
|
|
|
|
if llop1.lo = 0 then
|
|
|
|
llop1.hi := llop1.hi + 1;
|
|
|
|
end;
|
2022-12-09 19:03:38 -06:00
|
|
|
uplus : ; {unary +}
|
2021-02-13 15:07:16 -06:00
|
|
|
uasterisk : Error(79); {unary *}
|
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
|
|
|
op^.token.kind := ekind;
|
|
|
|
if ekind in [longlongconst,ulonglongconst] then begin
|
|
|
|
op^.token.class := longlongConstant;
|
|
|
|
op^.token.qval := llop1;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
op^.token.class := intConstant;
|
|
|
|
op^.token.ival := long(op1).lsw;
|
|
|
|
end; {else}
|
|
|
|
end {else if}
|
2021-03-07 00:48:51 -06:00
|
|
|
else if op^.left^.token.kind in
|
|
|
|
[floatconst,doubleconst,extendedconst,compconst] then begin
|
2021-03-06 00:57:13 -06:00
|
|
|
if fenvAccess then
|
|
|
|
if kind in [normalExpression, autoInitializerExpression] then
|
|
|
|
goto 4;
|
2021-03-07 00:48:51 -06:00
|
|
|
ekind := op^.left^.token.kind;
|
2017-10-21 18:40:19 -05:00
|
|
|
rop1 := RealVal(op^.left^.token);
|
|
|
|
dispose(op^.left);
|
|
|
|
op^.left := nil;
|
|
|
|
case op^.token.kind of
|
|
|
|
uminus : begin {unary -}
|
2021-03-04 23:52:41 -06:00
|
|
|
op^.token.class := realConstant;
|
2021-03-07 00:48:51 -06:00
|
|
|
op^.token.kind := ekind;
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.rval := -rop1;
|
|
|
|
end;
|
2022-12-09 19:03:38 -06:00
|
|
|
uplus : begin {unary +}
|
|
|
|
op^.token.class := realConstant;
|
|
|
|
op^.token.kind := ekind;
|
|
|
|
op^.token.rval := rop1;
|
|
|
|
end;
|
2016-12-20 21:41:48 -06:00
|
|
|
excch : begin {!}
|
|
|
|
op^.token.class := intConstant;
|
|
|
|
op^.token.kind := intconst;
|
|
|
|
op^.token.ival := ord(rop1 = 0.0);
|
|
|
|
end;
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise : begin {illegal operation}
|
|
|
|
Error(66);
|
2021-03-04 23:52:41 -06:00
|
|
|
op^.token.class := realConstant;
|
2021-03-07 00:48:51 -06:00
|
|
|
op^.token.kind := ekind;
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.rval := rop1;
|
|
|
|
end;
|
|
|
|
end; {case}
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
2021-03-06 00:57:13 -06:00
|
|
|
4:
|
2017-10-21 18:40:19 -05:00
|
|
|
end;
|
|
|
|
|
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
|
|
|
op^.next := stack; {place the operation on the operand stack}
|
|
|
|
stack := op;
|
|
|
|
end; {Operation}
|
|
|
|
|
|
|
|
|
|
|
|
procedure Skip;
|
|
|
|
|
2020-01-29 17:09:52 -06:00
|
|
|
{ skip all tokens in the remainder of the expression }
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
begin {Skip}
|
|
|
|
while not (token.kind in stopSym+[eofsy]) do
|
|
|
|
NextToken;
|
|
|
|
errorFound := true;
|
|
|
|
end; {Skip}
|
2021-03-07 21:59:23 -06:00
|
|
|
|
|
|
|
|
|
|
|
procedure DoGeneric;
|
|
|
|
|
|
|
|
{ process a generic selection expression }
|
|
|
|
|
|
|
|
label 10;
|
|
|
|
|
|
|
|
type
|
|
|
|
typeListPtr = ^typeList;
|
|
|
|
typeList = record
|
|
|
|
next: typeListPtr;
|
|
|
|
theType: typePtr;
|
|
|
|
end;
|
|
|
|
|
|
|
|
var
|
|
|
|
lCodeGeneration: boolean; {local copy of codeGeneration}
|
|
|
|
tempExpr: tokenPtr; {temporary to hold expression trees}
|
|
|
|
controllingType: typeRecord; {type of controlling expression}
|
|
|
|
typesSeen: typeListPtr; {types that already have associations}
|
|
|
|
tl: typeListPtr; {temporary type list pointer}
|
|
|
|
resultExpr: tokenPtr; {the result expression}
|
|
|
|
defaultExpr: tokenPtr; {the default expression}
|
|
|
|
currentType: typePtr; {the type for the current association}
|
|
|
|
typesMatch: boolean; {does the current type match}
|
|
|
|
foundMatch: boolean; {have we found a matching type?}
|
|
|
|
foundDefault: boolean; {have we found the default case?}
|
|
|
|
|
|
|
|
begin {DoGeneric}
|
|
|
|
if not expectingTerm then begin
|
|
|
|
Error(36);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
NextToken;
|
|
|
|
if token.kind <> lparench then begin
|
|
|
|
Error(36);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
new(op); {record it like a parenthesized expr}
|
|
|
|
op^.next := opStack;
|
|
|
|
op^.left := nil;
|
|
|
|
op^.middle := nil;
|
|
|
|
op^.right := nil;
|
|
|
|
opStack := op;
|
|
|
|
op^.token.kind := lparench;
|
|
|
|
op^.token.class := reservedSymbol;
|
|
|
|
parenCount := parenCount+1;
|
|
|
|
NextToken; {process the controlling expression}
|
|
|
|
tempExpr := ExpressionTree(normalExpression, [commach]);
|
|
|
|
lCodeGeneration := codeGeneration;
|
|
|
|
codeGeneration := false;
|
|
|
|
GenerateCode(tempExpr);
|
|
|
|
codeGeneration := lCodeGeneration and (numErrors = 0);
|
|
|
|
{get controlling type after conversions}
|
|
|
|
if expressionType^.kind = functionType then begin
|
2022-06-18 19:30:20 -05:00
|
|
|
controllingType.size := cgPointerSize;
|
2022-01-22 18:21:11 -06:00
|
|
|
controllingType.kind := pointerType;
|
|
|
|
controllingType.pType := expressionType;
|
2021-03-07 21:59:23 -06:00
|
|
|
end {if}
|
2021-03-07 23:35:12 -06:00
|
|
|
else if expressionType^.kind in [structType,unionType] then begin
|
|
|
|
controllingType.size := expressionType^.size;
|
|
|
|
controllingType.kind := definedType;
|
|
|
|
controllingType.dType := expressionType;
|
|
|
|
end {else if}
|
2021-03-07 21:59:23 -06:00
|
|
|
else
|
|
|
|
controllingType := expressionType^;
|
2022-06-18 19:30:20 -05:00
|
|
|
if controllingType.kind = arrayType then begin
|
2021-03-07 21:59:23 -06:00
|
|
|
controllingType.kind := pointerType;
|
2022-06-18 19:30:20 -05:00
|
|
|
controllingType.size := cgPointerSize;
|
|
|
|
end; {if}
|
Record volatile and restrict qualifiers in types.
These are needed to correctly distinguish pointer types in _Generic. They should also be used for type compatibility checks in other contexts, but currently are not.
This also fixes a couple small problems related to type qualifiers:
*restrict was not allowed to appear after * in type-names
*volatile status was not properly recorded in sym files
Here is an example of using _Generic to distinguish pointer types based on the qualifiers of the pointed-to type:
#include <stdio.h>
#define f(e) _Generic((e),\
int * restrict *: 1,\
int * volatile const *: 2,\
int **: 3,\
default: 0)
#define g(e) _Generic((e),\
int *: 1,\
const int *: 2,\
volatile int *: 3,\
default: 0)
int main(void) {
int * restrict * p1;
int * volatile const * p2;
int * const * p3;
// should print "1 2 0 1"
printf("%i %i %i %i\n", f(p1), f(p2), f(p3), f((int * restrict *)0));
int *q1;
const int *q2;
volatile int *q3;
const volatile int *q4;
// should print "1 2 3 0"
printf("%i %i %i %i\n", g(q1), g(q2), g(q3), g(q4));
}
Here is an example of a problem resulting from volatile not being recorded in sym files (if a sym file was present, the read of x was lifted out of the loop):
#pragma optimize -1
static volatile int x;
#include <stdio.h>
int main(void) {
int y;
for (unsigned i = 0; i < 100; i++) {
y = x*2 + 7;
}
}
2021-08-29 21:10:20 -05:00
|
|
|
controllingType.qualifiers := [];
|
2022-07-18 21:19:44 -05:00
|
|
|
controllingType.saveDisp := 0;
|
2021-03-07 21:59:23 -06:00
|
|
|
|
|
|
|
typesSeen := nil;
|
|
|
|
resultExpr := nil;
|
|
|
|
defaultExpr := nil;
|
|
|
|
foundMatch := false;
|
|
|
|
foundDefault := false;
|
|
|
|
while token.kind = commach do begin {process the generic associations}
|
|
|
|
NextToken;
|
|
|
|
typesMatch := false;
|
|
|
|
if token.kind <> defaultsy then begin
|
2021-03-09 17:45:49 -06:00
|
|
|
if not (token.kind in specifierQualifierListElement) then begin
|
|
|
|
Error(26);
|
|
|
|
while not (token.kind in [colonch,commach,rparench,eofsy]) do
|
|
|
|
NextToken;
|
|
|
|
end; {if}
|
2022-10-01 21:28:16 -05:00
|
|
|
currentType := TypeName; {get the type name}
|
2021-03-07 21:59:23 -06:00
|
|
|
if (currentType^.size = 0) or (currentType^.kind = functionType) then
|
2021-03-07 23:35:12 -06:00
|
|
|
Error(133);
|
2021-03-07 21:59:23 -06:00
|
|
|
tl := typesSeen; {check if it is a duplicate}
|
|
|
|
while tl <> nil do begin
|
2021-03-07 23:39:30 -06:00
|
|
|
if StrictCompTypes(currentType, tl^.theType) then begin
|
2021-03-07 21:59:23 -06:00
|
|
|
Error(158);
|
|
|
|
goto 10;
|
|
|
|
end; {if}
|
|
|
|
tl := tl^.next;
|
|
|
|
end; {while}
|
|
|
|
new(tl); {record it as seen}
|
|
|
|
tl^.next := typesSeen;
|
|
|
|
tl^.theType := currentType;
|
|
|
|
typesSeen := tl;
|
|
|
|
{see if the types match}
|
2021-03-07 23:39:30 -06:00
|
|
|
typesMatch := StrictCompTypes(currentType, controllingType);
|
2021-03-07 21:59:23 -06:00
|
|
|
if typesMatch then begin
|
|
|
|
if foundMatch then begin {sanity check - should never happen}
|
|
|
|
typesMatch := false;
|
|
|
|
Error(158);
|
|
|
|
end; {if}
|
|
|
|
foundMatch := true;
|
|
|
|
end; {if}
|
|
|
|
end {if}
|
|
|
|
else begin {handle default association}
|
|
|
|
NextToken;
|
|
|
|
currentType := nil;
|
|
|
|
if foundDefault then
|
|
|
|
Error(159);
|
|
|
|
foundDefault := true;
|
|
|
|
end; {else}
|
|
|
|
10:
|
|
|
|
if token.kind = colonch then {skip the colon}
|
|
|
|
NextToken
|
|
|
|
else
|
|
|
|
Error(29);
|
|
|
|
{get the expression in this association}
|
|
|
|
if (currentType = nil) and (defaultExpr = nil) and not foundMatch then
|
|
|
|
defaultExpr := ExpressionTree(kind, [commach,rparench])
|
|
|
|
else if typesMatch then
|
|
|
|
resultExpr := ExpressionTree(kind, [commach,rparench])
|
|
|
|
else
|
|
|
|
tempExpr := ExpressionTree(normalExpression, [commach,rparench]);
|
|
|
|
end; {while}
|
|
|
|
if token.kind <> rparench then
|
|
|
|
Error(12);
|
2021-09-07 19:30:57 -05:00
|
|
|
while typesSeen <> nil do begin {dispose of the list of types seen}
|
|
|
|
tl := typesSeen^.next;
|
|
|
|
dispose(typesSeen);
|
|
|
|
typesSeen := tl;
|
|
|
|
end; {while}
|
2021-03-07 21:59:23 -06:00
|
|
|
|
|
|
|
if not foundMatch then {use default if no match found}
|
|
|
|
if foundDefault then
|
|
|
|
resultExpr := defaultExpr;
|
|
|
|
if not (foundMatch or foundDefault) then begin
|
|
|
|
Error(160); {report error & synthesize a token}
|
|
|
|
resultExpr := pointer(Calloc(sizeof(tokenRecord)));
|
|
|
|
resultExpr^.token.kind := intconst;
|
|
|
|
resultExpr^.token.class := intConstant;
|
2022-07-18 21:19:44 -05:00
|
|
|
{resultExpr^.token.ival := 0;}
|
2021-03-07 21:59:23 -06:00
|
|
|
end; {if}
|
|
|
|
if resultExpr <> nil then begin
|
|
|
|
resultExpr^.next := stack; {stack the resulting expression}
|
|
|
|
stack := resultExpr;
|
|
|
|
end; {if}
|
|
|
|
expectingTerm := false;
|
|
|
|
end; {DoGeneric}
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2021-09-15 22:22:16 -05:00
|
|
|
procedure DoCompoundLiteral;
|
|
|
|
|
|
|
|
{ process a compound literal expression }
|
|
|
|
|
|
|
|
label 1;
|
|
|
|
|
|
|
|
var
|
|
|
|
id: identPtr;
|
|
|
|
sp: tokenPtr;
|
|
|
|
|
|
|
|
begin {DoCompoundLiteral}
|
|
|
|
if kind in [preprocessorExpression,arrayExpression] then begin
|
|
|
|
op := opStack;
|
|
|
|
while op <> nil do begin
|
|
|
|
if op^.token.kind = sizeofsy then
|
|
|
|
goto 1;
|
|
|
|
op := op^.next;
|
|
|
|
end; {while}
|
|
|
|
Error(41);
|
|
|
|
errorFound := true;
|
|
|
|
end; {if}
|
|
|
|
1:
|
|
|
|
id := MakeCompoundLiteral(opStack^.castType);
|
|
|
|
opStack := opStack^.next;
|
|
|
|
|
|
|
|
{create an operand on the stack}
|
|
|
|
new(sp);
|
2022-06-08 20:58:52 -05:00
|
|
|
if id^.class = staticsy then
|
|
|
|
sp^.token.kind := ident
|
|
|
|
else
|
|
|
|
sp^.token.kind := compoundliteral;
|
2021-09-15 22:22:16 -05:00
|
|
|
sp^.token.class := identifier;
|
|
|
|
sp^.token.symbolPtr := id;
|
|
|
|
sp^.token.name := id^.name;
|
|
|
|
sp^.id := id;
|
|
|
|
sp^.next := stack;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
stack := sp;
|
|
|
|
|
|
|
|
ComplexTerm;
|
|
|
|
expectingTerm := false;
|
|
|
|
end; {DoCompoundLiteral}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
begin {ExpressionTree}
|
|
|
|
opStack := nil;
|
|
|
|
stack := nil;
|
|
|
|
if token.kind = typedef then {handle typedefs that are hidden}
|
2023-03-06 21:38:05 -06:00
|
|
|
if FindSymbol(token,variableSpace,false,true) <> nil then
|
2017-10-21 18:40:19 -05:00
|
|
|
if token.symbolPtr^.class <> typedefsy then
|
|
|
|
token.kind := ident;
|
|
|
|
if token.kind in startExpression then begin
|
|
|
|
expressionValue := 0; {initialize the expression value}
|
|
|
|
expectingTerm := true; {the first item should be a term}
|
|
|
|
done := false; {convert the expression to postfix form}
|
|
|
|
parenCount := 0;
|
|
|
|
repeat {scan the token list...}
|
|
|
|
if token.kind in startTerm then begin
|
|
|
|
|
|
|
|
{we must expect a term or unary operand}
|
|
|
|
if not expectingTerm then begin
|
|
|
|
Error(36);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
if token.kind = ident then
|
|
|
|
|
|
|
|
{handle a complex operand}
|
|
|
|
DoOperand
|
|
|
|
else begin
|
|
|
|
{handle a constant operand}
|
|
|
|
new(sp);
|
|
|
|
sp^.token := token;
|
|
|
|
sp^.next := stack;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
|
|
|
stack := sp;
|
|
|
|
if kind in [preprocessorExpression,arrayExpression] then
|
2021-03-07 00:48:51 -06:00
|
|
|
if token.kind in [stringconst,floatconst,doubleconst,
|
|
|
|
extendedconst,compconst] then begin
|
2016-01-01 16:23:30 -06:00
|
|
|
if kind = arrayExpression then begin
|
|
|
|
op := opStack;
|
2021-03-07 00:48:51 -06:00
|
|
|
if token.kind <> stringconst then
|
2016-12-19 14:36:48 -06:00
|
|
|
if op <> nil then
|
|
|
|
if op^.token.kind = castoper then
|
|
|
|
if op^.casttype^.kind = scalarType then
|
|
|
|
if op^.casttype^.baseType in [cgByte,cgUByte,
|
2021-02-19 21:57:31 -06:00
|
|
|
cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad]
|
|
|
|
then goto 3;
|
2016-01-01 16:23:30 -06:00
|
|
|
while op <> nil do begin
|
|
|
|
if op^.token.kind = sizeofsy then
|
|
|
|
goto 3;
|
|
|
|
op := op^.next;
|
|
|
|
end; {while}
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(41);
|
|
|
|
errorFound := true;
|
|
|
|
end; {if}
|
2016-01-01 16:23:30 -06:00
|
|
|
3:
|
2017-10-21 18:40:19 -05:00
|
|
|
NextToken;
|
|
|
|
ComplexTerm;
|
|
|
|
end; {else}
|
|
|
|
expectingTerm := false; {the next thing should be an operation}
|
|
|
|
end {else}
|
|
|
|
{handle a closing parenthesis}
|
|
|
|
else if (token.kind = rparench) and (parenCount > 0) then begin
|
|
|
|
if expectingTerm then begin {make sure it is in a legal spot}
|
|
|
|
Error(37);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
while opStack^.token.kind <> lparench do
|
|
|
|
Operation; {do pending operations}
|
|
|
|
op := opStack;
|
|
|
|
opStack := op^.next;
|
|
|
|
dispose(op);
|
|
|
|
parenCount := parenCount-1;
|
|
|
|
NextToken; {skip the ')'}
|
|
|
|
ComplexTerm; {handle subscripts, selection, etc.}
|
|
|
|
end {else}
|
|
|
|
else if token.kind = lparench then begin
|
|
|
|
|
|
|
|
{handle open paren and type casts}
|
|
|
|
if not expectingTerm then begin
|
|
|
|
Error(38);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
NextToken;
|
2020-01-12 15:42:56 -06:00
|
|
|
if token.kind in specifierQualifierListElement then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
doingSizeof := false;
|
2020-01-05 18:09:26 -06:00
|
|
|
doingAlignof := false;
|
2017-10-21 18:40:19 -05:00
|
|
|
if opStack <> nil then
|
|
|
|
if opStack^.token.kind = sizeofsy then
|
2020-01-05 18:09:26 -06:00
|
|
|
doingSizeof := true
|
2024-08-29 13:16:01 -05:00
|
|
|
else if opStack^.token.kind in [_Alignofsy,alignofsy] then
|
2020-01-05 18:09:26 -06:00
|
|
|
doingAlignof := true;
|
2022-10-01 21:28:16 -05:00
|
|
|
tType := TypeName;
|
2024-08-11 20:33:36 -05:00
|
|
|
Match(rparench,12);
|
|
|
|
if (doingSizeof and (token.kind <> lbracech)) or doingAlignof then
|
|
|
|
begin
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{handle a sizeof operator}
|
|
|
|
op := opStack;
|
|
|
|
opStack := op^.next;
|
|
|
|
dispose(op);
|
|
|
|
new(sp);
|
|
|
|
sp^.next := stack;
|
|
|
|
sp^.left := nil;
|
|
|
|
sp^.middle := nil;
|
|
|
|
sp^.right := nil;
|
2016-11-05 00:24:22 -05:00
|
|
|
sp^.token.kind := ulongconst;
|
2017-10-21 18:40:19 -05:00
|
|
|
sp^.token.class := longConstant;
|
2020-01-05 18:09:26 -06:00
|
|
|
if doingSizeof then
|
2022-10-01 21:28:16 -05:00
|
|
|
sp^.token.lval := tType^.size
|
2020-01-05 18:09:26 -06:00
|
|
|
else {if doingAlignof then}
|
|
|
|
sp^.token.lval := 1;
|
2022-10-01 21:28:16 -05:00
|
|
|
with tType^ do
|
2016-11-05 00:47:35 -05:00
|
|
|
if (size = 0) or ((kind = arrayType) and (elements = 0)) then
|
2020-01-05 18:09:26 -06:00
|
|
|
Error(133);
|
2017-10-21 18:40:19 -05:00
|
|
|
sp^.next := stack;
|
|
|
|
stack := sp;
|
|
|
|
expectingTerm := false;
|
|
|
|
end {if}
|
|
|
|
else {doing a cast} begin
|
|
|
|
|
|
|
|
{handle a type cast}
|
|
|
|
new(op); {stack the cast operator}
|
|
|
|
op^.left := nil;
|
|
|
|
op^.middle := nil;
|
|
|
|
op^.right := nil;
|
2022-10-01 21:28:16 -05:00
|
|
|
op^.castType := tType;
|
2017-10-21 18:40:19 -05:00
|
|
|
op^.token.kind := castoper;
|
|
|
|
op^.token.class := reservedWord;
|
|
|
|
op^.next := opStack;
|
|
|
|
opStack := op;
|
|
|
|
end; {else}
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
new(op); {record the '('}
|
|
|
|
op^.next := opStack;
|
|
|
|
op^.left := nil;
|
|
|
|
op^.middle := nil;
|
|
|
|
op^.right := nil;
|
|
|
|
opStack := op;
|
|
|
|
op^.token.kind := lparench;
|
|
|
|
op^.token.class := reservedSymbol;
|
|
|
|
parenCount := parenCount+1;
|
|
|
|
end;
|
|
|
|
end {else if}
|
2021-09-15 22:22:16 -05:00
|
|
|
else if (token.kind = lbracech) {handle a compound literal}
|
|
|
|
and (opstack <> nil) and (opStack^.token.kind = castoper) then begin
|
|
|
|
DoCompoundLiteral
|
|
|
|
end {else if}
|
2021-03-07 21:59:23 -06:00
|
|
|
else if token.kind = _Genericsy then {handle _Generic}
|
|
|
|
DoGeneric
|
2017-10-21 18:40:19 -05:00
|
|
|
else begin {handle an operation...}
|
|
|
|
if expectingTerm then {convert unary operators to separate tokens}
|
|
|
|
if token.kind in [asteriskch,minusch,plusch,andch] then
|
|
|
|
case token.kind of
|
|
|
|
asteriskch: token.kind := uasterisk;
|
|
|
|
minusch : token.kind := uminus;
|
|
|
|
andch : token.kind := uand;
|
2022-12-09 19:03:38 -06:00
|
|
|
plusch : token.kind := uplus;
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise : Error(57);
|
|
|
|
end; {case}
|
|
|
|
if icp[token.kind] = notAnOperation then
|
|
|
|
done := true {end of expression found...}
|
2021-03-16 18:20:22 -05:00
|
|
|
else if (token.kind in stopSym) and (parenCount = 0)
|
|
|
|
and ((opStack = nil) or (opStack^.token.kind <> questionch)) then
|
2017-10-21 18:40:19 -05:00
|
|
|
done := true
|
|
|
|
else begin
|
|
|
|
if not (kind in [normalExpression, autoInitializerExpression]) then
|
|
|
|
if (token.kind in
|
|
|
|
[plusplusop,minusminusop,eqch,pluseqop,minuseqop,
|
|
|
|
opplusplus,opminusminus,
|
|
|
|
asteriskeqop,slasheqop,percenteqop,ltlteqop,
|
|
|
|
gtgteqop,caroteqop,bareqop,commach])
|
|
|
|
or ((kind = preprocessorExpression)
|
|
|
|
and (token.kind = sizeofsy))
|
|
|
|
or ((kind <> initializerExpression)
|
|
|
|
and (token.kind = uand)) then begin
|
2021-08-22 20:33:34 -05:00
|
|
|
Error(161);
|
2017-10-21 18:40:19 -05:00
|
|
|
errorFound := true;
|
|
|
|
end; {if}
|
|
|
|
if token.kind in {make sure we get what we want}
|
2024-08-29 13:16:01 -05:00
|
|
|
[plusplusop,minusminusop,sizeofsy,_Alignofsy,alignofsy,tildech,
|
|
|
|
excch,uasterisk,uminus,uplus,uand] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
if not expectingTerm then begin
|
|
|
|
Error(38);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
if expectingTerm then begin
|
|
|
|
Error(37);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
expectingTerm := true;
|
|
|
|
{handle 2nd half of ternary operator}
|
|
|
|
if token.kind = colonch then begin
|
|
|
|
done2 := false; {do pending operations}
|
|
|
|
repeat
|
|
|
|
if opStack = nil then
|
|
|
|
done2 := true
|
|
|
|
else if opStack^.token.kind <> questionch then
|
|
|
|
Operation
|
|
|
|
else
|
|
|
|
done2 := true;
|
|
|
|
until done2;
|
|
|
|
if (opStack = nil) or
|
|
|
|
(opStack^.token.kind <> questionch) then begin
|
|
|
|
Error(39);
|
|
|
|
Skip;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
op := opStack;
|
|
|
|
opStack := op^.next;
|
|
|
|
dispose(op);
|
|
|
|
end {if}
|
|
|
|
else begin
|
2020-01-29 17:09:52 -06:00
|
|
|
done2 := false; {do operations with less precedence}
|
2017-10-21 18:40:19 -05:00
|
|
|
repeat
|
|
|
|
if opStack = nil then
|
|
|
|
done2 := true
|
|
|
|
else if isp[opStack^.token.kind] >= icp[token.kind] then
|
|
|
|
Operation
|
|
|
|
else
|
|
|
|
done2 := true;
|
|
|
|
until done2;
|
|
|
|
end; {else}
|
|
|
|
end; {else}
|
|
|
|
new(op); {record the operation}
|
|
|
|
op^.next := opStack;
|
|
|
|
op^.left := nil;
|
|
|
|
op^.middle := nil;
|
|
|
|
op^.right := nil;
|
|
|
|
opStack := op;
|
|
|
|
op^.token := token;
|
|
|
|
NextToken;
|
|
|
|
end; {else}
|
|
|
|
end; {else}
|
|
|
|
2:
|
|
|
|
until done;
|
|
|
|
if parenCount > 0 then begin
|
|
|
|
Error(12);
|
|
|
|
errorFound := true;
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
while opStack <> nil do {do pending operations}
|
|
|
|
Operation;
|
|
|
|
{there should be exactly one operand left}
|
|
|
|
if (stack = nil) or (stack^.next <> nil) then begin
|
|
|
|
Error(36);
|
|
|
|
errorFound := true;
|
|
|
|
end; {if}
|
|
|
|
end; {else}
|
|
|
|
end {if}
|
|
|
|
else begin {the start of an expression was not found}
|
|
|
|
Error(35);
|
|
|
|
if not (token.kind in stopSym) then
|
|
|
|
NextToken;
|
|
|
|
Skip;
|
|
|
|
end; {else}
|
|
|
|
1:
|
|
|
|
if errorFound then begin
|
|
|
|
while opStack <> nil do begin
|
|
|
|
op := opStack;
|
|
|
|
opStack := op^.next;
|
|
|
|
dispose(op);
|
|
|
|
end; {while}
|
|
|
|
while stack <> nil do begin
|
|
|
|
sp := stack;
|
|
|
|
stack := sp^.next;
|
|
|
|
DisposeTree(sp);
|
|
|
|
end; {while}
|
|
|
|
ExpressionTree := nil;
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
ExpressionTree := stack;
|
|
|
|
end; {ExpressionTree}
|
|
|
|
|
|
|
|
|
|
|
|
procedure CompareToZero {op: pcodes};
|
|
|
|
|
|
|
|
{ Compare the result on tos to zero. }
|
|
|
|
{ }
|
|
|
|
{ This procedure is used by the logical statements to compare }
|
|
|
|
{ _any_ scalar result to zero, giving a boolean result. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ op - operation to use on the compare }
|
|
|
|
|
|
|
|
var
|
|
|
|
bt: baseTypeEnum; {base type of loaded value}
|
|
|
|
|
|
|
|
begin {CompareToZero}
|
2015-12-30 20:32:40 -06:00
|
|
|
if expressionType^.kind in [pointerType,arrayType] then
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := uLongPtr;
|
|
|
|
if expressionType^.kind = scalarType then begin
|
|
|
|
bt := UsualUnaryConversions;
|
|
|
|
case bt of
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen1t(pc_ldc, 0, cgWord);
|
|
|
|
cgLong,cgULong:
|
|
|
|
GenLdcLong(0);
|
2021-02-04 12:39:27 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
GenLdcQuad(longlong0);
|
2017-10-21 18:40:19 -05:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
|
|
|
GenLdcReal(0.0);
|
|
|
|
otherwise:
|
|
|
|
Error(47);
|
|
|
|
end; {case}
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0t(op, bt);
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(47);
|
|
|
|
end; {CompareToZero}
|
|
|
|
|
|
|
|
|
|
|
|
procedure FreeTemp{labelNum, size: integer};
|
|
|
|
|
|
|
|
{ place a temporary label in the available label list }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ labelNum - number of the label to free }
|
|
|
|
{ size - size of the variable }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ tempList - list of free labels }
|
|
|
|
|
|
|
|
var
|
|
|
|
tl: tempPtr; {work pointer}
|
|
|
|
|
|
|
|
begin {FreeTemp}
|
|
|
|
new(tl);
|
|
|
|
tl^.next := tempList;
|
|
|
|
tl^.last := nil;
|
|
|
|
tl^.labelNum := labelNum;
|
|
|
|
tl^.size := size;
|
|
|
|
if tempList <> nil then
|
|
|
|
tempList^.last := tl;
|
|
|
|
tempList := tl;
|
|
|
|
end; {FreeTemp}
|
|
|
|
|
|
|
|
|
|
|
|
function GetTemp{size: integer): integer};
|
|
|
|
|
|
|
|
{ find a temporary work variable }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ size - size of the variable }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ tempList - list of free labels }
|
|
|
|
{ }
|
|
|
|
{ Returns the label number. }
|
|
|
|
|
|
|
|
label 1;
|
|
|
|
|
|
|
|
var
|
|
|
|
lcodeGeneration: boolean; {local copy of codeGeneration}
|
|
|
|
ln: integer; {label number}
|
|
|
|
tl: tempPtr; {work pointer}
|
|
|
|
|
|
|
|
begin {GetTemp}
|
|
|
|
{try to find a temp from the existing list}
|
|
|
|
tl := tempList;
|
|
|
|
while tl <> nil do begin
|
|
|
|
if tl^.size = size then begin
|
|
|
|
|
|
|
|
{found an old one - use it}
|
|
|
|
if tl^.last = nil then
|
|
|
|
tempList := tl^.next
|
|
|
|
else
|
|
|
|
tl^.last^.next := tl^.next;
|
|
|
|
if tl^.next <> nil then
|
|
|
|
tl^.next^.last := tl^.last;
|
|
|
|
GetTemp := tl^.labelNum;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
tl := tl^.next;
|
|
|
|
end; {while}
|
|
|
|
|
|
|
|
{none found - get a new one}
|
|
|
|
ln := GetLocalLabel;
|
|
|
|
GetTemp := ln;
|
|
|
|
lcodeGeneration := codeGeneration;
|
|
|
|
codeGeneration := true;
|
|
|
|
Gen2(dc_loc, ln, size);
|
|
|
|
codeGeneration := lCodeGeneration and (numErrors = 0);
|
|
|
|
1:
|
|
|
|
end; {GetTemp}
|
|
|
|
|
|
|
|
|
|
|
|
procedure LoadScalar (id: identPtr);
|
|
|
|
|
|
|
|
{ Load a scalar value. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ id - ident for value to load }
|
|
|
|
|
|
|
|
var
|
|
|
|
tp: baseTypeEnum; {base type}
|
|
|
|
|
|
|
|
begin {LoadScalar}
|
2019-01-27 11:53:09 -06:00
|
|
|
if id^.itype^.kind = scalarType then
|
|
|
|
tp := id^.itype^.baseType
|
|
|
|
else {if id^.itype^.kind in [pointerType,arrayType] then}
|
|
|
|
tp := cgULong;
|
2017-10-21 18:40:19 -05:00
|
|
|
case id^.storage of
|
|
|
|
stackFrame, parameter:
|
|
|
|
Gen2t(pc_lod, id^.lln, 0, tp);
|
|
|
|
external, global, private:
|
|
|
|
Gen1tName(pc_ldo, 0, tp, id^.name);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
|
|
|
end; {LoadScalar}
|
|
|
|
|
|
|
|
|
|
|
|
procedure Cast(tp: typePtr);
|
|
|
|
|
|
|
|
{ Cast the current expression to the stated type }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tp - type to cast to }
|
|
|
|
{ }
|
|
|
|
{ inputs: }
|
|
|
|
{ expressionType - type of the expression to cast }
|
|
|
|
{ }
|
|
|
|
{ outputs: }
|
|
|
|
{ expressionType - set to result type }
|
|
|
|
|
|
|
|
var
|
|
|
|
et,rt: baseTypeEnum; {work variables}
|
|
|
|
|
|
|
|
begin {Cast}
|
2021-01-25 21:22:58 -06:00
|
|
|
if (tp^.kind = scalarType) and (tp^.cType = ctBool) then begin
|
|
|
|
CompareToZero(pc_neq);
|
|
|
|
end {if}
|
|
|
|
else if (tp^.kind = scalarType) and (expressionType^.kind = scalarType) then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
rt := tp^.baseType;
|
|
|
|
et := expressionType^.baseType;
|
2021-03-06 22:28:39 -06:00
|
|
|
if (rt <> et) or (rt in [cgReal,cgDouble,cgComp]) then
|
2016-10-14 21:22:55 -05:00
|
|
|
if et <> cgVoid then
|
|
|
|
Gen2(pc_cnv, ord(et), ord(rt))
|
|
|
|
else
|
|
|
|
Error(40);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else if (tp^.kind = enumType) and (expressionType^.kind = scalarType) then begin
|
2016-10-14 21:22:55 -05:00
|
|
|
if expressionType^.baseType <> cgVoid then begin
|
|
|
|
rt := cgWord;
|
|
|
|
et := Unary(expressionType^.baseType);
|
|
|
|
if rt <> et then
|
|
|
|
Gen2(pc_cnv, ord(et), ord(rt));
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(40);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else if (tp^.kind = scalarType) and (expressionType^.kind = enumType) then begin
|
|
|
|
rt := Unary(tp^.baseType);
|
|
|
|
et := cgWord;
|
|
|
|
if rt <> et then
|
|
|
|
Gen2(pc_cnv, ord(et), ord(rt));
|
|
|
|
end {if}
|
|
|
|
else if tp^.kind = pointerType then begin
|
|
|
|
case expressionType^.kind of
|
|
|
|
|
|
|
|
scalarType:
|
|
|
|
if expressionType^.baseType in
|
2022-06-20 21:55:54 -05:00
|
|
|
[cgByte,cgUByte,cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen2(pc_cnv, ord(Unary(expressionType^.baseType)),
|
|
|
|
ord(cgULong))
|
|
|
|
else if doDispose then
|
|
|
|
Error(40);
|
|
|
|
|
|
|
|
arrayType,pointerType: ;
|
|
|
|
|
|
|
|
functionType,enumConst,enumType,definedType,structType,unionType:
|
|
|
|
if doDispose then
|
|
|
|
Error(40);
|
|
|
|
|
|
|
|
otherwise: Error(57);
|
|
|
|
|
|
|
|
end; {case}
|
|
|
|
end {else if}
|
|
|
|
else if expressionType^.kind in [pointerType,arrayType] then begin
|
|
|
|
case tp^.kind of
|
|
|
|
|
|
|
|
scalarType:
|
|
|
|
if tp^.baseType in
|
2022-06-20 21:55:54 -05:00
|
|
|
[cgByte,cgUByte,cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen2(pc_cnv, ord(cgULong),
|
|
|
|
ord(Unary(tp^.baseType)))
|
|
|
|
else if tp^.baseType = cgVoid then
|
|
|
|
Gen0t(pc_pop, UsualUnaryConversions)
|
|
|
|
else
|
|
|
|
Error(40);
|
|
|
|
|
|
|
|
otherwise:
|
|
|
|
Error(40);
|
|
|
|
end; {case}
|
|
|
|
end {else if}
|
|
|
|
else if expressionType^.kind in [structType,unionType] then begin
|
|
|
|
if tp^.kind = scalarType then
|
|
|
|
if tp^.baseType = cgVoid then
|
|
|
|
Gen0t(pc_pop, UsualUnaryConversions)
|
|
|
|
else Error(40)
|
|
|
|
else Error(40);
|
|
|
|
end {else if}
|
|
|
|
else
|
|
|
|
Error(40);
|
|
|
|
expressionType := tp;
|
|
|
|
end; {Cast}
|
|
|
|
|
|
|
|
|
|
|
|
procedure DoSelection {lType: typePtr; tree: tokenPtr; var size: longint};
|
|
|
|
|
|
|
|
{ Find the displacement & type for a selection operation }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ lType - structure/union type }
|
|
|
|
{ tree - right-hand tree }
|
|
|
|
{ size - disp into the structure/union }
|
|
|
|
{ }
|
|
|
|
{ returned in non-local variables: }
|
|
|
|
{ bitDisp - displacement to bit field }
|
|
|
|
{ bitSize - size of bit field }
|
|
|
|
{ unsigned - is the bit field unsigned? }
|
|
|
|
{ isBitField - is the field a bit field? }
|
|
|
|
{ }
|
2020-01-29 17:09:52 -06:00
|
|
|
{ variables: }
|
2017-10-21 18:40:19 -05:00
|
|
|
{ expressionType - set to the type of the field }
|
|
|
|
|
|
|
|
label 1;
|
|
|
|
|
|
|
|
var
|
|
|
|
ip: identPtr; {for scanning for the field}
|
2021-09-09 18:39:19 -05:00
|
|
|
qualifiers: typeQualifierSet; {type qualifiers}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
begin {DoSelection}
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr; {set defaults in case there is an error}
|
2017-10-21 18:40:19 -05:00
|
|
|
size := 0;
|
|
|
|
if tree^.token.class = identifier then begin
|
2021-09-09 18:39:19 -05:00
|
|
|
qualifiers := lType^.qualifiers;
|
|
|
|
while lType^.kind = definedType do begin
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := lType^.dType;
|
2021-09-09 18:39:19 -05:00
|
|
|
qualifiers := qualifiers + lType^.qualifiers;
|
|
|
|
end; {while}
|
2017-10-21 18:40:19 -05:00
|
|
|
if lType^.kind in [structType,unionType] then begin
|
|
|
|
ip := lType^.fieldList; {find a matching field}
|
|
|
|
while ip <> nil do begin
|
|
|
|
if ip^.name^ = tree^.token.name^ then begin
|
|
|
|
if ip^.isForwardDeclared then
|
|
|
|
ResolveForwardReference(ip);
|
|
|
|
size := ip^.disp; {match found - record parameters}
|
2021-09-09 18:39:19 -05:00
|
|
|
expressionType := MakeQualifiedType(ip^.itype, qualifiers);
|
2017-10-21 18:40:19 -05:00
|
|
|
bitDisp := ip^.bitDisp;
|
|
|
|
bitSize := ip^.bitSize;
|
|
|
|
isBitField := (bitSize+bitDisp) <> 0;
|
2021-01-25 21:22:58 -06:00
|
|
|
unsigned := (ip^.itype^.baseType in [cgUByte,cgUWord,cgULong])
|
|
|
|
or (ip^.itype^.cType = ctBool);
|
2017-10-21 18:40:19 -05:00
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
ip := ip^.next;
|
|
|
|
end; {while}
|
|
|
|
Error(81);
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(80);
|
|
|
|
end; {if}
|
|
|
|
1:
|
|
|
|
end; {DoSelection}
|
|
|
|
|
|
|
|
|
|
|
|
procedure L_Value(tree: tokenPtr);
|
|
|
|
|
|
|
|
{ Check for an l-value }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - expression tree to check }
|
|
|
|
|
|
|
|
var
|
|
|
|
kind: tokenEnum; {for efficiency}
|
|
|
|
|
|
|
|
begin {L_Value}
|
|
|
|
kind := tree^.token.kind;
|
|
|
|
|
|
|
|
{A variable identifier is an l-value unless it is a function or }
|
|
|
|
{non-parameter array }
|
2022-06-08 20:58:52 -05:00
|
|
|
if kind in [ident,compoundliteral] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
if tree^.id^.itype^.kind = arrayType then begin
|
|
|
|
if tree^.id^.storage <> parameter then
|
|
|
|
if doDispose then {prevent spurious errors}
|
|
|
|
Error(78);
|
|
|
|
end {if}
|
|
|
|
else if tree^.id^.itype^.kind in
|
|
|
|
[functionType,enumConst,enumType] then
|
|
|
|
if doDispose then {prevent spurious errors}
|
|
|
|
Error(78);
|
|
|
|
end {if}
|
|
|
|
|
|
|
|
{e.field is an l-value if and only if e is an l-value}
|
|
|
|
else if kind = dotch then
|
|
|
|
L_Value(tree^.left)
|
|
|
|
|
|
|
|
{Bypass cast operators }
|
|
|
|
{following test removed to flag bug for: }
|
|
|
|
{ int *p; long l; }
|
|
|
|
{ (long) p = l; }
|
|
|
|
{else if kind = castoper then
|
|
|
|
L_Value(tree^.left)}
|
|
|
|
|
|
|
|
{The result of an array subscript (a[i]), indirect selection, }
|
|
|
|
{or the indirection operator all show up as the uasterisk }
|
|
|
|
{operator at this point. All are l-values, but nothing else }
|
|
|
|
{not already allowed is an l-value. }
|
|
|
|
else if kind <> uasterisk then
|
|
|
|
if doDispose then {prevent spurious errors}
|
|
|
|
Error(78);
|
|
|
|
end; {L_Value}
|
|
|
|
|
|
|
|
|
|
|
|
procedure ChangePointer (op: pcodes; size: longint; tp: baseTypeEnum);
|
|
|
|
|
|
|
|
{ Add or subtract an integer to a pointer }
|
|
|
|
{ }
|
|
|
|
{ The stack has a pointer and an integer (integer on TOS). }
|
|
|
|
{ The integer is removed, multiplied by size, and either }
|
|
|
|
{ added to or subtracted from the pointer; the result }
|
|
|
|
{ replaces the pointer on the stack }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ op - operation (pc_adl or pc_sbl) }
|
|
|
|
{ size - size of one pointer element }
|
|
|
|
{ tp - type of the integer operand }
|
|
|
|
|
|
|
|
begin {ChangePointer}
|
2017-10-29 20:21:36 -05:00
|
|
|
if size = 0 then
|
|
|
|
Error(122);
|
2023-02-12 18:56:02 -06:00
|
|
|
if checkNullPointers then
|
|
|
|
Gen0(pc_ckn);
|
2017-10-21 18:40:19 -05:00
|
|
|
case tp of
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord: begin
|
|
|
|
if (size = long(size).lsw) and (op = pc_adl)
|
|
|
|
and smallMemoryModel and (tp in [cgUByte,cgUWord]) then begin
|
2017-10-29 20:25:43 -05:00
|
|
|
if size <> 1 then begin
|
|
|
|
Gen1t(pc_ldc, long(size).lsw, cgWord);
|
|
|
|
Gen0(pc_umi);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0t(pc_ixa, cgUWord);
|
|
|
|
end {if}
|
|
|
|
else if smallMemoryModel and (size = long(size).lsw) then begin
|
2017-10-29 20:25:43 -05:00
|
|
|
if size <> 1 then begin
|
|
|
|
Gen1t(pc_ldc, long(size).lsw, cgWord);
|
2017-11-12 17:21:05 -06:00
|
|
|
Gen0(pc_umi);
|
2017-10-29 20:25:43 -05:00
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen2(pc_cnv, ord(tp), ord(cgLong));
|
|
|
|
Gen0(op);
|
|
|
|
end {else if}
|
|
|
|
else begin
|
|
|
|
Gen2(pc_cnv, ord(tp), ord(cgLong));
|
2017-10-29 20:25:43 -05:00
|
|
|
if size <> 1 then begin
|
|
|
|
GenLdcLong(size);
|
|
|
|
Gen0(pc_mpl);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(op);
|
|
|
|
end;
|
|
|
|
end;
|
2021-02-04 22:05:02 -06:00
|
|
|
cgLong,cgULong,cgQuad,cgUQuad: begin
|
|
|
|
if tp in [cgQuad,cgUQuad] then
|
|
|
|
Gen2(pc_cnv, ord(tp), ord(cgLong));
|
2017-10-29 20:25:43 -05:00
|
|
|
if size <> 1 then begin
|
|
|
|
GenLdcLong(size);
|
2021-02-04 22:05:02 -06:00
|
|
|
if tp in [cgLong,cgQuad] then
|
2017-10-29 20:25:43 -05:00
|
|
|
Gen0(pc_mpl)
|
|
|
|
else
|
|
|
|
Gen0(pc_uml);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(op);
|
|
|
|
end;
|
|
|
|
otherwise:
|
|
|
|
Error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {ChangePointer}
|
|
|
|
|
|
|
|
|
|
|
|
procedure GenerateCode {tree: tokenPtr};
|
|
|
|
|
|
|
|
{ generate code from a fully formed expression tree }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - top of the expression tree to generate code from }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ expressionType - result type of the expression }
|
|
|
|
|
|
|
|
var
|
|
|
|
doingScalar: boolean; {temp; for assignment operators}
|
|
|
|
et: baseTypeEnum; {temp storage for a base type}
|
|
|
|
i: integer; {loop variable}
|
2023-02-28 22:36:42 -06:00
|
|
|
isConst: boolean; {is this a constant?}
|
2022-06-23 22:04:03 -05:00
|
|
|
isNullPtrConst: boolean; {is this a null pointer constant?}
|
2022-05-23 21:10:29 -05:00
|
|
|
isVolatile: boolean; {is this a volatile op?}
|
2017-10-21 18:40:19 -05:00
|
|
|
lType: typePtr; {type of operands}
|
|
|
|
kind: typeKind; {temp type kind}
|
|
|
|
size: longint; {size of an array element}
|
|
|
|
t1: integer; {temporary work space label number}
|
2022-06-23 22:04:03 -05:00
|
|
|
tlastWasNullPtrConst: boolean; {temp lastWasNullPtrConst}
|
2017-10-21 18:40:19 -05:00
|
|
|
tp: tokenPtr; {work pointer}
|
|
|
|
tType: typePtr; {temp type of operand}
|
|
|
|
|
|
|
|
lbitDisp,lbitSize: integer; {for temp storage}
|
|
|
|
lisBitField: boolean;
|
2022-06-08 20:58:52 -05:00
|
|
|
ldoDispose: boolean; {local copy of doDispose}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
|
2023-01-09 21:58:05 -06:00
|
|
|
procedure CheckForIncompleteStructType;
|
|
|
|
|
|
|
|
{ Check if expressionType is an incomplete struct/union type. }
|
|
|
|
|
|
|
|
var
|
|
|
|
tp: typePtr; {the type}
|
|
|
|
|
|
|
|
begin
|
|
|
|
tp := expressionType;
|
|
|
|
while tp^.kind = definedType do
|
|
|
|
tp := tp^.dType;
|
|
|
|
if tp^.kind in [structType,unionType] then
|
|
|
|
if tp^.size = 0 then
|
|
|
|
Error(187);
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
function ExpressionKind (tree: tokenPtr): typeKind;
|
|
|
|
|
|
|
|
{ returns the type of an expression }
|
|
|
|
{ }
|
2020-01-29 17:09:52 -06:00
|
|
|
{ This subroutine is used to see if + and - operations }
|
2017-10-21 18:40:19 -05:00
|
|
|
{ should do pointer addition. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - top of the expression tree to check }
|
|
|
|
|
|
|
|
var
|
|
|
|
ldoDispose: boolean; {local copy of doDispose}
|
|
|
|
lcodeGeneration: boolean; {local copy of codeGeneration}
|
|
|
|
lexpressionType: typePtr; {local copy of expressionType}
|
|
|
|
|
|
|
|
begin {ExpressionKind}
|
|
|
|
ldoDispose := doDispose; {inhibit disposing of the tree}
|
|
|
|
doDispose := false;
|
|
|
|
lcodeGeneration := codeGeneration; {inhibit code generation}
|
|
|
|
codeGeneration := false;
|
|
|
|
lexpressionType := expressionType; {save the expression type}
|
|
|
|
|
|
|
|
GenerateCode(tree); {get the type}
|
2018-03-25 20:56:15 -05:00
|
|
|
while expressionType^.kind = definedType do
|
|
|
|
expressionType := expressionType^.dType;
|
2017-10-21 18:40:19 -05:00
|
|
|
ExpressionKind := expressionType^.kind;
|
|
|
|
|
2020-01-29 17:09:52 -06:00
|
|
|
doDispose := ldoDispose; {restore the volatile variables}
|
2017-10-21 18:40:19 -05:00
|
|
|
codeGeneration := lCodeGeneration and (numErrors = 0);
|
|
|
|
expressionType := lexpressionType;
|
|
|
|
end; {ExpressionKind}
|
|
|
|
|
|
|
|
|
2023-02-12 18:56:02 -06:00
|
|
|
procedure LoadAddress (tree: tokenPtr; nullCheck: boolean);
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{ load the address of an l-value }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - top of the expression tree to load the }
|
|
|
|
{ address of }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ expressionType - result type of the expression }
|
|
|
|
{ isBitField - this variable is set to false so that }
|
|
|
|
{ it can be used to see if DoSelection was called }
|
|
|
|
{ and located a bit field }
|
|
|
|
|
|
|
|
label 1;
|
|
|
|
|
|
|
|
var
|
|
|
|
eType: typePtr; {work pointer}
|
|
|
|
i: integer; {loop variable}
|
|
|
|
size: longint; {disp in record}
|
|
|
|
tname: stringPtr; {temp name pointer}
|
|
|
|
|
|
|
|
begin {LoadAddress}
|
|
|
|
isBitField := false;
|
|
|
|
if tree^.token.kind = ident then begin
|
|
|
|
|
|
|
|
{load the address of an identifier}
|
|
|
|
with tree^.id^ do begin
|
|
|
|
tname := name;
|
|
|
|
if itype^.kind = functionType then begin
|
|
|
|
if itype^.isPascal then begin
|
|
|
|
tname := pointer(Malloc(length(name^)+1));
|
|
|
|
CopyString(pointer(tname), pointer(name));
|
|
|
|
for i := 1 to length(tname^) do
|
|
|
|
if tname^[i] in ['a'..'z'] then
|
|
|
|
tname^[i] := chr(ord(tname^[i]) & $5F);
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
|
|
|
case storage of
|
|
|
|
stackFrame: Gen2(pc_lda, lln, 0);
|
|
|
|
parameter: if itype^.kind = arrayType then
|
|
|
|
Gen2t(pc_lod, pln, 0, cgULong)
|
|
|
|
else
|
|
|
|
Gen2(pc_lda, pln, 0);
|
|
|
|
external,
|
|
|
|
global,
|
|
|
|
private: Gen1Name(pc_lao, 0, tname);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
2022-06-19 17:55:08 -05:00
|
|
|
expressionType := MakePointerTo(iType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {with}
|
|
|
|
end {if}
|
2022-06-08 20:58:52 -05:00
|
|
|
else if tree^.token.kind = compoundliteral then begin
|
|
|
|
|
|
|
|
{evaluate a compound literal and load its address}
|
|
|
|
AutoInit(tree^.id, 0, true);
|
|
|
|
tree^.token.kind := ident;
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree, false);
|
2022-06-08 20:58:52 -05:00
|
|
|
tree^.token.kind := compoundliteral;
|
|
|
|
Gen0t(pc_bno, cgULong);
|
|
|
|
end {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
else if tree^.token.kind = uasterisk then begin
|
|
|
|
|
|
|
|
{load the address of the item pointed to by the pointer}
|
|
|
|
GenerateCode(tree^.left);
|
2023-02-12 18:56:02 -06:00
|
|
|
if nullCheck then
|
|
|
|
Gen0(pc_ckp);
|
2015-12-30 23:03:49 -06:00
|
|
|
isBitField := false;
|
2020-01-29 22:52:10 -06:00
|
|
|
if not (expressionType^.kind in [pointerType,arrayType,functionType]) then
|
|
|
|
Error(79);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
|
|
|
else if tree^.token.kind = dotch then begin
|
|
|
|
|
|
|
|
{load the address of a field of a record}
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree^.left, nullCheck);
|
2017-10-21 18:40:19 -05:00
|
|
|
eType := expressionType;
|
|
|
|
if eType^.kind in [arrayType,pointerType] then begin
|
|
|
|
if eType^.kind = arrayType then
|
|
|
|
eType := eType^.aType
|
|
|
|
else if eType^.kind = pointerType then
|
|
|
|
eType := eType^.pType;
|
|
|
|
DoSelection(eType, tree^.right, size);
|
|
|
|
if size <> 0 then
|
|
|
|
if size & $00007FFF = size then
|
|
|
|
Gen1t(pc_inc, long(size).lsw, cgULong)
|
|
|
|
else begin
|
|
|
|
GenLdcLong(size);
|
|
|
|
Gen0(pc_adl);
|
|
|
|
end; {else}
|
2022-06-19 17:55:08 -05:00
|
|
|
expressionType := MakePointerTo(expressionType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(79);
|
|
|
|
end {else if}
|
|
|
|
else if tree^.token.kind = castoper then begin
|
|
|
|
|
|
|
|
{load the address of a field of a record}
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree^.left, nullCheck);
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := tree^.castType;
|
2022-06-19 17:55:08 -05:00
|
|
|
if expressionType^.kind <> arrayType then
|
|
|
|
expressionType := MakePointerTo(expressionType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {else if}
|
|
|
|
|
2021-09-17 22:04:10 -05:00
|
|
|
else if ExpressionKind(tree) in [arrayType,pointerType,structType,unionType]
|
2023-02-12 18:56:02 -06:00
|
|
|
then begin
|
|
|
|
GenerateCode(tree);
|
|
|
|
if nullCheck then
|
|
|
|
Gen0(pc_ckp);
|
|
|
|
end {else if}
|
2019-12-23 20:34:54 -06:00
|
|
|
else begin
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr; {set default type in case of error}
|
2017-10-21 18:40:19 -05:00
|
|
|
if doDispose then {prevent spurious errors}
|
|
|
|
Error(78);
|
2019-12-23 20:34:54 -06:00
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
1:
|
|
|
|
end; {LoadAddress}
|
|
|
|
|
|
|
|
|
|
|
|
procedure DoIncDec (tree: tokenPtr; pc_l, pc_g, pc_i: pcodes);
|
|
|
|
|
|
|
|
{ do ++ and -- }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ tree - tree to generate the instruction for }
|
|
|
|
{ pc_l - op code for a local ++ or -- }
|
|
|
|
{ pc_g - op code for a global ++ or -- }
|
|
|
|
{ pc_i - op code for an indirect ++ or -- }
|
|
|
|
|
|
|
|
label 1;
|
|
|
|
|
|
|
|
var
|
|
|
|
baseType: baseTypeEnum; {type of operation}
|
|
|
|
lSize: longint; {number to inc or dec by}
|
|
|
|
iSize: integer; {number to inc or dec by}
|
|
|
|
tp: baseTypeEnum; {type of operand}
|
|
|
|
|
|
|
|
|
|
|
|
procedure IncOrDec (inc: boolean);
|
|
|
|
|
|
|
|
{ Increment or decrement a number on TOS }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ inc - increment the number? }
|
|
|
|
|
|
|
|
begin {IncOrDec}
|
|
|
|
case expressionType^.kind of
|
|
|
|
|
|
|
|
scalarType:
|
|
|
|
case tp of
|
|
|
|
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord: begin
|
|
|
|
Gen1t(pc_ldc, 1, cgWord);
|
|
|
|
if inc then
|
|
|
|
Gen0(pc_adi)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbi);
|
2021-01-25 21:22:58 -06:00
|
|
|
if expressionType^.cType = ctBool then begin
|
|
|
|
CompareToZero(pc_neq);
|
|
|
|
expressionType := boolPtr;
|
|
|
|
end {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
end;
|
|
|
|
|
|
|
|
cgLong,cgULong: begin
|
|
|
|
GenLdcLong(1);
|
|
|
|
if inc then
|
|
|
|
Gen0(pc_adl)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbl);
|
|
|
|
end;
|
|
|
|
|
2021-02-04 12:35:28 -06:00
|
|
|
cgQuad,cgUQuad: begin
|
|
|
|
GenLdcQuad(longlong1);
|
|
|
|
if inc then
|
|
|
|
Gen0(pc_adq)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbq);
|
|
|
|
end;
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended: begin
|
|
|
|
GenLdcReal(1.0);
|
|
|
|
if inc then
|
|
|
|
Gen0(pc_adr)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbr);
|
|
|
|
end;
|
|
|
|
|
|
|
|
otherwise: Error(57);
|
|
|
|
|
|
|
|
end; {case}
|
|
|
|
|
|
|
|
pointerType,arrayType: begin
|
2023-02-12 18:56:02 -06:00
|
|
|
if checkNullPointers then
|
|
|
|
Gen0(pc_ckp);
|
2017-10-21 18:40:19 -05:00
|
|
|
GenldcLong(expressionType^.pType^.size);
|
|
|
|
if inc then
|
|
|
|
Gen0(pc_adl)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbl);
|
|
|
|
end;
|
|
|
|
|
|
|
|
otherwise: ;
|
|
|
|
|
|
|
|
end; {case}
|
|
|
|
end; {IncOrDec}
|
|
|
|
|
|
|
|
|
|
|
|
begin {DoIncDec}
|
|
|
|
L_Value(tree);
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
if (tree^.token.kind = ident)
|
|
|
|
and ((tree^.id^.iType^.kind in [scalarType,pointerType])
|
|
|
|
or ((tree^.id^.iType^.kind = arrayType) and (tree^.id^.storage = parameter)))
|
|
|
|
then
|
|
|
|
with tree^.id^ do begin
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{check for ++ or -- of a constant}
|
Record volatile and restrict qualifiers in types.
These are needed to correctly distinguish pointer types in _Generic. They should also be used for type compatibility checks in other contexts, but currently are not.
This also fixes a couple small problems related to type qualifiers:
*restrict was not allowed to appear after * in type-names
*volatile status was not properly recorded in sym files
Here is an example of using _Generic to distinguish pointer types based on the qualifiers of the pointed-to type:
#include <stdio.h>
#define f(e) _Generic((e),\
int * restrict *: 1,\
int * volatile const *: 2,\
int **: 3,\
default: 0)
#define g(e) _Generic((e),\
int *: 1,\
const int *: 2,\
volatile int *: 3,\
default: 0)
int main(void) {
int * restrict * p1;
int * volatile const * p2;
int * const * p3;
// should print "1 2 0 1"
printf("%i %i %i %i\n", f(p1), f(p2), f(p3), f((int * restrict *)0));
int *q1;
const int *q2;
volatile int *q3;
const volatile int *q4;
// should print "1 2 3 0"
printf("%i %i %i %i\n", g(q1), g(q2), g(q3), g(q4));
}
Here is an example of a problem resulting from volatile not being recorded in sym files (if a sym file was present, the read of x was lifted out of the loop):
#pragma optimize -1
static volatile int x;
#include <stdio.h>
int main(void) {
int y;
for (unsigned i = 0; i < 100; i++) {
y = x*2 + 7;
}
}
2021-08-29 21:10:20 -05:00
|
|
|
if tqConst in iType^.qualifiers then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(93);
|
|
|
|
|
|
|
|
{do an efficient ++ or -- on a named location}
|
|
|
|
if iType^.kind = scalarType then begin
|
|
|
|
iSize := 1;
|
|
|
|
baseType := iType^.baseType;
|
2021-02-04 12:35:28 -06:00
|
|
|
if (baseType in [cgReal,cgDouble,cgComp,cgExtended,cgQuad,cgUQuad])
|
2021-01-25 21:22:58 -06:00
|
|
|
or (iType^.cType = ctBool) then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2021-01-25 21:22:58 -06:00
|
|
|
{do real or bool inc or dec}
|
2017-10-21 18:40:19 -05:00
|
|
|
LoadScalar(tree^.id); {load the value}
|
2021-01-25 21:22:58 -06:00
|
|
|
if pc_l in [pc_lli,pc_lld] then
|
2021-03-09 18:54:31 -06:00
|
|
|
if iType^.cType in [ctBool,ctFloat,ctDouble,ctLongDouble,
|
|
|
|
ctComp] then begin
|
|
|
|
t1 := GetTemp(ord(iType^.size));
|
|
|
|
Gen2t(pc_cop, t1, 0, iType^.baseType);
|
2021-01-25 21:22:58 -06:00
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
tp := baseType;
|
|
|
|
expressionType := iType;
|
|
|
|
IncOrDec(pc_l in [pc_lli,pc_lil]); {do the ++ or --}
|
|
|
|
case storage of {save the result}
|
|
|
|
stackFrame, parameter:
|
|
|
|
Gen2t(pc_cop, lln, 0, baseType);
|
|
|
|
external, global, private:
|
|
|
|
Gen1tName(pc_cpo, 0, baseType, name);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
|
|
|
{correct the value for postfix ops}
|
2021-01-25 21:22:58 -06:00
|
|
|
if pc_l in [pc_lli,pc_lld] then
|
2021-03-09 18:54:31 -06:00
|
|
|
if iType^.cType in [ctBool,ctFloat,ctDouble,ctLongDouble,
|
|
|
|
ctComp] then begin
|
|
|
|
Gen0t(pc_pop, iType^.baseType);
|
|
|
|
Gen2t(pc_lod, t1, 0, iType^.baseType);
|
|
|
|
Gen0t(pc_bno, iType^.baseType);
|
|
|
|
FreeTemp(t1, ord(iType^.size));
|
2021-01-25 21:22:58 -06:00
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
IncOrDec(pc_l = pc_lld);
|
|
|
|
if iType^.cType = ctBool then
|
|
|
|
expressionType := boolPtr
|
2021-02-04 12:35:28 -06:00
|
|
|
else if baseType = cgQuad then
|
|
|
|
expressionType := longLongPtr
|
|
|
|
else if baseType = cgUQuad then
|
|
|
|
expressionType := ulongLongPtr
|
2021-01-25 21:22:58 -06:00
|
|
|
else
|
|
|
|
expressionType := doublePtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
goto 1;
|
2022-07-12 18:35:52 -05:00
|
|
|
end {if}
|
|
|
|
else if baseType = cgVoid then
|
|
|
|
Error(65);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
2019-01-27 11:53:09 -06:00
|
|
|
else {if iType^.kind in [pointerType,arrayType] then} begin
|
2017-10-21 18:40:19 -05:00
|
|
|
lSize := iType^.pType^.size;
|
2017-10-29 20:21:36 -05:00
|
|
|
if lSize = 0 then
|
|
|
|
Error(122);
|
2023-02-12 18:56:02 -06:00
|
|
|
if (long(lSize).msw <> 0) or checkNullPointers then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2023-02-12 18:56:02 -06:00
|
|
|
{handle inc/dec of >64K or with null pointer check}
|
2017-10-21 18:40:19 -05:00
|
|
|
LoadScalar(tree^.id);
|
2023-02-12 18:56:02 -06:00
|
|
|
if checkNullPointers then
|
|
|
|
Gen0(pc_ckp);
|
2017-10-21 18:40:19 -05:00
|
|
|
GenLdcLong(lSize);
|
|
|
|
if pc_l in [pc_lli,pc_lil] then
|
|
|
|
Gen0(pc_adl)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbl);
|
2019-12-23 19:59:18 -06:00
|
|
|
with tree^.id^ do
|
2017-10-21 18:40:19 -05:00
|
|
|
case storage of
|
|
|
|
stackFrame, parameter:
|
|
|
|
Gen2t(pc_cop, lln, 0, cgULong);
|
|
|
|
external, global, private:
|
|
|
|
Gen1tName(pc_cpo, 0, cgULong, name);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
|
|
|
if pc_l in [pc_lli,pc_lld] then begin
|
|
|
|
GenLdcLong(lSize);
|
|
|
|
if pc_l = pc_lld then
|
|
|
|
Gen0(pc_adl)
|
|
|
|
else
|
|
|
|
Gen0(pc_sbl);
|
|
|
|
end; {if}
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
baseType := cgULong;
|
|
|
|
iSize := long(lSize).lsw;
|
|
|
|
end; {else}
|
|
|
|
case storage of
|
|
|
|
stackFrame, parameter:
|
|
|
|
Gen2t(pc_l, lln, iSize, baseType);
|
|
|
|
external, global, private:
|
|
|
|
Gen2tName(pc_g, iSize, 0, baseType, name);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
|
|
|
expressionType := itype;
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
end {with}
|
|
|
|
else begin
|
2017-10-21 18:40:19 -05:00
|
|
|
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
{do an indirect ++ or --}
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree, checkNullPointers); {get the address to save to}
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
if expressionType^.kind = arrayType then
|
|
|
|
expressionType := expressionType^.aType
|
|
|
|
else if expressionType^.kind = pointerType then
|
|
|
|
expressionType := expressionType^.pType;
|
2022-07-12 18:35:52 -05:00
|
|
|
if tqConst in expressionType^.qualifiers then
|
|
|
|
Error(93);
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
if expressionType^.kind = scalarType then
|
|
|
|
if expressionType^.baseType in
|
|
|
|
[cgByte,cgUByte,cgWord,cgUWord,cgReal,cgDouble,cgComp,cgExtended] then
|
|
|
|
tp := expressionType^.baseType
|
|
|
|
else
|
|
|
|
tp := UsualUnaryConversions
|
2022-07-12 18:35:52 -05:00
|
|
|
else begin
|
|
|
|
if expressionType^.kind in [structType,unionType,definedType] then
|
|
|
|
Error(66);
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
tp := UsualUnaryConversions;
|
2022-07-12 18:35:52 -05:00
|
|
|
end; {else}
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
if (tp in [cgByte,cgUByte,cgWord,cgUword])
|
|
|
|
and (expressionType^.cType <> ctBool)
|
|
|
|
and not isBitField then
|
|
|
|
Gen0t(pc_i, tp) {do indirect inc/dec}
|
2022-07-12 18:35:52 -05:00
|
|
|
else if tp = cgVoid then
|
|
|
|
Error(65)
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
else begin
|
|
|
|
t1 := GetTemp(cgLongSize);
|
|
|
|
Gen2t(pc_str, t1, 0, cgULong);
|
|
|
|
Gen2t(pc_lod, t1, 0, cgULong);
|
|
|
|
Gen2t(pc_lod, t1, 0, cgULong);
|
|
|
|
FreeTemp(t1, cgLongSize);
|
|
|
|
{load the value}
|
|
|
|
if isBitField then begin
|
|
|
|
if unsigned then
|
|
|
|
Gen2t(pc_lbu, bitDisp, bitSize, tp)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
Gen2t(pc_lbf, bitDisp, bitSize, tp);
|
|
|
|
end {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(tqVolatile in expressionType^.qualifiers), 0, tp);
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
if pc_l in [pc_lli,pc_lld] then
|
2024-02-09 20:41:53 -06:00
|
|
|
if (expressionType^.kind = scalarType) and
|
|
|
|
(expressionType^.cType in
|
|
|
|
[ctBool,ctFloat,ctDouble,ctLongDouble,ctComp])
|
|
|
|
then begin
|
2021-03-09 18:54:31 -06:00
|
|
|
t1 := GetTemp(ord(expressionType^.size));
|
|
|
|
Gen2t(pc_cop, t1, 0, expressionType^.baseType);
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
end; {if}
|
|
|
|
IncOrDec(pc_l in [pc_lli,pc_lil]); {do the ++ or --}
|
|
|
|
if isBitField then {copy the value}
|
|
|
|
Gen2t(pc_cbf, bitDisp, bitSize, tp)
|
|
|
|
else
|
|
|
|
Gen0t(pc_cpi, tp);
|
|
|
|
Gen0t(pc_bno, tp);
|
|
|
|
if pc_l in [pc_lli,pc_lld] then {correct the value for postfix ops}
|
2024-02-09 20:41:53 -06:00
|
|
|
if (expressionType^.kind = scalarType) and
|
|
|
|
(expressionType^.cType in
|
|
|
|
[ctBool,ctFloat,ctDouble,ctLongDouble,ctComp])
|
|
|
|
then begin
|
2021-03-09 18:54:31 -06:00
|
|
|
Gen0t(pc_pop, expressionType^.baseType);
|
|
|
|
Gen2t(pc_lod, t1, 0, expressionType^.baseType);
|
|
|
|
Gen0t(pc_bno, expressionType^.baseType);
|
|
|
|
FreeTemp(t1, ord(expressionType^.size));
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
IncOrDec(pc_l = pc_lld);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
Fix several issues with evaluation of the ++ and -- operators.
These would generally not work correctly on bit-fields, or on floating-point values that were in a structure or were accessed via a pointer.
The below program is an example that would demonstrate problems:
#include <stdio.h>
int main(void) {
struct {
signed int i:7;
unsigned long int j:6;
_Bool b:1;
double d;
} s = {-10, -20, 0, 5.0};
double d = 70.0, *dp = &d;
printf("%i\n", (int)s.i++);
printf("%i\n", (int)s.i--);
printf("%i\n", (int)++s.i);
printf("%i\n", (int)--s.i);
printf("%i\n", (int)s.i);
printf("%i\n", (int)s.j++);
printf("%i\n", (int)s.j--);
printf("%i\n", (int)++s.j);
printf("%i\n", (int)--s.j);
printf("%i\n", (int)s.j);
printf("%i\n", s.b++);
printf("%i\n", s.b--);
printf("%i\n", ++s.b);
printf("%i\n", --s.b);
printf("%i\n", s.b);
printf("%f\n", s.d++);
printf("%f\n", s.d--);
printf("%f\n", ++s.d);
printf("%f\n", --s.d);
printf("%f\n", s.d);
printf("%f\n", (*dp)++);
printf("%f\n", (*dp)--);
printf("%f\n", ++*dp);
printf("%f\n", --*dp);
printf("%f\n", *dp);
}
2021-01-26 12:31:54 -06:00
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
1:
|
|
|
|
end; {DoIncDec}
|
|
|
|
|
|
|
|
|
|
|
|
procedure FunctionCall (tree: tokenPtr);
|
|
|
|
|
|
|
|
{ generate the actual function call }
|
|
|
|
|
|
|
|
var
|
|
|
|
fName: stringPtr; {uppercase file name}
|
|
|
|
fntype: typePtr; {temp function type}
|
|
|
|
ftree: tokenPtr; {function address tree}
|
|
|
|
ftype: typePtr; {function type}
|
2022-12-08 19:27:37 -06:00
|
|
|
hasVarargs: boolean; {varargs call with 1+ varargs passed?}
|
2017-10-21 18:40:19 -05:00
|
|
|
i: integer; {loop variable}
|
|
|
|
indirect: boolean; {is this an indirect call?}
|
|
|
|
ldoDispose: boolean; {local copy of doDispose}
|
|
|
|
lcodeGeneration: boolean; {local copy of codeGeneration}
|
|
|
|
|
|
|
|
|
|
|
|
procedure FunctionParms (parms: tokenPtr; fType: typePtr);
|
|
|
|
|
|
|
|
{ Generate a function call. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ parms - parameter list }
|
|
|
|
{ fType - function type }
|
|
|
|
|
|
|
|
var
|
|
|
|
ldoDispose: boolean; {local copy of doDispose}
|
|
|
|
numParms: integer; {# of parameters generated}
|
|
|
|
parameters: parameterPtr; {next prototyped parameter}
|
|
|
|
pCount: integer; {# of parameters prototyped}
|
|
|
|
prototype: boolean; {is the function prototyped?}
|
2018-03-23 21:51:27 -04:00
|
|
|
tp: tokenPtr; {work pointers}
|
|
|
|
fp, tfp: fmtArgPtr;
|
|
|
|
fmt: fmt_type;
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
|
|
|
|
procedure Reverse;
|
|
|
|
|
|
|
|
{ Reverse the parameter list }
|
|
|
|
|
|
|
|
var
|
|
|
|
p1,p2,p3: tokenPtr; {work pointers}
|
|
|
|
|
|
|
|
begin {Reverse}
|
|
|
|
p3 := parms; {remove the last entry}
|
|
|
|
p1 := parms;
|
|
|
|
p2 := nil;
|
|
|
|
while p3^.right <> nil do begin
|
|
|
|
p2 := p3;
|
|
|
|
p3 := p3^.right;
|
|
|
|
end; {while}
|
|
|
|
if p2 <> nil then
|
|
|
|
p2^.right := nil
|
|
|
|
else
|
|
|
|
p1 := nil;
|
|
|
|
while p1 <> nil do begin {reverse the remaining parms}
|
|
|
|
p2 := p1;
|
|
|
|
p1 := p1^.right;
|
|
|
|
p2^.right := p3;
|
|
|
|
p3 := p2;
|
|
|
|
end; {while}
|
|
|
|
parms := p3;
|
|
|
|
end; {Reverse}
|
|
|
|
|
|
|
|
|
|
|
|
begin {FunctionParms}
|
|
|
|
{check the validity of the parameter list}
|
|
|
|
if ftype^.isPascal then {reverse parms for pascal calls}
|
|
|
|
Reverse;
|
|
|
|
tp := parms; {set up to check types}
|
|
|
|
prototype := ftype^.prototyped;
|
|
|
|
parameters := ftype^.parameterList;
|
|
|
|
pCount := 1;
|
2018-03-23 21:51:27 -04:00
|
|
|
fmt := fmt_none;
|
|
|
|
fp := nil;
|
|
|
|
|
2023-02-12 19:19:28 -06:00
|
|
|
if (lint & lintPrintf) <> 0 then
|
|
|
|
if fType^.varargs then
|
|
|
|
if not indirect then
|
|
|
|
if ftree^.id^.storage <> private then
|
|
|
|
fmt := FormatClassify(ftree^.id^.name^);
|
2018-03-23 21:51:27 -04:00
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
while parameters <> nil do begin {count the prototypes}
|
|
|
|
pCount := pCount+1;
|
|
|
|
parameters := parameters^.next;
|
|
|
|
end; {while}
|
|
|
|
parameters := ftype^.parameterList;
|
|
|
|
if prototype then begin {check for wrong # of parms}
|
|
|
|
while tp <> nil do begin {count the parms}
|
|
|
|
pCount := pCount-1;
|
|
|
|
tp := tp^.right;
|
|
|
|
end; {while}
|
|
|
|
tp := parms;
|
2022-12-08 19:27:37 -06:00
|
|
|
if pCount <> 0 then
|
|
|
|
if ftype^.varargs and (pcount < 0) then
|
|
|
|
hasVarargs := true
|
|
|
|
else
|
|
|
|
Error(85);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {if}
|
|
|
|
|
2018-03-23 21:51:27 -04:00
|
|
|
tp := parms;
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
{generate the parameters}
|
|
|
|
numParms := 0;
|
|
|
|
lDoDispose := doDispose;
|
|
|
|
doDispose := false;
|
|
|
|
while tp <> nil do begin
|
|
|
|
if tp^.middle <> nil then begin
|
2023-03-01 22:20:33 -06:00
|
|
|
GenerateCode(tp^.middle);
|
2024-04-01 20:06:26 -05:00
|
|
|
if expressionType^.kind in [structType,unionType,definedType]
|
|
|
|
then begin
|
|
|
|
tType := expressionType;
|
|
|
|
while tType^.kind = definedType do
|
|
|
|
tType := tType^.dType;
|
|
|
|
if tType^.kind in [structType,unionType] then begin
|
|
|
|
if tType^.size & $FFFF8000 <> 0 then
|
|
|
|
Error(61);
|
|
|
|
Gen1t(pc_ldc, long(tType^.size).lsw, cgWord);
|
|
|
|
Gen0(pc_psh);
|
|
|
|
end; {if}
|
2023-03-01 22:20:33 -06:00
|
|
|
end; {if}
|
2018-03-23 21:51:27 -04:00
|
|
|
if fmt <> fmt_none then begin
|
2023-03-01 22:20:33 -06:00
|
|
|
new(tfp);
|
|
|
|
tfp^.next := fp;
|
|
|
|
tfp^.tk := tp^.middle;
|
|
|
|
tfp^.ty := expressionType;
|
|
|
|
fp := tfp;
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
if prototype then begin
|
|
|
|
if pCount = 0 then begin
|
|
|
|
if parameters <> nil then begin
|
|
|
|
AssignmentConversion(parameters^.parameterType,
|
2016-12-13 21:46:11 -06:00
|
|
|
expressionType, lastWasConst, lastConst, true, false);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {if}
|
|
|
|
parameters := parameters^.next;
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
pCount := pCount+1;
|
|
|
|
end; {if}
|
|
|
|
Gen0t(pc_stk, UsualUnaryConversions);
|
|
|
|
if numParms <> 0 then
|
|
|
|
Gen0t(pc_bno, UsualUnaryConversions);
|
|
|
|
numParms := numParms+1;
|
|
|
|
end; {if}
|
|
|
|
tp := tp^.right;
|
|
|
|
end; {while}
|
2018-03-23 21:51:27 -04:00
|
|
|
|
|
|
|
if fmt <> fmt_none then FormatCheck(fmt, fp);
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
doDispose := lDoDispose;
|
|
|
|
if numParms = 0 then
|
|
|
|
Gen0(pc_nop);
|
|
|
|
|
|
|
|
if ftype^.isPascal then {restore parm order}
|
|
|
|
Reverse;
|
|
|
|
|
|
|
|
if doDispose then begin {dispose of leaf nodes}
|
|
|
|
DisposeTree(parms^.middle);
|
|
|
|
DisposeTree(parms^.right);
|
|
|
|
end; {if}
|
|
|
|
end; {FunctionParms}
|
|
|
|
|
|
|
|
|
|
|
|
begin {FunctionCall}
|
|
|
|
{find the type of the function}
|
|
|
|
indirect := true; {assume an indirect call}
|
2022-12-08 19:27:37 -06:00
|
|
|
hasVarargs := false; {assume no variable arguments}
|
2017-10-21 18:40:19 -05:00
|
|
|
ftree := tree^.left; {get the function tree}
|
|
|
|
if ftree^.token.kind = ident then {check for direct calls}
|
|
|
|
if ftree^.id^.itype^.kind = functionType then begin
|
|
|
|
indirect := false;
|
|
|
|
fType := ftree^.id^.itype; {get the function type}
|
|
|
|
end; {if}
|
|
|
|
if indirect then begin {get type for indirect call}
|
|
|
|
ldoDispose := doDispose;
|
|
|
|
doDispose := false;
|
|
|
|
lcodeGeneration := codeGeneration;
|
|
|
|
codeGeneration := false;
|
|
|
|
GenerateCode(ftree);
|
|
|
|
doDispose := ldoDispose;
|
|
|
|
codeGeneration := lCodeGeneration and (numErrors = 0);
|
|
|
|
ftype := expressionType;
|
|
|
|
while ftype^.kind in [pointerType,arrayType] do
|
|
|
|
ftype := ftype^.ptype;
|
|
|
|
end; {if}
|
|
|
|
|
|
|
|
{make sure the identifier is really a function}
|
|
|
|
if ftype^.kind <> functionType then
|
|
|
|
Error(114)
|
|
|
|
else begin
|
|
|
|
|
|
|
|
{generate function parameters}
|
|
|
|
FunctionParms (tree, fType);
|
|
|
|
|
|
|
|
{generate the function call}
|
|
|
|
expressionType := ftype^.fType;
|
|
|
|
if expressionType^.kind in [structType,unionType] then
|
|
|
|
expressionType := uLongPtr;
|
|
|
|
if (ftype^.toolNum = 0) and (ftype^.dispatcher = 0) then begin
|
|
|
|
if indirect then begin
|
|
|
|
fntype := expressionType;
|
2023-02-12 18:56:02 -06:00
|
|
|
GenerateCode(ftree);
|
|
|
|
if checkNullPointers then
|
|
|
|
Gen0(pc_ckp);
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := fntype;
|
2022-12-08 19:27:37 -06:00
|
|
|
Gen1t(pc_cui, ord(hasVarargs and strictVararg),
|
2017-10-21 18:40:19 -05:00
|
|
|
UsualUnaryConversions);
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
fname := ftree^.id^.name;
|
|
|
|
if ftype^.isPascal then begin
|
|
|
|
fname := pointer(Malloc(length(fname^)+1));
|
|
|
|
CopyString(pointer(fname), pointer(ftree^.id^.name));
|
|
|
|
for i := 1 to length(fname^) do
|
|
|
|
if fName^[i] in ['a'..'z'] then
|
|
|
|
fName^[i] := chr(ord(fName^[i]) & $5F);
|
|
|
|
end; {if}
|
2022-12-08 19:27:37 -06:00
|
|
|
Gen1tName(pc_cup, ord(hasVarargs and strictVararg),
|
2017-10-21 18:40:19 -05:00
|
|
|
UsualUnaryConversions, fname);
|
|
|
|
end; {else}
|
2023-01-08 15:15:32 -06:00
|
|
|
if hasVarargs then
|
|
|
|
hasVarargsCall := true;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
GenTool(pc_tl1, ftype^.toolNum, long(ftype^.ftype^.size).lsw,
|
|
|
|
ftype^.dispatcher);
|
|
|
|
expressionType := ftype^.fType;
|
2023-01-09 21:58:05 -06:00
|
|
|
CheckForIncompleteStructType;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
|
|
|
end; {FunctionCall}
|
|
|
|
|
|
|
|
|
2021-09-10 21:00:23 -05:00
|
|
|
procedure CompareCompatible (var t1,t2: typePtr; equality: boolean);
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{ Make sure that it is legal to compare t1 to t2 }
|
2021-09-10 21:00:23 -05:00
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ t1,t2 - the types to compare }
|
|
|
|
{ equality - is this for an (in)equality comparison? }
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
begin {CompareCompatible}
|
|
|
|
if (t1^.kind = functionType) or (t2^.kind = functionType) then begin
|
|
|
|
if not CompTypes(t1, t2) then
|
2021-09-10 21:00:23 -05:00
|
|
|
Error(47)
|
|
|
|
else if not looseTypeChecks and not equality then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(47);
|
|
|
|
end {if}
|
|
|
|
else if t1^.kind in [pointerType,arrayType] then begin
|
|
|
|
if t2^.kind in [pointerType,arrayType] then begin
|
2021-09-10 21:00:23 -05:00
|
|
|
if CompTypes(t1^.ptype, t2^.ptype) then begin
|
|
|
|
if not looseTypeChecks and not equality then
|
|
|
|
if t1^.ptype^.kind = functionType then
|
|
|
|
Error(47);
|
|
|
|
end {if}
|
|
|
|
else if (t1^.ptype^.kind=scalarType) and (t1^.ptype^.basetype=cgVoid)
|
|
|
|
then begin
|
|
|
|
if not looseTypeChecks then begin
|
|
|
|
if not equality then
|
|
|
|
Error(47)
|
2024-08-02 20:26:40 -05:00
|
|
|
else if not tlastWasNullPtrConst then
|
2021-09-10 21:00:23 -05:00
|
|
|
if t2^.ptype^.kind = functionType then
|
|
|
|
Error(47);
|
|
|
|
end {if}
|
|
|
|
end {else if}
|
|
|
|
else if (t2^.ptype^.kind=scalarType) and (t2^.ptype^.basetype=cgVoid)
|
|
|
|
then begin
|
|
|
|
if not looseTypeChecks then begin
|
|
|
|
if not equality then
|
|
|
|
Error(47)
|
2024-08-02 20:26:40 -05:00
|
|
|
else if not lastWasNullPtrConst then
|
2021-09-10 21:00:23 -05:00
|
|
|
if t1^.ptype^.kind = functionType then
|
|
|
|
Error(47);
|
|
|
|
end {if}
|
|
|
|
end {else if}
|
|
|
|
else
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(47);
|
|
|
|
t2 := ulongPtr;
|
|
|
|
end {if}
|
2024-08-02 20:26:40 -05:00
|
|
|
else if not lastWasNullPtrConst
|
2021-09-10 21:00:23 -05:00
|
|
|
or (not equality and not looseTypeChecks) then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(47);
|
|
|
|
t1 := ulongPtr;
|
|
|
|
end {if}
|
2024-08-02 20:26:40 -05:00
|
|
|
else if t2^.kind in [pointerType,arrayType] then begin
|
|
|
|
if not equality or not tlastWasNullPtrConst then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(47);
|
|
|
|
t2 := ulongPtr;
|
|
|
|
end; {else if}
|
|
|
|
end; {CompareCompatible}
|
|
|
|
|
|
|
|
|
2018-09-05 23:43:58 -05:00
|
|
|
procedure CheckDivByZero (var divisor: tokenType; opType: typePtr);
|
|
|
|
|
|
|
|
{ Check for division by (constant) zero. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ divisor - token for divisor }
|
|
|
|
{ opType - type of the result of the operation }
|
|
|
|
|
|
|
|
begin {CheckDivByZero}
|
|
|
|
if opType^.kind = scalarType then
|
2021-02-04 17:53:37 -06:00
|
|
|
if opType^.baseType in
|
|
|
|
[cgByte,cgWord,cgUByte,cgUWord,cgLong,cgULong,cgQuad,cgUQuad] then
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((divisor.class = intConstant) and (divisor.ival = 0))
|
|
|
|
or ((divisor.class = longConstant) and (divisor.lval = 0))
|
2021-02-03 23:11:23 -06:00
|
|
|
or ((divisor.class = longlongConstant)
|
|
|
|
and (divisor.qval.lo = 0) and (divisor.qval.hi = 0))
|
2021-03-04 23:52:41 -06:00
|
|
|
or ((divisor.class = realConstant) and (divisor.rval = 0.0)) then
|
2018-09-05 23:43:58 -05:00
|
|
|
Error(129);
|
|
|
|
end; {CheckDivByZero}
|
|
|
|
|
|
|
|
|
|
|
|
procedure CheckShiftOverflow (var shiftCountTok: tokenType; opType: typePtr);
|
|
|
|
|
|
|
|
{ Check for invalid (too large or negative) shift count. }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ shiftCountTok - token for shift count }
|
|
|
|
{ opType - type of the result of the operation }
|
|
|
|
|
|
|
|
var
|
|
|
|
shiftCount: longint;
|
|
|
|
|
|
|
|
begin {CheckShiftOverflow}
|
|
|
|
if shiftCountTok.class = intConstant then
|
|
|
|
shiftCount := shiftCountTok.ival
|
|
|
|
else if shiftCountTok.class = longConstant then
|
|
|
|
shiftCount := shiftCountTok.lval
|
2021-02-03 23:11:23 -06:00
|
|
|
else if shiftCountTok.class = longlongConstant then begin
|
|
|
|
if shiftCountTok.qval.hi = 0 then
|
|
|
|
shiftCount := shiftCountTok.qval.lo
|
|
|
|
else
|
|
|
|
shiftCount := -1;
|
|
|
|
end {else if}
|
2018-09-05 23:43:58 -05:00
|
|
|
else
|
|
|
|
shiftCount := 0;
|
|
|
|
|
|
|
|
if (shiftCount <> 0) and (opType^.kind = scalarType) then begin
|
|
|
|
if opType^.baseType in [cgByte,cgWord,cgUByte,cgUWord] then
|
|
|
|
if (shiftCount < 0) or (shiftCount > 15) then
|
|
|
|
Error(130);
|
|
|
|
if opType^.baseType in [cgLong,cgULong] then
|
|
|
|
if (shiftCount < 0) or (shiftCount > 31) then
|
|
|
|
Error(130);
|
2021-02-04 17:53:37 -06:00
|
|
|
if opType^.baseType in [cgQuad,cgUQuad] then
|
|
|
|
if (shiftCount < 0) or (shiftCount > 63) then
|
|
|
|
Error(130);
|
2018-09-05 23:43:58 -05:00
|
|
|
end; {if}
|
|
|
|
end; {CheckShiftOverflow}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
begin {GenerateCode}
|
2023-02-28 22:36:42 -06:00
|
|
|
isConst := false;
|
2022-06-23 22:04:03 -05:00
|
|
|
isNullPtrConst := false;
|
2017-10-21 18:40:19 -05:00
|
|
|
case tree^.token.kind of
|
|
|
|
|
|
|
|
parameterOper:
|
|
|
|
FunctionCall(tree);
|
|
|
|
|
|
|
|
ident: begin
|
2018-03-25 20:56:15 -05:00
|
|
|
tType := tree^.id^.itype;
|
|
|
|
while tType^.kind = definedType do
|
|
|
|
tType := tType^.dType;
|
|
|
|
case tType^.kind of
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
scalarType: begin
|
|
|
|
LoadScalar(tree^.id);
|
|
|
|
expressionType := tree^.id^.itype;
|
|
|
|
end;
|
|
|
|
|
|
|
|
pointerType: begin
|
|
|
|
LoadScalar(tree^.id);
|
|
|
|
expressionType := tree^.id^.itype;
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
|
|
arrayType: begin
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree, false);
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := expressionType^.ptype;
|
|
|
|
end;
|
|
|
|
|
|
|
|
functionType:
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree, false);
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
structType, unionType: begin
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree, false);
|
2017-10-21 18:40:19 -05:00
|
|
|
if expressionType^.kind = pointerType then
|
|
|
|
expressionType := expressionType^.ptype;
|
2023-01-09 21:58:05 -06:00
|
|
|
CheckForIncompleteStructType;
|
2017-10-21 18:40:19 -05:00
|
|
|
end;
|
|
|
|
|
|
|
|
enumConst: begin
|
|
|
|
Gen1t(pc_ldc, tree^.id^.itype^.eval, cgWord);
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end;
|
|
|
|
|
|
|
|
end; {case}
|
|
|
|
end;
|
|
|
|
|
2022-06-08 20:58:52 -05:00
|
|
|
compoundLiteral: begin
|
|
|
|
AutoInit(tree^.id, 0, true);
|
|
|
|
tree^.token.kind := ident;
|
|
|
|
ldoDispose := doDispose;
|
|
|
|
doDispose := false;
|
|
|
|
GenerateCode(tree);
|
|
|
|
doDispose := ldoDispose;
|
|
|
|
tree^.token.kind := compoundliteral;
|
|
|
|
if expressionType^.kind = scalarType then
|
|
|
|
Gen0t(pc_bno, expressionType^.baseType)
|
|
|
|
else
|
|
|
|
Gen0t(pc_bno, cgULong);
|
|
|
|
end;
|
|
|
|
|
2021-10-11 20:54:37 -05:00
|
|
|
intConst,uintConst,ushortConst,charConst,scharConst,ucharConst: begin
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen1t(pc_ldc, tree^.token.ival, cgWord);
|
2023-02-28 22:36:42 -06:00
|
|
|
isConst := true;
|
2017-10-21 18:40:19 -05:00
|
|
|
lastconst := tree^.token.ival;
|
2022-06-23 22:04:03 -05:00
|
|
|
isNullPtrConst := tree^.token.ival = 0;
|
2017-10-21 18:40:19 -05:00
|
|
|
if tree^.token.kind = intConst then
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr
|
2021-03-07 13:38:21 -06:00
|
|
|
else if tree^.token.kind = uintConst then
|
|
|
|
expressionType := uIntPtr
|
2021-10-11 20:54:37 -05:00
|
|
|
else if tree^.token.kind = ushortConst then
|
|
|
|
expressionType := uShortPtr
|
2021-03-07 13:38:21 -06:00
|
|
|
else if tree^.token.kind = charConst then
|
|
|
|
expressionType := charPtr
|
|
|
|
else if tree^.token.kind = scharConst then
|
|
|
|
expressionType := scharPtr
|
|
|
|
else {if tree^.token.kind = ucharConst then}
|
|
|
|
expressionType := ucharPtr;
|
2021-10-11 20:54:37 -05:00
|
|
|
end; {case intConst,uintConst,ushortConst,charConst,scharConst,ucharConst}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
longConst,ulongConst: begin
|
|
|
|
GenLdcLong(tree^.token.lval);
|
|
|
|
if tree^.token.kind = longConst then
|
|
|
|
expressionType := longPtr
|
|
|
|
else
|
|
|
|
expressionType := ulongPtr;
|
2023-02-28 22:36:42 -06:00
|
|
|
isConst := true;
|
2017-10-21 18:40:19 -05:00
|
|
|
lastconst := tree^.token.lval;
|
2022-06-23 22:04:03 -05:00
|
|
|
isNullPtrConst := tree^.token.lval = 0;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case longConst}
|
|
|
|
|
2021-02-03 23:11:23 -06:00
|
|
|
longlongConst,ulonglongConst: begin
|
|
|
|
GenLdcQuad(tree^.token.qval);
|
|
|
|
if tree^.token.kind = longlongConst then
|
|
|
|
expressionType := longlongPtr
|
|
|
|
else
|
|
|
|
expressionType := ulonglongPtr;
|
|
|
|
if (tree^.token.qval.hi = 0) and (tree^.token.qval.lo >= 0) then begin
|
2023-02-28 22:36:42 -06:00
|
|
|
isConst := true;
|
2021-02-03 23:11:23 -06:00
|
|
|
lastconst := tree^.token.qval.lo;
|
|
|
|
end; {if}
|
2022-06-23 22:04:03 -05:00
|
|
|
isNullPtrConst := (tree^.token.qval.hi = 0) and (tree^.token.qval.lo = 0);
|
2021-02-03 23:11:23 -06:00
|
|
|
end; {case longlongConst}
|
|
|
|
|
2021-03-07 00:48:51 -06:00
|
|
|
floatConst: begin
|
|
|
|
GenLdcReal(tree^.token.rval);
|
|
|
|
expressionType := floatPtr;
|
|
|
|
end; {case floatConst}
|
|
|
|
|
|
|
|
doubleConst: begin
|
2017-10-21 18:40:19 -05:00
|
|
|
GenLdcReal(tree^.token.rval);
|
|
|
|
expressionType := doublePtr;
|
2021-03-07 00:48:51 -06:00
|
|
|
end; {case doubleConst}
|
|
|
|
|
|
|
|
extendedConst: begin
|
|
|
|
GenLdcReal(tree^.token.rval);
|
|
|
|
expressionType := extendedPtr;
|
2021-03-04 23:52:41 -06:00
|
|
|
end; {case extendedConst}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
2021-03-07 00:48:51 -06:00
|
|
|
compConst: begin
|
|
|
|
GenLdcReal(tree^.token.rval);
|
|
|
|
expressionType := compPtr;
|
|
|
|
end; {case compConst}
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
stringConst: begin
|
|
|
|
GenS(pc_lca, tree^.token.sval);
|
2021-10-11 20:54:37 -05:00
|
|
|
expressionType := StringType(tree^.token.prefix);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case stringConst}
|
|
|
|
|
|
|
|
eqch: begin {=}
|
|
|
|
L_Value(tree^.left);
|
|
|
|
with tree^.left^ do begin
|
|
|
|
if token.kind = ident then
|
|
|
|
kind := id^.itype^.kind
|
|
|
|
else
|
|
|
|
kind := definedType;
|
|
|
|
if kind = arrayType then
|
|
|
|
if id^.storage = parameter then
|
|
|
|
kind := pointerType;
|
|
|
|
if (token.kind = ident)
|
|
|
|
and (kind in [scalarType,pointerType]) then begin
|
|
|
|
GenerateCode(tree^.right);
|
|
|
|
with tree^.left^.id^ do begin
|
|
|
|
if itype^.kind in [pointerType,arrayType] then
|
|
|
|
lType := uLongPtr
|
|
|
|
else
|
|
|
|
lType := itype;
|
|
|
|
AssignmentConversion(itype, expressionType, lastWasConst,
|
|
|
|
lastConst, true, true);
|
|
|
|
case storage of
|
|
|
|
stackFrame, parameter:
|
|
|
|
Gen2t(pc_cop, lln, 0, lType^.baseType);
|
|
|
|
external, global, private:
|
|
|
|
Gen1tName(pc_cpo, 0, lType^.baseType, name);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
|
|
|
end; {with}
|
|
|
|
end {if}
|
|
|
|
else begin
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree^.left, checkNullPointers);
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
|
|
|
lisBitField := isBitField;
|
|
|
|
lbitDisp := bitDisp;
|
|
|
|
lbitSize := bitSize;
|
|
|
|
if lType^.kind = arrayType then
|
|
|
|
lType := lType^.aType
|
|
|
|
else if lType^.kind = pointerType then
|
|
|
|
lType := lType^.pType;
|
|
|
|
GenerateCode(tree^.right);
|
|
|
|
AssignmentConversion(lType, expressionType, lastWasConst,
|
|
|
|
lastConst, true, true);
|
Record volatile and restrict qualifiers in types.
These are needed to correctly distinguish pointer types in _Generic. They should also be used for type compatibility checks in other contexts, but currently are not.
This also fixes a couple small problems related to type qualifiers:
*restrict was not allowed to appear after * in type-names
*volatile status was not properly recorded in sym files
Here is an example of using _Generic to distinguish pointer types based on the qualifiers of the pointed-to type:
#include <stdio.h>
#define f(e) _Generic((e),\
int * restrict *: 1,\
int * volatile const *: 2,\
int **: 3,\
default: 0)
#define g(e) _Generic((e),\
int *: 1,\
const int *: 2,\
volatile int *: 3,\
default: 0)
int main(void) {
int * restrict * p1;
int * volatile const * p2;
int * const * p3;
// should print "1 2 0 1"
printf("%i %i %i %i\n", f(p1), f(p2), f(p3), f((int * restrict *)0));
int *q1;
const int *q2;
volatile int *q3;
const volatile int *q4;
// should print "1 2 3 0"
printf("%i %i %i %i\n", g(q1), g(q2), g(q3), g(q4));
}
Here is an example of a problem resulting from volatile not being recorded in sym files (if a sym file was present, the read of x was lifted out of the loop):
#pragma optimize -1
static volatile int x;
#include <stdio.h>
int main(void) {
int y;
for (unsigned i = 0; i < 100; i++) {
y = x*2 + 7;
}
}
2021-08-29 21:10:20 -05:00
|
|
|
while lType^.kind = definedType do
|
|
|
|
lType := lType^.dType;
|
2017-10-21 18:40:19 -05:00
|
|
|
case lType^.kind of
|
|
|
|
scalarType:
|
|
|
|
if lisBitField then
|
|
|
|
Gen2t(pc_cbf, lbitDisp, lbitSize, lType^.baseType)
|
|
|
|
else
|
|
|
|
Gen0t(pc_cpi, lType^.baseType);
|
|
|
|
|
|
|
|
pointerType:
|
|
|
|
Gen0t(pc_cpi, cgULong);
|
|
|
|
|
|
|
|
structType,unionType:
|
|
|
|
Gen2(pc_mov, long(lType^.size).msw, long(lType^.size).lsw);
|
|
|
|
|
|
|
|
otherwise:
|
|
|
|
Error(47);
|
|
|
|
|
|
|
|
end; {case}
|
|
|
|
end; {else}
|
|
|
|
end; {with}
|
|
|
|
end; {=}
|
|
|
|
|
|
|
|
pluseqop, {+=}
|
|
|
|
minuseqop, {-=}
|
|
|
|
asteriskeqop, {*=}
|
|
|
|
slasheqop, {/=}
|
|
|
|
percenteqop, {%=}
|
|
|
|
ltlteqop, {<<=}
|
|
|
|
gtgteqop, {>>=}
|
|
|
|
andeqop, {&=}
|
|
|
|
caroteqop, {^=}
|
|
|
|
bareqop: with tree^.left^ do {|=}
|
|
|
|
begin
|
|
|
|
L_Value(tree^.left);
|
|
|
|
if (token.kind = ident)
|
2015-12-31 23:38:21 -06:00
|
|
|
and ((id^.itype^.kind in [scalarType,pointerType])
|
|
|
|
or ((id^.itype^.kind = arrayType) and (id^.storage = parameter))) then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
doingScalar := true;
|
|
|
|
LoadScalar(id);
|
|
|
|
lType := id^.itype;
|
2017-07-12 23:39:02 -05:00
|
|
|
t1 := 0;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
doingScalar := false;
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree^.left, checkNullPointers);
|
2017-10-21 18:40:19 -05:00
|
|
|
lisBitField := isBitField;
|
|
|
|
lbitDisp := bitDisp;
|
|
|
|
lbitSize := bitSize;
|
|
|
|
t1 := GetTemp(cgLongSize);
|
|
|
|
Gen2t(pc_str, t1, 0, cgULong);
|
|
|
|
Gen2t(pc_lod, t1, 0, cgULong);
|
|
|
|
Gen2t(pc_lod, t1, 0, cgULong);
|
|
|
|
lType := expressionType^.pType;
|
2022-05-23 21:10:29 -05:00
|
|
|
isVolatile := tqVolatile in lType^.qualifiers;
|
2017-10-21 18:40:19 -05:00
|
|
|
if isBitField then begin
|
|
|
|
if unsigned then
|
|
|
|
Gen2t(pc_lbu, bitDisp, bitSize, lType^.baseType)
|
|
|
|
else
|
|
|
|
Gen2t(pc_lbf, bitDisp, bitSize, lType^.baseType);
|
|
|
|
end {if}
|
|
|
|
else if lType^.kind = pointerType then
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), 0, cgULong)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), 0, lType^.baseType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
Record volatile and restrict qualifiers in types.
These are needed to correctly distinguish pointer types in _Generic. They should also be used for type compatibility checks in other contexts, but currently are not.
This also fixes a couple small problems related to type qualifiers:
*restrict was not allowed to appear after * in type-names
*volatile status was not properly recorded in sym files
Here is an example of using _Generic to distinguish pointer types based on the qualifiers of the pointed-to type:
#include <stdio.h>
#define f(e) _Generic((e),\
int * restrict *: 1,\
int * volatile const *: 2,\
int **: 3,\
default: 0)
#define g(e) _Generic((e),\
int *: 1,\
const int *: 2,\
volatile int *: 3,\
default: 0)
int main(void) {
int * restrict * p1;
int * volatile const * p2;
int * const * p3;
// should print "1 2 0 1"
printf("%i %i %i %i\n", f(p1), f(p2), f(p3), f((int * restrict *)0));
int *q1;
const int *q2;
volatile int *q3;
const volatile int *q4;
// should print "1 2 3 0"
printf("%i %i %i %i\n", g(q1), g(q2), g(q3), g(q4));
}
Here is an example of a problem resulting from volatile not being recorded in sym files (if a sym file was present, the read of x was lifted out of the loop):
#pragma optimize -1
static volatile int x;
#include <stdio.h>
int main(void) {
int y;
for (unsigned i = 0; i < 100; i++) {
y = x*2 + 7;
}
}
2021-08-29 21:10:20 -05:00
|
|
|
if tqConst in lType^.qualifiers then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(93);
|
2015-12-31 23:38:21 -06:00
|
|
|
if doingScalar
|
|
|
|
and (ltype^.kind = arrayType) and (id^.storage = parameter) then
|
|
|
|
kind := pointerType
|
|
|
|
else
|
|
|
|
kind := lType^.kind;
|
2017-10-21 18:40:19 -05:00
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if expressionType^.kind <> scalarType then
|
2021-01-29 12:49:28 -06:00
|
|
|
Error(66);
|
2016-11-21 00:25:58 -06:00
|
|
|
if tree^.token.kind in [gtgteqop,ltlteqop] then
|
|
|
|
if kind = scalarType then
|
|
|
|
if expressionType^.kind = scalarType then begin
|
2021-01-29 12:49:28 -06:00
|
|
|
if expressionType^.baseType in
|
|
|
|
[cgReal,cgDouble,cgComp,cgExtended,cgVoid] then
|
|
|
|
Error(66);
|
2016-11-21 00:25:58 -06:00
|
|
|
et := UsualUnaryConversions;
|
2021-02-12 15:06:15 -06:00
|
|
|
if ltype^.baseType in [cgQuad,cgUQuad] then begin
|
|
|
|
if not (et in [cgWord,cgUWord]) then begin
|
|
|
|
Gen2(pc_cnv, et, ord(cgWord));
|
|
|
|
end; {if}
|
2016-11-21 00:25:58 -06:00
|
|
|
expressionType := lType;
|
2021-02-12 15:06:15 -06:00
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
if et <> Unary(ltype^.baseType) then begin
|
|
|
|
Gen2(pc_cnv, et, ord(Unary(ltype^.baseType)));
|
|
|
|
expressionType := lType;
|
|
|
|
end; {if}
|
2016-11-21 00:25:58 -06:00
|
|
|
end; {if}
|
2015-12-31 23:38:21 -06:00
|
|
|
if kind <> pointerType then
|
2016-11-21 00:25:58 -06:00
|
|
|
et := UsualBinaryConversions(lType)
|
|
|
|
else
|
|
|
|
et := ccPointer;
|
2017-10-21 18:40:19 -05:00
|
|
|
case tree^.token.kind of
|
|
|
|
|
|
|
|
pluseqop:
|
2015-12-31 23:38:21 -06:00
|
|
|
if kind = pointerType then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
ChangePointer(pc_adl, lType^.pType^.size, UsualUnaryConversions);
|
|
|
|
expressionType := lType;
|
|
|
|
end
|
|
|
|
else if et in [cgWord,cgUWord] then
|
|
|
|
Gen0(pc_adi)
|
|
|
|
else if et in [cgLong,cgULong] then
|
|
|
|
Gen0(pc_adl)
|
2021-01-30 23:31:18 -06:00
|
|
|
else if et in [cgQuad,cgUQuad] then
|
|
|
|
Gen0(pc_adq)
|
2021-03-06 23:54:55 -06:00
|
|
|
else if et in [cgReal,cgDouble,cgComp,cgExtended] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_adr)
|
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
minuseqop:
|
2015-12-31 23:38:21 -06:00
|
|
|
if kind = pointerType then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
ChangePointer(pc_sbl, lType^.pType^.size, UsualUnaryConversions);
|
|
|
|
expressionType := lType;
|
|
|
|
end
|
|
|
|
else if et in [cgWord,cgUWord] then
|
|
|
|
Gen0(pc_sbi)
|
|
|
|
else if et in [cgLong,cgULong] then
|
|
|
|
Gen0(pc_sbl)
|
2021-01-30 23:31:18 -06:00
|
|
|
else if et in [cgQuad,cgUQuad] then
|
|
|
|
Gen0(pc_sbq)
|
2021-03-06 23:54:55 -06:00
|
|
|
else if et in [cgReal,cgDouble,cgComp,cgExtended] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_sbr)
|
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
asteriskeqop:
|
|
|
|
if et = cgWord then
|
|
|
|
Gen0(pc_mpi)
|
|
|
|
else if et = cgUWord then
|
|
|
|
Gen0(pc_umi)
|
|
|
|
else if et = cgLong then
|
|
|
|
Gen0(pc_mpl)
|
|
|
|
else if et = cgULong then
|
|
|
|
Gen0(pc_uml)
|
2021-02-04 22:23:59 -06:00
|
|
|
else if et = cgQuad then
|
|
|
|
Gen0(pc_mpq)
|
|
|
|
else if et = cgUQuad then
|
|
|
|
Gen0(pc_umq)
|
2021-03-06 23:54:55 -06:00
|
|
|
else if et in [cgReal,cgDouble,cgComp,cgExtended] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_mpr)
|
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
slasheqop:
|
|
|
|
if et = cgWord then
|
|
|
|
Gen0(pc_dvi)
|
|
|
|
else if et = cgUWord then
|
|
|
|
Gen0(pc_udi)
|
|
|
|
else if et = cgLong then
|
|
|
|
Gen0(pc_dvl)
|
|
|
|
else if et = cgULong then
|
|
|
|
Gen0(pc_udl)
|
2021-02-05 12:42:48 -06:00
|
|
|
else if et = cgQuad then
|
|
|
|
Gen0(pc_dvq)
|
|
|
|
else if et = cgUQuad then
|
|
|
|
Gen0(pc_udq)
|
2021-03-06 23:54:55 -06:00
|
|
|
else if et in [cgReal,cgDouble,cgComp,cgExtended] then
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_dvr)
|
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
percenteqop:
|
|
|
|
if et = cgWord then
|
|
|
|
Gen0(pc_mod)
|
|
|
|
else if et = cgUWord then
|
|
|
|
Gen0(pc_uim)
|
|
|
|
else if et = cgLong then
|
|
|
|
Gen0(pc_mdl)
|
|
|
|
else if et = cgULong then
|
|
|
|
Gen0(pc_ulm)
|
2021-02-05 12:42:48 -06:00
|
|
|
else if et = cgQuad then
|
|
|
|
Gen0(pc_mdq)
|
|
|
|
else if et = cgUQuad then
|
|
|
|
Gen0(pc_uqm)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
ltlteqop:
|
|
|
|
if et in [cgWord,cgUWord] then
|
|
|
|
Gen0(pc_shl)
|
|
|
|
else if et in [cgLong,cgULong] then
|
|
|
|
Gen0(pc_sll)
|
2021-02-12 15:06:15 -06:00
|
|
|
else if et in [cgQuad,cgUQuad] then
|
|
|
|
Gen0(pc_slq)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
gtgteqop:
|
|
|
|
if et = cgWord then
|
|
|
|
Gen0(pc_shr)
|
|
|
|
else if et = cgUWord then
|
|
|
|
Gen0(pc_usr)
|
|
|
|
else if et = cgLong then
|
|
|
|
Gen0(pc_slr)
|
|
|
|
else if et = cgULong then
|
|
|
|
Gen0(pc_vsr)
|
2021-02-12 15:06:15 -06:00
|
|
|
else if et = cgQuad then
|
|
|
|
Gen0(pc_sqr)
|
|
|
|
else if et = cgUQuad then
|
|
|
|
Gen0(pc_wsr)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
andeqop:
|
|
|
|
if et in [cgWord,cgUWord] then
|
|
|
|
Gen0(pc_bnd)
|
|
|
|
else if et in [cgLong,cgULong] then
|
|
|
|
Gen0(pc_bal)
|
2021-01-30 00:25:15 -06:00
|
|
|
else if et in [cgQuad,cgUQuad] then
|
|
|
|
Gen0(pc_baq)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
caroteqop:
|
|
|
|
if et in [cgWord,cgUWord] then
|
|
|
|
Gen0(pc_bxr)
|
|
|
|
else if et in [cgLong,cgULong] then
|
|
|
|
Gen0(pc_blx)
|
2021-01-30 00:25:15 -06:00
|
|
|
else if et in [cgQuad,cgUQuad] then
|
|
|
|
Gen0(pc_bqx)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
bareqop:
|
|
|
|
if et in [cgWord,cgUWord] then
|
|
|
|
Gen0(pc_bor)
|
|
|
|
else if et in [cgLong,cgULong] then
|
|
|
|
Gen0(pc_blr)
|
2021-01-30 00:25:15 -06:00
|
|
|
else if et in [cgQuad,cgUQuad] then
|
|
|
|
Gen0(pc_bqr)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
|
|
|
Error(66);
|
|
|
|
|
|
|
|
otherwise: Error(57);
|
|
|
|
end; {case}
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((lint & lintOverflow) <> 0) then begin
|
|
|
|
if tree^.token.kind in [slasheqop,percenteqop] then
|
|
|
|
CheckDivByZero(tree^.right^.token, lType)
|
|
|
|
else if tree^.token.kind in [ltlteqop,gtgteqop] then
|
|
|
|
CheckShiftOverflow(tree^.right^.token, lType);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
AssignmentConversion(lType,expressionType,false,0,true,true);
|
|
|
|
if doingScalar then begin
|
2015-12-31 23:38:21 -06:00
|
|
|
if kind = pointerType then
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := uLongPtr;
|
|
|
|
case id^.storage of
|
|
|
|
stackFrame, parameter:
|
|
|
|
Gen2t(pc_cop, id^.lln, 0, lType^.baseType);
|
|
|
|
external, global, private:
|
|
|
|
Gen1tName(pc_cpo, 0, lType^.baseType, id^.name);
|
|
|
|
otherwise: ;
|
|
|
|
end; {case}
|
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
if lisBitField then
|
|
|
|
Gen2t(pc_cbf, lbitDisp, lbitSize, lType^.baseType)
|
|
|
|
else begin
|
|
|
|
if ltype^.kind in [pointerType,arrayType] then
|
|
|
|
lType := uLongPtr;
|
|
|
|
Gen0t(pc_cpi, lType^.baseType);
|
|
|
|
end; {else}
|
|
|
|
Gen0t(pc_bno, lType^.baseType);
|
|
|
|
end; {else}
|
2017-07-12 23:39:02 -05:00
|
|
|
if t1 <> 0 then
|
|
|
|
FreeTemp(t1, cgLongSize);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {with}
|
|
|
|
|
|
|
|
commach: begin {,}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
if expressionType^.baseType <> cgVoid then
|
|
|
|
Gen0t(pc_pop, UsualUnaryConversions);
|
|
|
|
GenerateCode(tree^.right);
|
|
|
|
Gen0t(pc_bno, UsualUnaryConversions);
|
|
|
|
{result type is already in expressionType}
|
|
|
|
end; {case commach}
|
|
|
|
|
|
|
|
barbarop: begin {||}
|
|
|
|
GenerateCode(tree^.left);
|
2015-12-30 20:32:40 -06:00
|
|
|
if expressionType^.kind in [pointerType,arrayType] then
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := uLongPtr
|
2021-02-14 17:37:55 -06:00
|
|
|
else begin
|
|
|
|
et := UsualUnaryConversions;
|
2021-03-06 23:54:55 -06:00
|
|
|
if et in [cgReal,cgDouble,cgComp,cgExtended] then begin
|
2021-02-14 17:37:55 -06:00
|
|
|
GenLdcReal(0.0);
|
|
|
|
Gen0t(pc_neq, cgExtended);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end {if}
|
|
|
|
else if et in [cgQuad,cgUQuad] then begin
|
|
|
|
GenLdcQuad(longlong0);
|
|
|
|
Gen0t(pc_neq, et);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end; {else if}
|
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2015-12-30 20:32:40 -06:00
|
|
|
if expressionType^.kind in [pointerType,arrayType] then
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := uLongPtr
|
2021-02-14 17:37:55 -06:00
|
|
|
else begin
|
|
|
|
et := UsualUnaryConversions;
|
2021-03-06 23:54:55 -06:00
|
|
|
if et in [cgReal,cgDouble,cgComp,cgExtended] then begin
|
2021-02-14 17:37:55 -06:00
|
|
|
GenLdcReal(0.0);
|
|
|
|
Gen0t(pc_neq, cgExtended);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end {if}
|
|
|
|
else if et in [cgQuad,cgUQuad] then begin
|
|
|
|
GenLdcQuad(longlong0);
|
|
|
|
Gen0t(pc_neq, et);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end; {else if}
|
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
case UsualBinaryConversions(lType) of
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_ior);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_lor);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case barbarop}
|
|
|
|
|
|
|
|
andandop: begin {&&}
|
|
|
|
GenerateCode(tree^.left);
|
2015-12-30 20:32:40 -06:00
|
|
|
if expressionType^.kind in [pointerType,arrayType] then
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := uLongPtr
|
2021-02-14 17:37:55 -06:00
|
|
|
else begin
|
|
|
|
et := UsualUnaryConversions;
|
2021-03-06 23:54:55 -06:00
|
|
|
if et in [cgReal,cgDouble,cgComp,cgExtended] then begin
|
2021-02-14 17:37:55 -06:00
|
|
|
GenLdcReal(0.0);
|
|
|
|
Gen0t(pc_neq, cgExtended);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end {if}
|
|
|
|
else if et in [cgQuad,cgUQuad] then begin
|
|
|
|
GenLdcQuad(longlong0);
|
|
|
|
Gen0t(pc_neq, et);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end; {else if}
|
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2015-12-30 20:32:40 -06:00
|
|
|
if expressionType^.kind in [pointerType,arrayType] then
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionType := uLongPtr
|
2021-02-14 17:37:55 -06:00
|
|
|
else begin
|
|
|
|
et := UsualUnaryConversions;
|
2021-03-06 23:54:55 -06:00
|
|
|
if et in [cgReal,cgDouble,cgComp,cgExtended] then begin
|
2021-02-14 17:37:55 -06:00
|
|
|
GenLdcReal(0.0);
|
|
|
|
Gen0t(pc_neq, cgExtended);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end {if}
|
|
|
|
else if et in [cgQuad,cgUQuad] then begin
|
|
|
|
GenLdcQuad(longlong0);
|
|
|
|
Gen0t(pc_neq, et);
|
|
|
|
expressionType := intPtr;
|
|
|
|
end; {else if}
|
|
|
|
end; {else}
|
2017-10-21 18:40:19 -05:00
|
|
|
case UsualBinaryConversions(lType) of
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_and);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_lnd);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case andandop}
|
|
|
|
|
|
|
|
carotch: begin {^}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (lType^.kind <> scalarType) or (expressionType^.kind <> scalarType) then
|
|
|
|
Error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_bxr);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_blx);
|
2021-01-30 00:25:15 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_bqx);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case carotch}
|
|
|
|
|
|
|
|
barch: begin {|}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (lType^.kind <> scalarType) or (expressionType^.kind <> scalarType) then
|
|
|
|
Error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_bor);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_blr);
|
2021-01-30 00:25:15 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_bqr);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case barch}
|
|
|
|
|
|
|
|
andch: begin {&}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (lType^.kind <> scalarType) or (expressionType^.kind <> scalarType) then
|
|
|
|
Error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_bnd);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_bal);
|
2021-01-30 00:25:15 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_baq);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case andch}
|
|
|
|
|
|
|
|
ltltop: begin {<<}
|
|
|
|
GenerateCode(tree^.left);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (expressionType^.kind <> scalarType) then
|
|
|
|
error(66);
|
2016-11-07 18:57:40 -06:00
|
|
|
et := UsualUnaryConversions;
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2016-11-07 18:57:40 -06:00
|
|
|
if (expressionType^.kind <> scalarType)
|
|
|
|
or not (expressionType^.baseType in
|
2021-02-12 15:06:15 -06:00
|
|
|
[cgByte,cgUByte,cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad]) then
|
2016-11-07 18:57:40 -06:00
|
|
|
error(66);
|
2021-02-12 15:06:15 -06:00
|
|
|
if et in [cgQuad,cgUQuad] then begin
|
|
|
|
if not (expressionType^.baseType in [cgWord,cgUWord]) then
|
|
|
|
Gen2(pc_cnv, ord(expressionType^.baseType), ord(cgWord));
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
if expressionType^.baseType <> et then
|
|
|
|
Gen2(pc_cnv, ord(expressionType^.baseType), ord(et));
|
2016-11-07 18:57:40 -06:00
|
|
|
case et of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_shl);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_sll);
|
2021-02-12 15:06:15 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_slq);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2016-11-07 18:57:40 -06:00
|
|
|
expressionType := lType;
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((lint & lintOverflow) <> 0) then
|
|
|
|
CheckShiftOverflow(tree^.right^.token, expressionType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case ltltop}
|
|
|
|
|
|
|
|
gtgtop: begin {>>}
|
|
|
|
GenerateCode(tree^.left);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (expressionType^.kind <> scalarType) then
|
|
|
|
error(66);
|
2016-11-07 18:57:40 -06:00
|
|
|
et := UsualUnaryConversions;
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2016-11-07 18:57:40 -06:00
|
|
|
if (expressionType^.kind <> scalarType)
|
|
|
|
or not (expressionType^.baseType in
|
2021-02-12 15:06:15 -06:00
|
|
|
[cgByte,cgUByte,cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad]) then
|
2016-11-07 18:57:40 -06:00
|
|
|
error(66);
|
2021-02-12 15:06:15 -06:00
|
|
|
if et in [cgQuad,cgUQuad] then begin
|
|
|
|
if not (expressionType^.baseType in [cgWord,cgUWord]) then
|
|
|
|
Gen2(pc_cnv, ord(expressionType^.baseType), ord(cgWord));
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
if expressionType^.baseType <> et then
|
|
|
|
Gen2(pc_cnv, ord(expressionType^.baseType), ord(et));
|
2016-11-07 18:57:40 -06:00
|
|
|
case et of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgWord:
|
|
|
|
Gen0(pc_shr);
|
|
|
|
cgUByte,cgUWord:
|
|
|
|
Gen0(pc_usr);
|
|
|
|
cgLong:
|
|
|
|
Gen0(pc_slr);
|
|
|
|
cgULong:
|
|
|
|
Gen0(pc_vsr);
|
2021-02-12 15:06:15 -06:00
|
|
|
cgQuad:
|
|
|
|
Gen0(pc_sqr);
|
|
|
|
cgUQuad:
|
|
|
|
Gen0(pc_wsr);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2016-11-07 18:57:40 -06:00
|
|
|
expressionType := lType;
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((lint & lintOverflow) <> 0) then
|
|
|
|
CheckShiftOverflow(tree^.right^.token, expressionType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case gtgtop}
|
|
|
|
|
|
|
|
plusch: begin {+}
|
|
|
|
if ExpressionKind(tree^.right) in [arrayType,pointerType] then begin
|
|
|
|
tree^.middle := tree^.right;
|
|
|
|
tree^.right := tree^.left;
|
|
|
|
tree^.left := tree^.middle;
|
|
|
|
end; {if}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
|
|
|
if lType^.kind in [arrayType,pointerType] then begin
|
2018-09-06 21:11:49 -05:00
|
|
|
if expressionType^.kind <> scalarType then
|
|
|
|
error(66);
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
{pointer addition}
|
|
|
|
et := UsualUnaryConversions;
|
|
|
|
expressionType := lType;
|
|
|
|
if lType^.kind = arrayType then
|
|
|
|
lType := lType^.aType
|
|
|
|
else
|
|
|
|
lType := lType^.pType;
|
|
|
|
ChangePointer(pc_adl, lType^.size, et);
|
2022-06-19 17:55:08 -05:00
|
|
|
if expressionType^.kind = arrayType then
|
|
|
|
expressionType := MakePointerTo(expressionType^.aType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
|
|
|
|
{scalar addition}
|
|
|
|
case UsualBinaryConversions(lType) of
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_adi);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_adl);
|
2021-01-30 23:31:18 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_adq);
|
2021-03-06 23:54:55 -06:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_adr);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {else}
|
|
|
|
end; {case plusch}
|
|
|
|
|
|
|
|
minusch: begin {-}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
|
|
|
if lType^.kind in [pointerType,arrayType] then begin
|
|
|
|
if lType^.kind = arrayType then
|
|
|
|
size := lType^.aType^.size
|
|
|
|
else
|
|
|
|
size := lType^.pType^.size;
|
|
|
|
if expressionType^.kind in [arrayType,pointerType] then begin
|
|
|
|
|
|
|
|
{subtraction of two pointers}
|
2017-10-29 20:21:36 -05:00
|
|
|
if size = 0 then
|
|
|
|
Error(122)
|
|
|
|
{NOTE: assumes aType & pType overlap in typeRecord}
|
|
|
|
else if not CompTypes(lType^.aType, expressionType^.aType) then
|
|
|
|
Error(47);
|
2023-02-12 18:56:02 -06:00
|
|
|
if checkNullPointers then begin
|
|
|
|
Gen0(pc_ckn);
|
|
|
|
Gen0(pc_ckp);
|
|
|
|
end; {if}
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_sbl);
|
|
|
|
if size <> 1 then begin
|
|
|
|
GenLdcLong(size);
|
|
|
|
Gen0(pc_dvl);
|
|
|
|
end; {if}
|
|
|
|
lType := longPtr;
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
{subtract a scalar from a pointer}
|
|
|
|
ChangePointer(pc_sbl, size, UsualUnaryConversions);
|
|
|
|
expressionType := lType;
|
2022-06-19 17:55:08 -05:00
|
|
|
if expressionType^.kind = arrayType then
|
|
|
|
expressionType := MakePointerTo(expressionType^.aType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
|
|
|
|
{scalar subtraction}
|
2018-09-06 21:11:49 -05:00
|
|
|
if expressionType^.kind <> scalarType then
|
|
|
|
error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_sbi);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_sbl);
|
2021-01-30 23:31:18 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_sbq);
|
2021-03-06 23:54:55 -06:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_sbr);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {else}
|
|
|
|
end; {case minusch}
|
|
|
|
|
|
|
|
asteriskch: begin {*}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (lType^.kind <> scalarType) or (expressionType^.kind <> scalarType) then
|
|
|
|
Error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgWord:
|
|
|
|
Gen0(pc_mpi);
|
|
|
|
cgUByte,cgUWord:
|
|
|
|
Gen0(pc_umi);
|
|
|
|
cgLong:
|
|
|
|
Gen0(pc_mpl);
|
|
|
|
cgULong:
|
|
|
|
Gen0(pc_uml);
|
2021-02-04 22:23:59 -06:00
|
|
|
cgQuad:
|
|
|
|
Gen0(pc_mpq);
|
|
|
|
cgUQuad:
|
|
|
|
Gen0(pc_umq);
|
2021-03-06 23:54:55 -06:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_mpr);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case asteriskch}
|
|
|
|
|
|
|
|
slashch: begin {/}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (lType^.kind <> scalarType) or (expressionType^.kind <> scalarType) then
|
|
|
|
Error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgWord:
|
|
|
|
Gen0(pc_dvi);
|
|
|
|
cgUByte,cgUWord:
|
|
|
|
Gen0(pc_udi);
|
|
|
|
cgLong:
|
|
|
|
Gen0(pc_dvl);
|
|
|
|
cgULong:
|
|
|
|
Gen0(pc_udl);
|
2021-02-05 12:42:48 -06:00
|
|
|
cgQuad:
|
|
|
|
Gen0(pc_dvq);
|
|
|
|
cgUQuad:
|
|
|
|
Gen0(pc_udq);
|
2021-03-06 23:54:55 -06:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_dvr);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((lint & lintOverflow) <> 0) then
|
|
|
|
CheckDivByZero(tree^.right^.token, expressionType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case slashch}
|
|
|
|
|
|
|
|
percentch: begin {%}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2018-09-06 21:11:49 -05:00
|
|
|
if (lType^.kind <> scalarType) or (expressionType^.kind <> scalarType) then
|
|
|
|
Error(66)
|
|
|
|
else case UsualBinaryConversions(lType) of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgWord:
|
|
|
|
Gen0(pc_mod);
|
|
|
|
cgUByte,cgUWord:
|
|
|
|
Gen0(pc_uim);
|
|
|
|
cgLong:
|
|
|
|
Gen0(pc_mdl);
|
|
|
|
cgULong:
|
|
|
|
Gen0(pc_ulm);
|
2021-02-05 12:42:48 -06:00
|
|
|
cgQuad:
|
|
|
|
Gen0(pc_mdq);
|
|
|
|
cgUQuad:
|
|
|
|
Gen0(pc_uqm);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2018-09-05 23:43:58 -05:00
|
|
|
if ((lint & lintOverflow) <> 0) then
|
|
|
|
CheckDivByZero(tree^.right^.token, expressionType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case percentch}
|
|
|
|
|
|
|
|
eqeqop, {==}
|
|
|
|
exceqop: begin {!=}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
2024-08-02 20:26:40 -05:00
|
|
|
tLastWasNullPtrConst := lastWasNullPtrConst;
|
2017-10-21 18:40:19 -05:00
|
|
|
GenerateCode(tree^.right);
|
2021-09-10 21:00:23 -05:00
|
|
|
CompareCompatible(ltype, expressionType, true);
|
2017-10-21 18:40:19 -05:00
|
|
|
if tree^.token.kind = eqeqop then
|
|
|
|
Gen0t(pc_equ, UsualBinaryConversions(lType))
|
|
|
|
else
|
|
|
|
Gen0t(pc_neq, UsualBinaryConversions(lType));
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case exceqop,eqeqop}
|
|
|
|
|
|
|
|
lteqop, {<=}
|
|
|
|
gteqop, {>=}
|
|
|
|
ltch, {<}
|
|
|
|
gtch: begin {>}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
GenerateCode(tree^.right);
|
2021-09-10 21:00:23 -05:00
|
|
|
CompareCompatible(ltype, expressionType, false);
|
2017-10-21 18:40:19 -05:00
|
|
|
if tree^.token.kind = lteqop then
|
|
|
|
Gen0t(pc_leq, UsualBinaryConversions(lType))
|
|
|
|
else if tree^.token.kind = gteqop then
|
|
|
|
Gen0t(pc_geq, UsualBinaryConversions(lType))
|
|
|
|
else if tree^.token.kind = ltch then
|
|
|
|
Gen0t(pc_les, UsualBinaryConversions(lType))
|
|
|
|
else {if tree^.token.kind = gtch then}
|
|
|
|
Gen0t(pc_grt, UsualBinaryConversions(lType));
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case lteqop,gteqop,ltch,gtch}
|
|
|
|
|
|
|
|
uminus: begin {unary -}
|
|
|
|
GenerateCode(tree^.left);
|
2018-09-06 21:11:49 -05:00
|
|
|
if expressionType^.kind <> scalarType then
|
|
|
|
error(66)
|
|
|
|
else case UsualUnaryConversions of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_ngi);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_ngl);
|
2021-01-30 13:49:06 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_ngq);
|
2021-03-06 23:54:55 -06:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
2017-10-21 18:40:19 -05:00
|
|
|
Gen0(pc_ngr);
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case uminus}
|
|
|
|
|
2022-12-09 19:03:38 -06:00
|
|
|
uplus: begin {unary +}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
if expressionType^.kind <> scalarType then
|
|
|
|
error(66)
|
|
|
|
else case UsualUnaryConversions of
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord,cgLong,cgULong,cgQuad,cgUQuad,
|
|
|
|
cgReal,cgDouble,cgComp,cgExtended:
|
|
|
|
;
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case uplus}
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
tildech: begin {~}
|
|
|
|
GenerateCode(tree^.left);
|
2018-09-06 21:11:49 -05:00
|
|
|
if expressionType^.kind <> scalarType then
|
|
|
|
error(66)
|
|
|
|
else case UsualUnaryConversions of
|
2017-10-21 18:40:19 -05:00
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_bnt);
|
|
|
|
cgLong,cgULong:
|
|
|
|
Gen0(pc_bnl);
|
2021-01-30 13:49:06 -06:00
|
|
|
cgQuad,cgUQuad:
|
|
|
|
Gen0(pc_bnq);
|
2017-10-21 18:40:19 -05:00
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
|
|
|
end; {case tildech}
|
|
|
|
|
|
|
|
excch: begin {!}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
if expressionType^.kind = pointerType then
|
|
|
|
expressionType := uLongPtr;
|
|
|
|
case UsualUnaryConversions of
|
|
|
|
|
|
|
|
cgByte,cgUByte,cgWord,cgUWord:
|
|
|
|
Gen0(pc_not);
|
|
|
|
|
|
|
|
cgLong,cgULong: begin
|
|
|
|
GenLdcLong(0);
|
|
|
|
Gen0t(pc_equ, cgLong);
|
|
|
|
end;
|
|
|
|
|
2021-02-04 17:53:10 -06:00
|
|
|
cgQuad,cgUQuad: begin
|
|
|
|
GenLdcQuad(longlong0);
|
|
|
|
Gen0t(pc_equ, cgQuad);
|
|
|
|
end;
|
|
|
|
|
2021-03-06 23:54:55 -06:00
|
|
|
cgReal,cgDouble,cgComp,cgExtended: begin
|
2017-10-21 18:40:19 -05:00
|
|
|
GenLdcReal(0.0);
|
|
|
|
Gen0t(pc_equ, cgExtended);
|
|
|
|
end;
|
|
|
|
|
|
|
|
otherwise:
|
|
|
|
error(66);
|
|
|
|
end; {case}
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case excch}
|
|
|
|
|
|
|
|
plusplusop: {prefix ++}
|
|
|
|
DoIncDec(tree^.left, pc_lil, pc_gil, pc_iil);
|
|
|
|
|
|
|
|
opplusplus: {postfix ++}
|
|
|
|
DoIncDec(tree^.left, pc_lli, pc_gli, pc_ili);
|
|
|
|
|
|
|
|
minusminusop: {prefix --}
|
|
|
|
DoIncDec(tree^.left, pc_ldl, pc_gdl, pc_idl);
|
|
|
|
|
|
|
|
opminusminus: {postfix --}
|
|
|
|
DoIncDec(tree^.left, pc_lld, pc_gld, pc_ild);
|
|
|
|
|
2021-09-17 22:04:10 -05:00
|
|
|
uand: begin {unary & (address operator)}
|
2022-06-17 18:45:11 -05:00
|
|
|
if not (tree^.left^.token.kind in
|
|
|
|
[ident,compoundliteral,stringconst,uasterisk]) then
|
2021-09-17 22:04:10 -05:00
|
|
|
L_Value(tree^.left);
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree^.left, false);
|
2022-06-17 18:45:11 -05:00
|
|
|
if tree^.left^.token.kind = stringconst then begin
|
|
|
|
{build pointer-to-array type for address of string constant}
|
|
|
|
tType := pointer(Malloc(sizeof(typeRecord)));
|
2022-06-19 17:55:08 -05:00
|
|
|
tType^ := StringType(tree^.left^.token.prefix)^;
|
|
|
|
tType^.size := tree^.left^.token.sval^.length;
|
2022-06-18 18:53:29 -05:00
|
|
|
tType^.saveDisp := 0;
|
2022-06-19 17:55:08 -05:00
|
|
|
tType^.elements := tType^.size div tType^.aType^.size;
|
|
|
|
expressionType := MakePointerTo(tType);
|
|
|
|
end {if}
|
|
|
|
else if expressionType^.kind = arrayType then
|
|
|
|
expressionType := MakePointerTo(expressionType^.aType);
|
2021-09-17 22:04:10 -05:00
|
|
|
end; {case uand}
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
uasterisk: begin {unary * (indirection)}
|
|
|
|
GenerateCode(tree^.left);
|
|
|
|
lType := expressionType;
|
|
|
|
if lType^.kind in [functiontype,arrayType,pointerType] then begin
|
|
|
|
if lType^.kind = arrayType then
|
|
|
|
lType := lType^.aType
|
|
|
|
else if lType^.kind = pointerType then
|
|
|
|
lType := lType^.pType;
|
|
|
|
expressionType := lType;
|
2022-05-23 21:10:29 -05:00
|
|
|
isVolatile := tqVolatile in lType^.qualifiers;
|
2023-02-12 18:56:02 -06:00
|
|
|
if checkNullPointers then
|
|
|
|
if lType^.kind <> functionType then
|
|
|
|
Gen0(pc_ckp);
|
2017-10-21 18:40:19 -05:00
|
|
|
if lType^.kind = scalarType then
|
|
|
|
if lType^.baseType = cgVoid then
|
2022-07-12 18:34:58 -05:00
|
|
|
Gen2(pc_cnv, cgULong, cgVoid)
|
2017-10-21 18:40:19 -05:00
|
|
|
else
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), 0, lType^.baseType)
|
2017-10-21 18:40:19 -05:00
|
|
|
else if lType^.kind = pointerType then
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), 0, cgULong)
|
2017-10-21 18:40:19 -05:00
|
|
|
else if not
|
2015-12-30 21:30:44 -06:00
|
|
|
((lType^.kind in [functionType,arrayType,structType,unionType])
|
|
|
|
or ((lType^.kind = definedType) and {handle const struct/union}
|
|
|
|
(lType^.dType^.kind in [structType,unionType]))) then
|
2023-01-09 21:58:05 -06:00
|
|
|
Error(79)
|
|
|
|
else
|
|
|
|
CheckForIncompleteStructType;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
2016-10-15 23:51:58 -05:00
|
|
|
else
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(79);
|
|
|
|
end; {case uasterisk}
|
|
|
|
|
|
|
|
dotch: begin {.}
|
2023-02-12 18:56:02 -06:00
|
|
|
LoadAddress(tree^.left, checkNullPointers);
|
2023-05-07 18:28:31 -05:00
|
|
|
isBitfield := false;
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
2021-09-17 22:04:10 -05:00
|
|
|
if lType^.kind in [arrayType,pointerType,structType,unionType] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
if lType^.kind = arrayType then
|
|
|
|
lType := lType^.aType
|
|
|
|
else if lType^.kind = pointerType then
|
|
|
|
lType := lType^.pType;
|
|
|
|
DoSelection(lType, tree^.right, size);
|
|
|
|
if (size & $00007FFF) <> size then begin
|
|
|
|
GenLdcLong(size);
|
|
|
|
Gen0(pc_adl);
|
|
|
|
size := 0;
|
|
|
|
end; {else}
|
|
|
|
kind := expressionType^.kind;
|
2022-05-23 21:10:29 -05:00
|
|
|
isVolatile := tqVolatile in expressionType^.qualifiers;
|
2017-10-21 18:40:19 -05:00
|
|
|
if kind = scalarType then begin
|
|
|
|
et := expressionType^.baseType;
|
|
|
|
if isBitField then begin
|
|
|
|
GenLdcLong(size);
|
|
|
|
Gen0(pc_adl);
|
|
|
|
if unsigned then
|
|
|
|
Gen2t(pc_lbu, bitDisp, bitSize, et)
|
|
|
|
else
|
|
|
|
Gen2t(pc_lbf, bitDisp, bitSize, et);
|
|
|
|
end {if}
|
|
|
|
else
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), long(size).lsw, et);
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else if kind = pointerType then
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), long(size).lsw, cgULong)
|
2017-10-21 18:40:19 -05:00
|
|
|
else if kind = enumType then
|
2022-05-23 21:10:29 -05:00
|
|
|
Gen2t(pc_ind, ord(isVolatile), long(size).lsw, cgWord)
|
2017-10-21 18:40:19 -05:00
|
|
|
else if size <> 0 then
|
|
|
|
Gen1t(pc_inc, long(size).lsw, cgULong);
|
|
|
|
end {if}
|
|
|
|
else
|
|
|
|
Error(79);
|
|
|
|
end; {case dotch}
|
|
|
|
|
|
|
|
colonch: begin {? :}
|
|
|
|
GenerateCode(tree^.left); {evaluate the condition}
|
|
|
|
CompareToZero(pc_neq);
|
|
|
|
GenerateCode(tree^.middle); {evaluate true expression}
|
2022-06-23 22:04:03 -05:00
|
|
|
ValueExpressionConversions;
|
2017-10-21 18:40:19 -05:00
|
|
|
lType := expressionType;
|
2022-06-23 22:04:03 -05:00
|
|
|
tlastWasNullPtrConst := lastWasNullPtrConst;
|
2017-10-21 18:40:19 -05:00
|
|
|
GenerateCode(tree^.right); {evaluate false expression}
|
2022-06-23 22:04:03 -05:00
|
|
|
ValueExpressionConversions;
|
|
|
|
{check, compute, and convert types}
|
|
|
|
if (lType^.kind = pointerType) or (expressionType^.kind = pointerType)
|
|
|
|
then begin
|
|
|
|
if tlastWasNullPtrConst then begin
|
|
|
|
if lType^.kind = scalarType then
|
|
|
|
Gen2(pc_cnn, ord(lType^.baseType), ord(cgULong));
|
|
|
|
end {if}
|
|
|
|
else if lastWasNullPtrConst then begin
|
|
|
|
if expressionType^.kind = scalarType then
|
|
|
|
Gen2(pc_cnv, ord(expressionType^.baseType), ord(cgULong));
|
|
|
|
expressionType := lType;
|
|
|
|
end {if}
|
|
|
|
else if lType^.kind <> expressionType^.kind then {not both pointers}
|
|
|
|
Error(47)
|
|
|
|
else if IsVoid(lType^.pType) or IsVoid(expressionType^.pType) then begin
|
|
|
|
if not looseTypeChecks then
|
|
|
|
if (lType^.pType^.kind = functionType) or
|
|
|
|
(expressionType^.pType^.kind = functionType) then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(47);
|
2022-06-23 22:04:03 -05:00
|
|
|
expressionType := MakePointerTo(MakeQualifiedType(voidPtr,
|
|
|
|
lType^.pType^.qualifiers+expressionType^.pType^.qualifiers));
|
|
|
|
end {else if}
|
|
|
|
else if CompTypes(Unqualify(lType^.pType),
|
|
|
|
Unqualify(expressionType^.pType)) then begin
|
|
|
|
if not looseTypeChecks then
|
|
|
|
if not StrictCompTypes(Unqualify(lType^.pType),
|
|
|
|
Unqualify(expressionType^.pType)) then
|
2017-10-21 18:40:19 -05:00
|
|
|
Error(47);
|
2022-06-23 22:04:03 -05:00
|
|
|
expressionType := MakePointerTo(MakeQualifiedType(MakeCompositeType(
|
|
|
|
Unqualify(lType^.pType),Unqualify(expressionType^.pType)),
|
|
|
|
lType^.pType^.qualifiers+expressionType^.pType^.qualifiers));
|
|
|
|
end {else if}
|
|
|
|
else
|
|
|
|
Error(47);
|
|
|
|
et := cgULong;
|
|
|
|
end {if}
|
|
|
|
else if lType^.kind in [structType, unionType] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
if not CompTypes(lType, expressionType) then
|
|
|
|
Error(47);
|
2022-06-23 22:04:03 -05:00
|
|
|
et := cgULong;
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else begin
|
2022-06-23 22:04:03 -05:00
|
|
|
if IsVoid(lType) and IsVoid(expressionType) then
|
2016-10-14 20:37:10 -05:00
|
|
|
et := cgVoid
|
|
|
|
else
|
|
|
|
et := UsualBinaryConversions(lType);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {else}
|
2022-06-23 22:04:03 -05:00
|
|
|
{generate the operation}
|
|
|
|
Gen0(pc_bno);
|
|
|
|
Gen0t(pc_tri, et);
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {case colonch}
|
|
|
|
|
|
|
|
castoper: begin {(cast)}
|
|
|
|
GenerateCode(tree^.left);
|
2022-06-23 22:04:03 -05:00
|
|
|
if lastWasNullPtrConst then
|
|
|
|
if expressionType^.kind = scalarType then
|
|
|
|
if tree^.castType^.kind = pointerType then
|
|
|
|
if IsVoid(tree^.castType^.pType) then
|
|
|
|
if tree^.castType^.pType^.qualifiers = [] then
|
|
|
|
isNullPtrConst := true;
|
2017-10-21 18:40:19 -05:00
|
|
|
Cast(tree^.castType);
|
|
|
|
end; {case castoper}
|
|
|
|
|
|
|
|
otherwise:
|
|
|
|
Error(57);
|
|
|
|
|
|
|
|
end; {case}
|
|
|
|
if doDispose then
|
|
|
|
dispose(tree);
|
2022-06-23 22:04:03 -05:00
|
|
|
lastWasNullPtrConst := isNullPtrConst;
|
2023-02-28 22:36:42 -06:00
|
|
|
lastWasConst := isConst;
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {GenerateCode}
|
|
|
|
|
|
|
|
|
|
|
|
procedure Expression {kind: expressionKind; stopSym: tokenSet};
|
|
|
|
|
|
|
|
{ handle an expression }
|
|
|
|
{ }
|
|
|
|
{ parameters: }
|
|
|
|
{ kind - Kind of expression; determines what operations }
|
|
|
|
{ and what kind of operands are allowed. }
|
|
|
|
{ stopSym - Set of symbols that can mark the end of an }
|
|
|
|
{ expression; used to skip tokens after syntax }
|
|
|
|
{ errors and to block certain operations. For }
|
|
|
|
{ example, the comma operator is not allowed in }
|
|
|
|
{ an expression when evaluating a function }
|
|
|
|
{ parameter list. }
|
|
|
|
{ }
|
|
|
|
{ variables: }
|
|
|
|
{ realExpressionValue - value of a real constant }
|
|
|
|
{ expression }
|
|
|
|
{ expressionValue - value of a constant expression }
|
2019-12-22 23:49:14 -06:00
|
|
|
{ expressionType - type of the expression }
|
2017-10-21 18:40:19 -05:00
|
|
|
|
|
|
|
label 1;
|
|
|
|
|
|
|
|
var
|
|
|
|
lcodeGeneration: boolean; {local copy of codeGeneration}
|
|
|
|
ldoDispose: boolean; {local copy of doDispose}
|
|
|
|
tree: tokenPtr; {expression tree}
|
|
|
|
castValue: tokenPtr; {element being type cast}
|
|
|
|
|
|
|
|
begin {Expression}
|
|
|
|
errorFound := false; {no error so far}
|
|
|
|
tree := ExpressionTree(kind, stopSym); {create the expression tree}
|
|
|
|
if kind = normalExpression then begin {generate code from the expression tree}
|
|
|
|
if not errorFound then begin
|
|
|
|
doDispose := true;
|
|
|
|
GenerateCode(tree);
|
2019-12-22 23:49:14 -06:00
|
|
|
end {if}
|
|
|
|
else
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr; {set default type in case of error}
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else begin {record the expression for an initializer}
|
|
|
|
initializerTree := tree;
|
|
|
|
isConstant := false;
|
2021-02-04 12:44:44 -06:00
|
|
|
llExpressionValue.lo := 0;
|
|
|
|
llExpressionValue.hi := 0;
|
2021-02-17 19:41:46 -06:00
|
|
|
expressionIsLongLong := false;
|
2017-10-21 18:40:19 -05:00
|
|
|
if errorFound then begin
|
|
|
|
DisposeTree(initializerTree);
|
|
|
|
initializerTree := nil;
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr; {set default type in case of error}
|
2017-10-21 18:40:19 -05:00
|
|
|
end {if}
|
|
|
|
else begin
|
|
|
|
ldoDispose := doDispose; {find the expression type}
|
|
|
|
doDispose := false;
|
|
|
|
lcodeGeneration := codeGeneration;
|
|
|
|
codeGeneration := false;
|
|
|
|
GenerateCode(tree);
|
|
|
|
doDispose := ldoDispose;
|
|
|
|
codeGeneration := lCodeGeneration and (numErrors = 0);
|
|
|
|
{record the expression}
|
|
|
|
if tree^.token.kind = castoper then begin
|
|
|
|
castValue := tree^.left;
|
|
|
|
while castValue^.token.kind = castoper do
|
|
|
|
castValue := castValue^.left;
|
2021-03-07 13:38:21 -06:00
|
|
|
if castValue^.token.kind in
|
2021-10-11 20:54:37 -05:00
|
|
|
[intconst,uintconst,ushortconst,charconst,scharconst,ucharconst]
|
|
|
|
then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionValue := castValue^.token.ival;
|
|
|
|
isConstant := true;
|
|
|
|
expressionType := tree^.castType;
|
2021-10-11 20:54:37 -05:00
|
|
|
if (castValue^.token.kind in [uintconst,ushortconst])
|
2017-10-21 18:40:19 -05:00
|
|
|
or (expressionType^.kind = pointerType) then
|
|
|
|
expressionValue := expressionValue & $0000FFFF;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
if castValue^.token.kind in [longconst,ulongconst] then begin
|
|
|
|
expressionValue := castValue^.token.lval;
|
|
|
|
isConstant := true;
|
|
|
|
expressionType := tree^.castType;
|
|
|
|
goto 1;
|
|
|
|
end; {if}
|
|
|
|
end; {if}
|
2021-03-07 13:38:21 -06:00
|
|
|
if tree^.token.kind in [intconst,charconst,scharconst,ucharconst] then
|
|
|
|
begin
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionValue := tree^.token.ival;
|
2021-03-07 13:38:21 -06:00
|
|
|
if tree^.token.kind = intconst then
|
|
|
|
expressionType := intPtr
|
|
|
|
else if tree^.token.kind = charconst then
|
|
|
|
expressionType := charPtr
|
|
|
|
else if tree^.token.kind = scharconst then
|
|
|
|
expressionType := scharPtr
|
|
|
|
else {if tree^.token.kind = ucharconst then}
|
|
|
|
expressionType := ucharPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
isConstant := true;
|
|
|
|
end {else if}
|
2021-10-11 20:54:37 -05:00
|
|
|
else if tree^.token.kind in [uintconst,ushortconst] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionValue := tree^.token.ival;
|
2016-10-12 20:36:03 -05:00
|
|
|
expressionValue := expressionValue & $0000FFFF;
|
2021-10-11 20:54:37 -05:00
|
|
|
if tree^.token.kind = uintconst then
|
|
|
|
expressionType := uIntPtr
|
|
|
|
else {if tree^.token.kind = ushortconst then}
|
|
|
|
expressionType := uShortPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
isConstant := true;
|
|
|
|
end {else if}
|
|
|
|
else if tree^.token.kind = longconst then begin
|
|
|
|
expressionValue := tree^.token.lval;
|
|
|
|
expressionType := longPtr;
|
|
|
|
isConstant := true;
|
|
|
|
end {else if}
|
|
|
|
else if tree^.token.kind = ulongconst then begin
|
|
|
|
expressionValue := tree^.token.lval;
|
|
|
|
expressionType := ulongPtr;
|
|
|
|
isConstant := true;
|
|
|
|
end {else if}
|
2021-02-03 23:11:23 -06:00
|
|
|
else if tree^.token.kind = longlongconst then begin
|
2021-02-04 12:44:44 -06:00
|
|
|
llExpressionValue := tree^.token.qval;
|
2021-02-17 19:41:46 -06:00
|
|
|
expressionIsLongLong := true;
|
2021-02-04 12:44:44 -06:00
|
|
|
if ((llExpressionValue.hi = 0) and (llExpressionValue.lo >= 0))
|
|
|
|
or ((llExpressionValue.hi = -1) and (llExpressionValue.lo < 0)) then
|
|
|
|
expressionValue := llExpressionValue.lo
|
|
|
|
else if llExpressionValue.hi < 0 then
|
|
|
|
expressionValue := $80000000
|
|
|
|
else
|
|
|
|
expressionValue := $7fffffff;
|
2021-02-03 23:11:23 -06:00
|
|
|
expressionType := longLongPtr;
|
|
|
|
isConstant := true;
|
|
|
|
end {else if}
|
|
|
|
else if tree^.token.kind = ulonglongconst then begin
|
2021-02-04 12:44:44 -06:00
|
|
|
llExpressionValue := tree^.token.qval;
|
2021-02-17 19:41:46 -06:00
|
|
|
expressionIsLongLong := true;
|
2021-02-04 12:44:44 -06:00
|
|
|
if llExpressionValue.hi = 0 then
|
|
|
|
expressionValue := llExpressionValue.lo
|
|
|
|
else
|
|
|
|
expressionValue := $FFFFFFFF;
|
2021-02-03 23:11:23 -06:00
|
|
|
expressionType := ulongLongPtr;
|
|
|
|
isConstant := true;
|
|
|
|
end {else if}
|
2021-03-07 00:48:51 -06:00
|
|
|
else if tree^.token.kind in
|
|
|
|
[floatconst,doubleconst,extendedconst,compconst] then begin
|
2017-10-21 18:40:19 -05:00
|
|
|
realExpressionValue := tree^.token.rval;
|
2021-03-07 00:48:51 -06:00
|
|
|
if tree^.token.kind = extendedconst then
|
|
|
|
expressionType := extendedPtr
|
|
|
|
else if tree^.token.kind = doubleconst then
|
|
|
|
expressionType := doublePtr
|
|
|
|
else if tree^.token.kind = floatconst then
|
|
|
|
expressionType := floatPtr
|
|
|
|
else {if tree^.token.kind = compconst then}
|
|
|
|
expressionType := compPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
isConstant := true;
|
|
|
|
if kind in [arrayExpression,preprocessorExpression] then begin
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionValue := 1;
|
|
|
|
Error(47);
|
|
|
|
end; {if}
|
|
|
|
end {else if}
|
|
|
|
else if tree^.token.kind = stringconst then begin
|
|
|
|
expressionValue := ord4(tree^.token.sval);
|
2021-10-11 20:54:37 -05:00
|
|
|
expressionType := StringType(tree^.token.prefix);
|
2017-10-21 18:40:19 -05:00
|
|
|
isConstant := true;
|
|
|
|
if kind in [arrayExpression,preprocessorExpression] then begin
|
2020-02-29 22:43:29 -06:00
|
|
|
expressionType := intPtr;
|
2017-10-21 18:40:19 -05:00
|
|
|
expressionValue := 1;
|
|
|
|
Error(47);
|
|
|
|
end; {if}
|
|
|
|
end {else if}
|
|
|
|
else if kind in [arrayExpression,preprocessorExpression] then begin
|
|
|
|
DisposeTree(initializerTree);
|
|
|
|
expressionValue := 1;
|
|
|
|
end; {else if}
|
|
|
|
end; {else}
|
|
|
|
end; {else}
|
|
|
|
1:
|
|
|
|
end; {Expression}
|
|
|
|
|
|
|
|
|
2021-02-17 19:41:46 -06:00
|
|
|
procedure GetLLExpressionValue {var val: longlong};
|
|
|
|
|
|
|
|
{ get the value of the last integer constant expression as a }
|
|
|
|
{ long long (whether it had long long type or not). }
|
|
|
|
|
|
|
|
begin {GetLLExpressionValue}
|
|
|
|
if expressionIsLongLong then
|
|
|
|
val := llExpressionValue
|
|
|
|
else begin
|
|
|
|
val.lo := expressionValue;
|
|
|
|
val.hi := 0;
|
|
|
|
if expressionValue < 0 then
|
|
|
|
if expressionType^.kind = scalarType then
|
|
|
|
if expressionType^.baseType in [cgByte,cgWord,cgLong] then
|
|
|
|
val.hi := -1;
|
|
|
|
end;
|
|
|
|
end; {GetLLExpressionValue}
|
|
|
|
|
|
|
|
|
2017-10-21 18:40:19 -05:00
|
|
|
procedure InitExpression;
|
|
|
|
|
|
|
|
{ initialize the expression handler }
|
|
|
|
|
|
|
|
begin {InitExpression}
|
2021-02-03 23:11:23 -06:00
|
|
|
startTerm := [ident,intconst,uintconst,longconst,ulongconst,longlongconst,
|
2021-03-07 00:48:51 -06:00
|
|
|
ulonglongconst,floatconst,doubleconst,extendedconst,compconst,
|
2021-10-11 20:54:37 -05:00
|
|
|
charconst,scharconst,ucharconst,ushortconst,stringconst];
|
2017-10-21 18:40:19 -05:00
|
|
|
startExpression:= startTerm +
|
|
|
|
[lparench,asteriskch,andch,plusch,minusch,excch,tildech,sizeofsy,
|
2024-08-29 13:16:01 -05:00
|
|
|
plusplusop,minusminusop,typedef,_Alignofsy,alignofsy,_Genericsy];
|
2017-10-21 18:40:19 -05:00
|
|
|
end; {InitExpression}
|
|
|
|
|
|
|
|
end.
|
|
|
|
|
|
|
|
{$append 'expression.asm'}
|