mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 01:01:34 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			2885 lines
		
	
	
		
			105 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			2885 lines
		
	
	
		
			105 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     $Id$
 | |
|     Copyright (c) 1998-2002 by Florian Klaempfl
 | |
| 
 | |
|     Does parsing of expression for Free Pascal
 | |
| 
 | |
|     This program is free software; you can redistribute it and/or modify
 | |
|     it under the terms of the GNU General Public License as published by
 | |
|     the Free Software Foundation; either version 2 of the License, or
 | |
|     (at your option) any later version.
 | |
| 
 | |
|     This program is distributed in the hope that it will be useful,
 | |
|     but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|     GNU General Public License for more details.
 | |
| 
 | |
|     You should have received a copy of the GNU General Public License
 | |
|     along with this program; if not, write to the Free Software
 | |
|     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 | |
| 
 | |
|  ****************************************************************************
 | |
| }
 | |
| unit pexpr;
 | |
| 
 | |
| {$i fpcdefs.inc}
 | |
| 
 | |
| interface
 | |
| 
 | |
|     uses
 | |
|       symtype,symdef,symbase,
 | |
|       node,
 | |
|       globals,
 | |
|       cpuinfo;
 | |
| 
 | |
|     { reads a whole expression }
 | |
|     function expr : tnode;
 | |
| 
 | |
|     { reads an expression without assignements and .. }
 | |
|     function comp_expr(accept_equal : boolean):tnode;
 | |
| 
 | |
|     { reads a single factor }
 | |
|     function factor(getaddr : boolean) : tnode;
 | |
| 
 | |
|     procedure string_dec(var t: ttype);
 | |
| 
 | |
|     procedure symlist_to_node(var p1:tnode;st:tsymtable;pl:tsymlist);
 | |
| 
 | |
|     function node_to_symlist(p1:tnode):tsymlist;
 | |
| 
 | |
|     function parse_paras(__colon,in_prop_paras : boolean) : tnode;
 | |
| 
 | |
|     { the ID token has to be consumed before calling this function }
 | |
|     procedure do_member_read(classh:tobjectdef;getaddr : boolean;sym : tsym;var p1 : tnode;var again : boolean;callnflags:tnodeflags);
 | |
| 
 | |
| {$ifdef int64funcresok}
 | |
|     function get_intconst:TConstExprInt;
 | |
| {$else int64funcresok}
 | |
|     function get_intconst:longint;
 | |
| {$endif int64funcresok}
 | |
| 
 | |
|     function get_stringconst:string;
 | |
| 
 | |
| implementation
 | |
| 
 | |
|     uses
 | |
| {$ifdef delphi}
 | |
|        SysUtils,
 | |
| {$endif}
 | |
|        { common }
 | |
|        cutils,
 | |
|        { global }
 | |
|        globtype,tokens,verbose,
 | |
|        systems,widestr,
 | |
|        { symtable }
 | |
|        symconst,symtable,symsym,defutil,defcmp,
 | |
|        { pass 1 }
 | |
|        pass_1,htypechk,
 | |
|        nmat,nadd,ncal,nmem,nset,ncnv,ninl,ncon,nld,nflw,nbas,
 | |
|        { parser }
 | |
|        scanner,
 | |
|        pbase,pinline,
 | |
|        { codegen }
 | |
|        procinfo
 | |
|        ;
 | |
| 
 | |
|     { sub_expr(opmultiply) is need to get -1 ** 4 to be
 | |
|       read as - (1**4) and not (-1)**4 PM }
 | |
|     type
 | |
|       Toperator_precedence=(opcompare,opaddition,opmultiply,oppower);
 | |
| 
 | |
|     const
 | |
|       highest_precedence = oppower;
 | |
| 
 | |
|     function sub_expr(pred_level:Toperator_precedence;accept_equal : boolean):tnode;forward;
 | |
| 
 | |
|     const
 | |
|        { true, if the inherited call is anonymous }
 | |
|        anon_inherited : boolean = false;
 | |
| 
 | |
| 
 | |
| 
 | |
|     procedure string_dec(var t: ttype);
 | |
|     { reads a string type with optional length }
 | |
|     { and returns a pointer to the string      }
 | |
|     { definition                               }
 | |
|       var
 | |
|          p : tnode;
 | |
|       begin
 | |
|          t:=cshortstringtype;
 | |
|          consume(_STRING);
 | |
|          if token=_LECKKLAMMER then
 | |
|            begin
 | |
|               consume(_LECKKLAMMER);
 | |
|               p:=comp_expr(true);
 | |
|               if not is_constintnode(p) then
 | |
|                 begin
 | |
|                   Message(cg_e_illegal_expression);
 | |
|                   { error recovery }
 | |
|                   consume(_RECKKLAMMER);
 | |
|                 end
 | |
|               else
 | |
|                 begin
 | |
|                  if (tordconstnode(p).value<=0) then
 | |
|                    begin
 | |
|                       Message(parser_e_invalid_string_size);
 | |
|                       tordconstnode(p).value:=255;
 | |
|                    end;
 | |
|                  consume(_RECKKLAMMER);
 | |
|                  if tordconstnode(p).value>255 then
 | |
|                   begin
 | |
|                     { longstring is currently unsupported (CEC)! }
 | |
| {                   t.setdef(tstringdef.createlong(tordconstnode(p).value))}
 | |
|                      Message(parser_e_invalid_string_size);
 | |
|                      tordconstnode(p).value:=255;
 | |
|                      t.setdef(tstringdef.createshort(tordconstnode(p).value));
 | |
|                   end
 | |
|                  else
 | |
|                    if tordconstnode(p).value<>255 then
 | |
|                      t.setdef(tstringdef.createshort(tordconstnode(p).value));
 | |
|                end;
 | |
|               p.free;
 | |
|            end
 | |
|           else
 | |
|             begin
 | |
|                if cs_ansistrings in aktlocalswitches then
 | |
|                  t:=cansistringtype
 | |
|                else
 | |
|                  t:=cshortstringtype;
 | |
|             end;
 | |
|        end;
 | |
| 
 | |
| 
 | |
| 
 | |
|     procedure symlist_to_node(var p1:tnode;st:tsymtable;pl:tsymlist);
 | |
|       var
 | |
|         plist : psymlistitem;
 | |
|       begin
 | |
|         plist:=pl.firstsym;
 | |
|         while assigned(plist) do
 | |
|          begin
 | |
|            case plist^.sltype of
 | |
|              sl_load :
 | |
|                begin
 | |
|                  if not assigned(st) then
 | |
|                    st:=plist^.sym.owner;
 | |
|                  { p1 can already contain the loadnode of
 | |
|                    the class variable. When there is no tree yet we
 | |
|                    may need to load it for with or objects }
 | |
|                  if not assigned(p1) then
 | |
|                   begin
 | |
|                     case st.symtabletype of
 | |
|                       withsymtable :
 | |
|                         p1:=tnode(twithsymtable(st).withrefnode).getcopy;
 | |
|                       objectsymtable :
 | |
|                         p1:=load_self_node;
 | |
|                     end;
 | |
|                   end;
 | |
|                  if assigned(p1) then
 | |
|                   p1:=csubscriptnode.create(plist^.sym,p1)
 | |
|                  else
 | |
|                   p1:=cloadnode.create(plist^.sym,st);
 | |
|                end;
 | |
|              sl_subscript :
 | |
|                p1:=csubscriptnode.create(plist^.sym,p1);
 | |
|              sl_typeconv :
 | |
|                p1:=ctypeconvnode.create_explicit(p1,plist^.tt);
 | |
|              sl_vec :
 | |
|                p1:=cvecnode.create(p1,cordconstnode.create(plist^.value,s32bittype,true));
 | |
|              else
 | |
|                internalerror(200110205);
 | |
|            end;
 | |
|            plist:=plist^.next;
 | |
|          end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function node_to_symlist(p1:tnode):tsymlist;
 | |
|       var
 | |
|         sl : tsymlist;
 | |
| 
 | |
|         procedure addnode(p:tnode);
 | |
|         begin
 | |
|           case p.nodetype of
 | |
|             subscriptn :
 | |
|               begin
 | |
|                 addnode(tsubscriptnode(p).left);
 | |
|                 sl.addsym(sl_subscript,tsubscriptnode(p).vs);
 | |
|               end;
 | |
|             typeconvn :
 | |
|               begin
 | |
|                 addnode(ttypeconvnode(p).left);
 | |
|                 sl.addtype(sl_typeconv,ttypeconvnode(p).totype);
 | |
|               end;
 | |
|             vecn :
 | |
|               begin
 | |
|                 addnode(tsubscriptnode(p).left);
 | |
|                 if tvecnode(p).right.nodetype=ordconstn then
 | |
|                   sl.addconst(sl_vec,tordconstnode(tvecnode(p).right).value)
 | |
|                 else
 | |
|                   begin
 | |
|                     Message(cg_e_illegal_expression);
 | |
|                     { recovery }
 | |
|                     sl.addconst(sl_vec,0);
 | |
|                   end;
 | |
|              end;
 | |
|             loadn :
 | |
|               sl.addsym(sl_load,tloadnode(p).symtableentry);
 | |
|             else
 | |
|               internalerror(200310282);
 | |
|           end;
 | |
|         end;
 | |
| 
 | |
|       begin
 | |
|         sl:=tsymlist.create;
 | |
|         addnode(p1);
 | |
|         result:=sl;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function parse_paras(__colon,in_prop_paras : boolean) : tnode;
 | |
| 
 | |
|       var
 | |
|          p1,p2 : tnode;
 | |
|          end_of_paras : ttoken;
 | |
|          prev_in_args : boolean;
 | |
|          old_allow_array_constructor : boolean;
 | |
|       begin
 | |
|          if in_prop_paras  then
 | |
|            end_of_paras:=_RECKKLAMMER
 | |
|          else
 | |
|            end_of_paras:=_RKLAMMER;
 | |
|          if token=end_of_paras then
 | |
|            begin
 | |
|               parse_paras:=nil;
 | |
|               exit;
 | |
|            end;
 | |
|          { save old values }
 | |
|          prev_in_args:=in_args;
 | |
|          old_allow_array_constructor:=allow_array_constructor;
 | |
|          { set para parsing values }
 | |
|          in_args:=true;
 | |
|          inc(parsing_para_level);
 | |
|          allow_array_constructor:=true;
 | |
|          p2:=nil;
 | |
|          while true do
 | |
|            begin
 | |
|               p1:=comp_expr(true);
 | |
|               p2:=ccallparanode.create(p1,p2);
 | |
|               { it's for the str(l:5,s); }
 | |
|               if __colon and (token=_COLON) then
 | |
|                 begin
 | |
|                    consume(_COLON);
 | |
|                    p1:=comp_expr(true);
 | |
|                    p2:=ccallparanode.create(p1,p2);
 | |
|                    include(tcallparanode(p2).callparaflags,cpf_is_colon_para);
 | |
|                    if token=_COLON then
 | |
|                      begin
 | |
|                         consume(_COLON);
 | |
|                         p1:=comp_expr(true);
 | |
|                         p2:=ccallparanode.create(p1,p2);
 | |
|                         include(tcallparanode(p2).callparaflags,cpf_is_colon_para);
 | |
|                      end
 | |
|                 end;
 | |
|               if token=_COMMA then
 | |
|                 consume(_COMMA)
 | |
|               else
 | |
|                 break;
 | |
|            end;
 | |
|          allow_array_constructor:=old_allow_array_constructor;
 | |
|          dec(parsing_para_level);
 | |
|          in_args:=prev_in_args;
 | |
|          parse_paras:=p2;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure check_tp_procvar(var p : tnode);
 | |
|       var
 | |
|          hp,
 | |
|          p1 : tnode;
 | |
|       begin
 | |
|          if (m_tp_procvar in aktmodeswitches) and
 | |
|             (token<>_ASSIGNMENT) and
 | |
|             (not got_addrn) and
 | |
|             (block_type=bt_body) then
 | |
|           begin
 | |
|             { ignore vecn,subscriptn }
 | |
|             hp:=p;
 | |
|             repeat
 | |
|               case hp.nodetype of
 | |
|                 vecn :
 | |
|                   hp:=tvecnode(hp).left;
 | |
|                 subscriptn :
 | |
|                   hp:=tsubscriptnode(hp).left;
 | |
|                 else
 | |
|                   break;
 | |
|               end;
 | |
|             until false;
 | |
|             if (hp.nodetype=loadn) then
 | |
|                begin
 | |
|                   { get the resulttype of p }
 | |
|                   do_resulttypepass(p);
 | |
|                   { convert the procvar load to a call:
 | |
|                      - not expecting a procvar
 | |
|                      - the procvar does not get arguments, when it
 | |
|                        requires arguments the callnode will fail
 | |
|                        Note: When arguments were passed there was no loadn }
 | |
|                   if (getprocvardef=nil) and
 | |
|                      (p.resulttype.def.deftype=procvardef) and
 | |
|                      (tprocvardef(p.resulttype.def).minparacount=0) then
 | |
|                     begin
 | |
|                        p1:=ccallnode.create_procvar(nil,p);
 | |
|                        resulttypepass(p1);
 | |
|                        p:=p1;
 | |
|                     end;
 | |
|                end;
 | |
|           end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|      function statement_syssym(l : longint) : tnode;
 | |
|       var
 | |
|         p1,p2,paras  : tnode;
 | |
|         err,
 | |
|         prev_in_args : boolean;
 | |
|       begin
 | |
|         prev_in_args:=in_args;
 | |
|         case l of
 | |
| 
 | |
|           in_new_x :
 | |
|             begin
 | |
|               if afterassignment or in_args then
 | |
|                statement_syssym:=new_function
 | |
|               else
 | |
|                statement_syssym:=new_dispose_statement(true);
 | |
|             end;
 | |
| 
 | |
|           in_dispose_x :
 | |
|             begin
 | |
|               statement_syssym:=new_dispose_statement(false);
 | |
|             end;
 | |
| 
 | |
|           in_ord_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               consume(_RKLAMMER);
 | |
|               p1:=geninlinenode(in_ord_x,false,p1);
 | |
|               statement_syssym := p1;
 | |
|             end;
 | |
| 
 | |
|           in_exit :
 | |
|             begin
 | |
|                if try_to_consume(_LKLAMMER) then
 | |
|                  begin
 | |
|                     p1:=comp_expr(true);
 | |
|                     consume(_RKLAMMER);
 | |
|                     if (block_type=bt_except) then
 | |
|                       begin
 | |
|                         Message(parser_e_exit_with_argument_not__possible);
 | |
|                         { recovery }
 | |
|                         p1.free;
 | |
|                         p1:=nil;
 | |
|                       end
 | |
|                     else if (not assigned(current_procinfo) or
 | |
|                         is_void(current_procinfo.procdef.rettype.def)) then
 | |
|                       begin
 | |
|                         Message(parser_e_void_function);
 | |
|                         { recovery }
 | |
|                         p1.free;
 | |
|                         p1:=nil;
 | |
|                       end;
 | |
|                  end
 | |
|                else
 | |
|                  p1:=nil;
 | |
|                statement_syssym:=cexitnode.create(p1);
 | |
|             end;
 | |
| 
 | |
|           in_break :
 | |
|             begin
 | |
|               statement_syssym:=cbreaknode.create;
 | |
|             end;
 | |
| 
 | |
|           in_continue :
 | |
|             begin
 | |
|               statement_syssym:=ccontinuenode.create;
 | |
|             end;
 | |
| 
 | |
|           in_typeof_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               consume(_RKLAMMER);
 | |
|               if p1.nodetype=typen then
 | |
|                 ttypenode(p1).allowed:=true;
 | |
|               { Allow classrefdef, which is required for
 | |
|                 Typeof(self) in static class methods }
 | |
|               if (p1.resulttype.def.deftype = objectdef) or
 | |
|                  (assigned(current_procinfo) and
 | |
|                   ((po_classmethod in current_procinfo.procdef.procoptions) or
 | |
|                    (po_staticmethod in current_procinfo.procdef.procoptions)) and
 | |
|                   (p1.resulttype.def.deftype=classrefdef)) then
 | |
|                statement_syssym:=geninlinenode(in_typeof_x,false,p1)
 | |
|               else
 | |
|                begin
 | |
|                  Message(parser_e_class_id_expected);
 | |
|                  p1.destroy;
 | |
|                  statement_syssym:=cerrornode.create;
 | |
|                end;
 | |
|             end;
 | |
| 
 | |
|           in_sizeof_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               consume(_RKLAMMER);
 | |
|               if (p1.nodetype<>typen) and
 | |
|                  (
 | |
|                   (is_object(p1.resulttype.def) and
 | |
|                    (oo_has_constructor in tobjectdef(p1.resulttype.def).objectoptions)) or
 | |
|                   is_open_array(p1.resulttype.def) or
 | |
|                   is_open_string(p1.resulttype.def)
 | |
|                  ) then
 | |
|                statement_syssym:=geninlinenode(in_sizeof_x,false,p1)
 | |
|               else
 | |
|                begin
 | |
|                  statement_syssym:=cordconstnode.create(p1.resulttype.def.size,s32bittype,true);
 | |
|                  { p1 not needed !}
 | |
|                  p1.destroy;
 | |
|                end;
 | |
|             end;
 | |
| 
 | |
|           in_typeinfo_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               if p1.nodetype=typen then
 | |
|                 ttypenode(p1).allowed:=true
 | |
|               else
 | |
|                 begin
 | |
|                    p1.destroy;
 | |
|                    p1:=cerrornode.create;
 | |
|                    Message(parser_e_illegal_parameter_list);
 | |
|                 end;
 | |
|               consume(_RKLAMMER);
 | |
|               p2:=geninlinenode(in_typeinfo_x,false,p1);
 | |
|               statement_syssym:=p2;
 | |
|             end;
 | |
| 
 | |
|           in_assigned_x :
 | |
|             begin
 | |
|               err:=false;
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               if not codegenerror then
 | |
|                begin
 | |
|                  { With tp procvars we allways need to load a
 | |
|                    procvar when it is passed, but not when the
 | |
|                    callnode is inserted due a property or has
 | |
|                    arguments }
 | |
|                  if (m_tp_procvar in aktmodeswitches) and
 | |
|                     (p1.nodetype=calln) and
 | |
|                     (tcallnode(p1).para_count=0) and
 | |
|                     not(nf_isproperty in tcallnode(p1).flags) then
 | |
|                    load_procvar_from_calln(p1);
 | |
| 
 | |
|                  case p1.resulttype.def.deftype of
 | |
|                    procdef, { procvar }
 | |
|                    pointerdef,
 | |
|                    procvardef,
 | |
|                    classrefdef : ;
 | |
|                    objectdef :
 | |
|                      if not is_class_or_interface(p1.resulttype.def) then
 | |
|                        begin
 | |
|                          Message(parser_e_illegal_parameter_list);
 | |
|                          err:=true;
 | |
|                        end;
 | |
|                    else
 | |
|                      begin
 | |
|                        Message(parser_e_illegal_parameter_list);
 | |
|                        err:=true;
 | |
|                      end;
 | |
|                  end;
 | |
|                end
 | |
|               else
 | |
|                err:=true;
 | |
|               if not err then
 | |
|                begin
 | |
|                  p2:=ccallparanode.create(p1,nil);
 | |
|                  p2:=geninlinenode(in_assigned_x,false,p2);
 | |
|                end
 | |
|               else
 | |
|                begin
 | |
|                  p1.free;
 | |
|                  p2:=cerrornode.create;
 | |
|                end;
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p2;
 | |
|             end;
 | |
| 
 | |
|           in_addr_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               p1:=caddrnode.create(p1);
 | |
|               if cs_typed_addresses in aktlocalswitches then
 | |
|                 include(p1.flags,nf_typedaddr);
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p1;
 | |
|             end;
 | |
| 
 | |
|           in_ofs_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               p1:=caddrnode.create(p1);
 | |
|               do_resulttypepass(p1);
 | |
|               { Ofs() returns a cardinal, not a pointer }
 | |
|               p1.resulttype:=u32bittype;
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p1;
 | |
|             end;
 | |
| 
 | |
|           in_seg_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               p1:=geninlinenode(in_seg_x,false,p1);
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p1;
 | |
|             end;
 | |
| 
 | |
|           in_high_x,
 | |
|           in_low_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               p2:=geninlinenode(l,false,p1);
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p2;
 | |
|             end;
 | |
| 
 | |
|           in_succ_x,
 | |
|           in_pred_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               p2:=geninlinenode(l,false,p1);
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p2;
 | |
|             end;
 | |
| 
 | |
|           in_inc_x,
 | |
|           in_dec_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               if token=_COMMA then
 | |
|                begin
 | |
|                  consume(_COMMA);
 | |
|                  p2:=ccallparanode.create(comp_expr(true),nil);
 | |
|                end
 | |
|               else
 | |
|                p2:=nil;
 | |
|               p2:=ccallparanode.create(p1,p2);
 | |
|               statement_syssym:=geninlinenode(l,false,p2);
 | |
|               consume(_RKLAMMER);
 | |
|             end;
 | |
| 
 | |
|           in_initialize_x:
 | |
|             begin
 | |
|               statement_syssym:=inline_initialize;
 | |
|             end;
 | |
| 
 | |
|           in_finalize_x:
 | |
|             begin
 | |
|               statement_syssym:=inline_finalize;
 | |
|             end;
 | |
| 
 | |
|           in_copy_x:
 | |
|             begin
 | |
|               statement_syssym:=inline_copy;
 | |
|             end;
 | |
| 
 | |
|           in_concat_x :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p2:=nil;
 | |
|               while true do
 | |
|                begin
 | |
|                  p1:=comp_expr(true);
 | |
|                  set_varstate(p1,vs_used,true);
 | |
|                  if not((p1.resulttype.def.deftype=stringdef) or
 | |
|                         ((p1.resulttype.def.deftype=orddef) and
 | |
|                          (torddef(p1.resulttype.def).typ=uchar))) then
 | |
|                    Message(parser_e_illegal_parameter_list);
 | |
|                  if p2<>nil then
 | |
|                   p2:=caddnode.create(addn,p2,p1)
 | |
|                  else
 | |
|                   p2:=p1;
 | |
|                  if token=_COMMA then
 | |
|                   consume(_COMMA)
 | |
|                  else
 | |
|                   break;
 | |
|                end;
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p2;
 | |
|             end;
 | |
| 
 | |
|           in_read_x,
 | |
|           in_readln_x :
 | |
|             begin
 | |
|               if token=_LKLAMMER then
 | |
|                begin
 | |
|                  consume(_LKLAMMER);
 | |
|                  paras:=parse_paras(false,false);
 | |
|                  consume(_RKLAMMER);
 | |
|                end
 | |
|               else
 | |
|                paras:=nil;
 | |
|               p1:=geninlinenode(l,false,paras);
 | |
|               statement_syssym := p1;
 | |
|             end;
 | |
| 
 | |
|           in_setlength_x:
 | |
|             begin
 | |
|               statement_syssym := inline_setlength;
 | |
|             end;
 | |
| 
 | |
|           in_length_x:
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               p2:=geninlinenode(l,false,p1);
 | |
|               consume(_RKLAMMER);
 | |
|               statement_syssym:=p2;
 | |
|             end;
 | |
| 
 | |
|           in_write_x,
 | |
|           in_writeln_x :
 | |
|             begin
 | |
|               if token=_LKLAMMER then
 | |
|                begin
 | |
|                  consume(_LKLAMMER);
 | |
|                  paras:=parse_paras(true,false);
 | |
|                  consume(_RKLAMMER);
 | |
|                end
 | |
|               else
 | |
|                paras:=nil;
 | |
|               p1 := geninlinenode(l,false,paras);
 | |
|               statement_syssym := p1;
 | |
|             end;
 | |
| 
 | |
|           in_str_x_string :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               paras:=parse_paras(true,false);
 | |
|               consume(_RKLAMMER);
 | |
|               p1 := geninlinenode(l,false,paras);
 | |
|               statement_syssym := p1;
 | |
|             end;
 | |
| 
 | |
|           in_val_x:
 | |
|             Begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args := true;
 | |
|               p1:= ccallparanode.create(comp_expr(true), nil);
 | |
|               consume(_COMMA);
 | |
|               p2 := ccallparanode.create(comp_expr(true),p1);
 | |
|               if (token = _COMMA) then
 | |
|                 Begin
 | |
|                   consume(_COMMA);
 | |
|                   p2 := ccallparanode.create(comp_expr(true),p2)
 | |
|                 End;
 | |
|               consume(_RKLAMMER);
 | |
|               p2 := geninlinenode(l,false,p2);
 | |
|               statement_syssym := p2;
 | |
|             End;
 | |
| 
 | |
|           in_include_x_y,
 | |
|           in_exclude_x_y :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               consume(_COMMA);
 | |
|               p2:=comp_expr(true);
 | |
|               statement_syssym:=geninlinenode(l,false,ccallparanode.create(p1,ccallparanode.create(p2,nil)));
 | |
|               consume(_RKLAMMER);
 | |
|             end;
 | |
| 
 | |
|           in_assert_x_y :
 | |
|             begin
 | |
|               consume(_LKLAMMER);
 | |
|               in_args:=true;
 | |
|               p1:=comp_expr(true);
 | |
|               if token=_COMMA then
 | |
|                begin
 | |
|                  consume(_COMMA);
 | |
|                  p2:=comp_expr(true);
 | |
|                end
 | |
|               else
 | |
|                begin
 | |
|                  { then insert an empty string }
 | |
|                  p2:=cstringconstnode.createstr('',st_default);
 | |
|                end;
 | |
|               statement_syssym:=geninlinenode(l,false,ccallparanode.create(p1,ccallparanode.create(p2,nil)));
 | |
|               consume(_RKLAMMER);
 | |
|             end;
 | |
| 
 | |
|           else
 | |
|             internalerror(15);
 | |
| 
 | |
|         end;
 | |
|         in_args:=prev_in_args;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function maybe_load_methodpointer(st:tsymtable;var p1:tnode):boolean;
 | |
|       begin
 | |
|         maybe_load_methodpointer:=false;
 | |
|         if not assigned(p1) then
 | |
|          begin
 | |
|            case st.symtabletype of
 | |
|              withsymtable :
 | |
|                begin
 | |
|                  if (st.defowner.deftype=objectdef) then
 | |
|                    p1:=tnode(twithsymtable(st).withrefnode).getcopy;
 | |
|                end;
 | |
|              objectsymtable :
 | |
|                begin
 | |
|                  p1:=load_self_node;
 | |
|                  { We are calling a member }
 | |
|                  maybe_load_methodpointer:=true;
 | |
|                end;
 | |
|            end;
 | |
|          end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     { reads the parameter for a subroutine call }
 | |
|     procedure do_proc_call(sym:tsym;st:tsymtable;obj:tobjectdef;getaddr:boolean;var again : boolean;var p1:tnode);
 | |
|       var
 | |
|          membercall,
 | |
|          prevafterassn : boolean;
 | |
|          vs : tvarsym;
 | |
|          para,p2 : tnode;
 | |
|          currpara : tparaitem;
 | |
|          aprocdef : tprocdef;
 | |
|       begin
 | |
|          prevafterassn:=afterassignment;
 | |
|          afterassignment:=false;
 | |
|          membercall:=false;
 | |
|          aprocdef:=nil;
 | |
| 
 | |
|          { when it is a call to a member we need to load the
 | |
|            methodpointer first }
 | |
|          membercall:=maybe_load_methodpointer(st,p1);
 | |
| 
 | |
|          { When we are expecting a procvar we also need
 | |
|            to get the address in some cases }
 | |
|          if assigned(getprocvardef) then
 | |
|           begin
 | |
|             if (block_type=bt_const) or
 | |
|                getaddr then
 | |
|              begin
 | |
|                aprocdef:=Tprocsym(sym).search_procdef_byprocvardef(getprocvardef);
 | |
|                getaddr:=true;
 | |
|              end
 | |
|             else
 | |
|              if (m_tp_procvar in aktmodeswitches) then
 | |
|               begin
 | |
|                 aprocdef:=Tprocsym(sym).search_procdef_byprocvardef(getprocvardef);
 | |
|                 if assigned(aprocdef) then
 | |
|                  getaddr:=true;
 | |
|               end;
 | |
|           end;
 | |
| 
 | |
|          { only need to get the address of the procedure? }
 | |
|          if getaddr then
 | |
|            begin
 | |
|              { Retrieve info which procvar to call. For tp_procvar the
 | |
|                aprocdef is already loaded above so we can reuse it }
 | |
|              if not assigned(aprocdef) and
 | |
|                 assigned(getprocvardef) then
 | |
|                aprocdef:=Tprocsym(sym).search_procdef_byprocvardef(getprocvardef);
 | |
| 
 | |
|              { generate a methodcallnode or proccallnode }
 | |
|              { we shouldn't convert things like @tcollection.load }
 | |
|              p2:=cloadnode.create_procvar(sym,aprocdef,st);
 | |
|              if assigned(p1) then
 | |
|               begin
 | |
|                 if (p1.nodetype<>typen) then
 | |
|                   tloadnode(p2).set_mp(p1)
 | |
|                 else
 | |
|                   p1.free;
 | |
|               end;
 | |
|              p1:=p2;
 | |
| 
 | |
|              { no postfix operators }
 | |
|              again:=false;
 | |
|            end
 | |
|          else
 | |
|            begin
 | |
|              para:=nil;
 | |
|              if anon_inherited then
 | |
|               begin
 | |
|                 if not assigned(current_procinfo) then
 | |
|                   internalerror(200305054);
 | |
|                 currpara:=tparaitem(current_procinfo.procdef.para.first);
 | |
|                 while assigned(currpara) do
 | |
|                  begin
 | |
|                    if not currpara.is_hidden then
 | |
|                     begin
 | |
|                       vs:=tvarsym(currpara.parasym);
 | |
|                       para:=ccallparanode.create(cloadnode.create(vs,vs.owner),para);
 | |
|                     end;
 | |
|                    currpara:=tparaitem(currpara.next);
 | |
|                  end;
 | |
|               end
 | |
|              else
 | |
|               begin
 | |
|                 if try_to_consume(_LKLAMMER) then
 | |
|                  begin
 | |
|                    para:=parse_paras(false,false);
 | |
|                    consume(_RKLAMMER);
 | |
|                  end;
 | |
|               end;
 | |
|              if assigned(obj) then
 | |
|                begin
 | |
|                  if (st.symtabletype<>objectsymtable) then
 | |
|                    internalerror(200310031);
 | |
|                  p1:=ccallnode.create(para,tprocsym(sym),obj.symtable,p1);
 | |
|                end
 | |
|              else
 | |
|                p1:=ccallnode.create(para,tprocsym(sym),st,p1);
 | |
|              { indicate if this call was generated by a member and
 | |
|                no explicit self is used, this is needed to determine
 | |
|                how to handle a destructor call (PFV) }
 | |
|              if membercall then
 | |
|                include(p1.flags,nf_member_call);
 | |
|            end;
 | |
|          afterassignment:=prevafterassn;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure handle_procvar(pv : tprocvardef;var p2 : tnode);
 | |
|       var
 | |
|         hp,hp2 : tnode;
 | |
|         hpp    : ^tnode;
 | |
|         currprocdef : tprocdef;
 | |
|       begin
 | |
|         if not assigned(pv) then
 | |
|          internalerror(200301121);
 | |
|         if (m_tp_procvar in aktmodeswitches) then
 | |
|          begin
 | |
|            hp:=p2;
 | |
|            hpp:=@p2;
 | |
|            while assigned(hp) and
 | |
|                  (hp.nodetype=typeconvn) do
 | |
|             begin
 | |
|               hp:=ttypeconvnode(hp).left;
 | |
|               { save orignal address of the old tree so we can replace the node }
 | |
|               hpp:=@hp;
 | |
|             end;
 | |
|            if (hp.nodetype=calln) and
 | |
|               { a procvar can't have parameters! }
 | |
|               not assigned(tcallnode(hp).left) then
 | |
|             begin
 | |
|               currprocdef:=tcallnode(hp).symtableprocentry.search_procdef_byprocvardef(pv);
 | |
|               if assigned(currprocdef) then
 | |
|                begin
 | |
|                  hp2:=cloadnode.create_procvar(tprocsym(tcallnode(hp).symtableprocentry),currprocdef,tcallnode(hp).symtableproc);
 | |
|                  if (po_methodpointer in pv.procoptions) then
 | |
|                    tloadnode(hp2).set_mp(tnode(tcallnode(hp).methodpointer).getcopy);
 | |
|                  hp.destroy;
 | |
|                  { replace the old callnode with the new loadnode }
 | |
|                  hpp^:=hp2;
 | |
|                end;
 | |
|             end;
 | |
|          end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     { the following procedure handles the access to a property symbol }
 | |
|     procedure handle_propertysym(sym : tsym;st : tsymtable;var p1 : tnode);
 | |
|       var
 | |
|          paras : tnode;
 | |
|          p2    : tnode;
 | |
|          membercall : boolean;
 | |
|       begin
 | |
|          paras:=nil;
 | |
|          { property parameters? read them only if the property really }
 | |
|          { has parameters                                             }
 | |
|          if (ppo_hasparameters in tpropertysym(sym).propoptions) then
 | |
|            begin
 | |
|               if token=_LECKKLAMMER then
 | |
|                 begin
 | |
|                    consume(_LECKKLAMMER);
 | |
|                    paras:=parse_paras(false,true);
 | |
|                    consume(_RECKKLAMMER);
 | |
|                 end;
 | |
|            end;
 | |
|          { indexed property }
 | |
|          if (ppo_indexed in tpropertysym(sym).propoptions) then
 | |
|            begin
 | |
|               p2:=cordconstnode.create(tpropertysym(sym).index,tpropertysym(sym).indextype,true);
 | |
|               paras:=ccallparanode.create(p2,paras);
 | |
|            end;
 | |
|          { we need only a write property if a := follows }
 | |
|          { if not(afterassignment) and not(in_args) then }
 | |
|          if token=_ASSIGNMENT then
 | |
|            begin
 | |
|               { write property: }
 | |
|               if not tpropertysym(sym).writeaccess.empty then
 | |
|                 begin
 | |
|                    case tpropertysym(sym).writeaccess.firstsym^.sym.typ of
 | |
|                      procsym :
 | |
|                        begin
 | |
|                          { generate the method call }
 | |
|                          membercall:=maybe_load_methodpointer(st,p1);
 | |
|                          p1:=ccallnode.create(paras,
 | |
|                                               tprocsym(tpropertysym(sym).writeaccess.firstsym^.sym),st,p1);
 | |
|                          if membercall then
 | |
|                            include(tcallnode(p1).flags,nf_member_call);
 | |
|                          paras:=nil;
 | |
|                          consume(_ASSIGNMENT);
 | |
|                          { read the expression }
 | |
|                          if tpropertysym(sym).proptype.def.deftype=procvardef then
 | |
|                            getprocvardef:=tprocvardef(tpropertysym(sym).proptype.def);
 | |
|                          p2:=comp_expr(true);
 | |
|                          if assigned(getprocvardef) then
 | |
|                            handle_procvar(getprocvardef,p2);
 | |
|                          tcallnode(p1).left:=ccallparanode.create(p2,tcallnode(p1).left);
 | |
|                          include(tcallnode(p1).flags,nf_isproperty);
 | |
|                          getprocvardef:=nil;
 | |
|                        end;
 | |
|                      varsym :
 | |
|                        begin
 | |
|                          { generate access code }
 | |
|                          symlist_to_node(p1,st,tpropertysym(sym).writeaccess);
 | |
|                          include(p1.flags,nf_isproperty);
 | |
|                          consume(_ASSIGNMENT);
 | |
|                          { read the expression }
 | |
|                          p2:=comp_expr(true);
 | |
|                          p1:=cassignmentnode.create(p1,p2);
 | |
|                       end
 | |
|                     else
 | |
|                       begin
 | |
|                         p1:=cerrornode.create;
 | |
|                         Message(parser_e_no_procedure_to_access_property);
 | |
|                       end;
 | |
|                   end;
 | |
|                 end
 | |
|               else
 | |
|                 begin
 | |
|                    p1:=cerrornode.create;
 | |
|                    Message(parser_e_no_procedure_to_access_property);
 | |
|                 end;
 | |
|            end
 | |
|          else
 | |
|            begin
 | |
|               { read property: }
 | |
|               if not tpropertysym(sym).readaccess.empty then
 | |
|                 begin
 | |
|                    case tpropertysym(sym).readaccess.firstsym^.sym.typ of
 | |
|                      varsym :
 | |
|                        begin
 | |
|                           { generate access code }
 | |
|                           symlist_to_node(p1,st,tpropertysym(sym).readaccess);
 | |
|                           include(p1.flags,nf_isproperty);
 | |
|                        end;
 | |
|                      procsym :
 | |
|                        begin
 | |
|                           { generate the method call }
 | |
|                           membercall:=maybe_load_methodpointer(st,p1);
 | |
|                           p1:=ccallnode.create(paras,tprocsym(tpropertysym(sym).readaccess.firstsym^.sym),st,p1);
 | |
|                           if membercall then
 | |
|                             include(tcallnode(p1).flags,nf_member_call);
 | |
|                           paras:=nil;
 | |
|                           include(p1.flags,nf_isproperty);
 | |
|                        end
 | |
|                      else
 | |
|                        begin
 | |
|                           p1:=cerrornode.create;
 | |
|                           Message(type_e_mismatch);
 | |
|                        end;
 | |
|                   end;
 | |
|                 end
 | |
|               else
 | |
|                 begin
 | |
|                    { error, no function to read property }
 | |
|                    p1:=cerrornode.create;
 | |
|                    Message(parser_e_no_procedure_to_access_property);
 | |
|                 end;
 | |
|            end;
 | |
|         { release paras if not used }
 | |
|         if assigned(paras) then
 | |
|          paras.free;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     { the ID token has to be consumed before calling this function }
 | |
|     procedure do_member_read(classh:tobjectdef;getaddr : boolean;sym : tsym;var p1 : tnode;var again : boolean;callnflags:tnodeflags);
 | |
| 
 | |
|       var
 | |
|          static_name : string;
 | |
|          isclassref : boolean;
 | |
|          srsymtable : tsymtable;
 | |
| {$ifdef CHECKINHERITEDRESULT}
 | |
|          newstatement : tstatementnode;
 | |
|          newblock     : tblocknode;
 | |
| {$endif CHECKINHERITEDRESULT}
 | |
|       begin
 | |
|          if sym=nil then
 | |
|            begin
 | |
|               { pattern is still valid unless
 | |
|               there is another ID just after the ID of sym }
 | |
|               Message1(sym_e_id_no_member,pattern);
 | |
|               p1.free;
 | |
|               p1:=cerrornode.create;
 | |
|               { try to clean up }
 | |
|               again:=false;
 | |
|            end
 | |
|          else
 | |
|            begin
 | |
|               if assigned(p1) then
 | |
|                begin
 | |
|                  if not assigned(p1.resulttype.def) then
 | |
|                   do_resulttypepass(p1);
 | |
|                  isclassref:=(p1.resulttype.def.deftype=classrefdef);
 | |
|                end
 | |
|               else
 | |
|                isclassref:=false;
 | |
| 
 | |
|               { we assume, that only procsyms and varsyms are in an object }
 | |
|               { symbol table, for classes, properties are allowed          }
 | |
|               case sym.typ of
 | |
|                  procsym:
 | |
|                    begin
 | |
|                       do_proc_call(sym,sym.owner,classh,
 | |
|                                    (getaddr and not(token in [_CARET,_POINT])),
 | |
|                                    again,p1);
 | |
|                       { add provided flags }
 | |
|                       if (p1.nodetype=calln) then
 | |
|                         p1.flags:=p1.flags+callnflags;
 | |
|                       { we need to know which procedure is called }
 | |
|                       do_resulttypepass(p1);
 | |
|                       { now we know the method that is called }
 | |
|                       if (p1.nodetype=calln) and
 | |
|                          assigned(tcallnode(p1).procdefinition) then
 | |
|                         begin
 | |
|                           { calling using classref? }
 | |
|                           if isclassref and
 | |
|                              not(po_classmethod in tcallnode(p1).procdefinition.procoptions) and
 | |
|                              not(tcallnode(p1).procdefinition.proctypeoption=potype_constructor) then
 | |
|                             Message(parser_e_only_class_methods_via_class_ref);
 | |
| {$ifdef CHECKINHERITEDRESULT}
 | |
|                            { when calling inherited constructor we need to check the return value }
 | |
|                            if (nf_inherited in callnflags) and
 | |
|                               (tcallnode(p1).procdefinition.proctypeoption=potype_constructor) then
 | |
|                              begin
 | |
|                                {
 | |
|                                  For Classes:
 | |
| 
 | |
|                                  self:=inherited constructor
 | |
|                                  if self=nil then
 | |
|                                    exit
 | |
| 
 | |
|                                  For objects:
 | |
|                                  if inherited constructor=false then
 | |
|                                    begin
 | |
|                                      self:=nil;
 | |
|                                      exit;
 | |
|                                    end;
 | |
|                                }
 | |
|                                if is_class(tprocdef(tcallnode(p1).procdefinition)._class) then
 | |
|                                  begin
 | |
|                                    newblock:=internalstatements(newstatement,true);
 | |
|                                    addstatement(newstatement,cassignmentnode.create(
 | |
|                                        ctypeconvnode.create(
 | |
|                                            load_self_pointer_node,
 | |
|                                            voidpointertype),
 | |
|                                        ctypeconvnode.create(
 | |
|                                            p1,
 | |
|                                            voidpointertype)));
 | |
|                                    addstatement(newstatement,cifnode.create(
 | |
|                                        caddnode.create(equaln,
 | |
|                                            load_self_pointer_node,
 | |
|                                            cnilnode.create),
 | |
|                                        cexitnode.create(nil),
 | |
|                                        nil));
 | |
|                                    p1:=newblock;
 | |
|                                  end
 | |
|                                else
 | |
|                                  if is_object(tprocdef(tcallnode(p1).procdefinition)._class) then
 | |
|                                    begin
 | |
|                                      newblock:=internalstatements(newstatement,true);
 | |
|                                      addstatement(newstatement,call_fail_node);
 | |
|                                      addstatement(newstatement,cexitnode.create(nil));
 | |
|                                      p1:=cifnode.create(
 | |
|                                          caddnode.create(equaln,
 | |
|                                              cordconstnode.create(0,booltype,false),
 | |
|                                              p1),
 | |
|                                          newblock,
 | |
|                                          nil);
 | |
|                                    end
 | |
|                                  else
 | |
|                                    internalerror(200305133);
 | |
|                              end;
 | |
| {$endif CHECKINHERITEDRESULT}
 | |
|                            do_resulttypepass(p1);
 | |
|                         end;
 | |
|                    end;
 | |
|                  varsym:
 | |
|                    begin
 | |
|                       if (sp_static in sym.symoptions) then
 | |
|                         begin
 | |
|                            static_name:=lower(sym.owner.name^)+'_'+sym.name;
 | |
|                            searchsym(static_name,sym,srsymtable);
 | |
|                            check_hints(sym);
 | |
|                            p1.free;
 | |
|                            p1:=cloadnode.create(sym,srsymtable);
 | |
|                         end
 | |
|                       else
 | |
|                         begin
 | |
|                           if isclassref then
 | |
|                             Message(parser_e_only_class_methods_via_class_ref);
 | |
|                           p1:=csubscriptnode.create(sym,p1);
 | |
|                         end;
 | |
|                    end;
 | |
|                  propertysym:
 | |
|                    begin
 | |
|                       if isclassref then
 | |
|                         Message(parser_e_only_class_methods_via_class_ref);
 | |
|                       handle_propertysym(sym,sym.owner,p1);
 | |
|                    end;
 | |
|                  else internalerror(16);
 | |
|               end;
 | |
|            end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
| {****************************************************************************
 | |
|                                Factor
 | |
| ****************************************************************************}
 | |
| {$ifdef fpc}
 | |
|   {$maxfpuregisters 0}
 | |
| {$endif fpc}
 | |
| 
 | |
|     function factor(getaddr : boolean) : tnode;
 | |
| 
 | |
|          {---------------------------------------------
 | |
|                          Factor_read_id
 | |
|          ---------------------------------------------}
 | |
| 
 | |
|        procedure factor_read_id(var p1:tnode;var again:boolean);
 | |
|          var
 | |
|            pc    : pchar;
 | |
|            len   : longint;
 | |
|            srsym : tsym;
 | |
|            possible_error : boolean;
 | |
|            srsymtable : tsymtable;
 | |
|            storesymtablestack : tsymtable;
 | |
|            htype : ttype;
 | |
|            static_name : string;
 | |
|          begin
 | |
|            { allow post fix operators }
 | |
|            again:=true;
 | |
|            consume_sym(srsym,srsymtable);
 | |
| 
 | |
|            { Access to funcret or need to call the function? }
 | |
|            if (srsym.typ in [absolutesym,varsym]) and
 | |
|               (vo_is_funcret in tvarsym(srsym).varoptions) and
 | |
|               (
 | |
|                (token=_LKLAMMER) or
 | |
|                (not(m_fpc in aktmodeswitches) and
 | |
|                 (afterassignment or in_args) and
 | |
|                 not(vo_is_result in tvarsym(srsym).varoptions))
 | |
|               ) then
 | |
|             begin
 | |
|               storesymtablestack:=symtablestack;
 | |
|               symtablestack:=srsym.owner.next;
 | |
|               searchsym(srsym.name,srsym,srsymtable);
 | |
|               if not assigned(srsym) then
 | |
|                srsym:=generrorsym;
 | |
|               if (srsym.typ<>procsym) then
 | |
|                Message(cg_e_illegal_expression);
 | |
|               symtablestack:=storesymtablestack;
 | |
|             end;
 | |
| 
 | |
|             begin
 | |
|               { check semantics of private }
 | |
|               if (srsym.typ in [propertysym,procsym,varsym]) and
 | |
|                  (srsym.owner.symtabletype=objectsymtable) then
 | |
|                begin
 | |
|                  if (sp_private in srsym.symoptions) and
 | |
|                     (tobjectdef(srsym.owner.defowner).owner.symtabletype=globalsymtable) and
 | |
|                     (tobjectdef(srsym.owner.defowner).owner.unitid<>0) then
 | |
|                    Message(parser_e_cant_access_private_member);
 | |
|                end;
 | |
|               case srsym.typ of
 | |
|                 absolutesym :
 | |
|                   begin
 | |
|                     if (tabsolutesym(srsym).abstyp=tovar) then
 | |
|                       begin
 | |
|                         p1:=nil;
 | |
|                         symlist_to_node(p1,nil,tabsolutesym(srsym).ref);
 | |
|                         p1:=ctypeconvnode.create(p1,tabsolutesym(srsym).vartype);
 | |
|                         include(p1.flags,nf_absolute);
 | |
|                       end
 | |
|                     else
 | |
|                       p1:=cloadnode.create(srsym,srsymtable);
 | |
|                   end;
 | |
| 
 | |
|                 varsym :
 | |
|                   begin
 | |
|                     if (sp_static in srsym.symoptions) then
 | |
|                      begin
 | |
|                        static_name:=lower(srsym.owner.name^)+'_'+srsym.name;
 | |
|                        searchsym(static_name,srsym,srsymtable);
 | |
|                        check_hints(srsym);
 | |
|                      end
 | |
|                     else
 | |
|                      begin
 | |
|                        { are we in a class method, we check here the
 | |
|                          srsymtable, because a field in another object
 | |
|                          also has objectsymtable. And withsymtable is
 | |
|                          not possible for self in class methods (PFV) }
 | |
|                        if (srsymtable.symtabletype=objectsymtable) and
 | |
|                           assigned(current_procinfo) and
 | |
|                           (po_classmethod in current_procinfo.procdef.procoptions) then
 | |
|                          Message(parser_e_only_class_methods);
 | |
|                      end;
 | |
| 
 | |
|                     case srsymtable.symtabletype of
 | |
|                       objectsymtable :
 | |
|                         p1:=csubscriptnode.create(srsym,load_self_node);
 | |
|                       withsymtable :
 | |
|                         p1:=csubscriptnode.create(srsym,tnode(twithsymtable(srsymtable).withrefnode).getcopy);
 | |
|                       else
 | |
|                         p1:=cloadnode.create(srsym,srsymtable);
 | |
|                     end;
 | |
|                   end;
 | |
| 
 | |
|                 typedconstsym :
 | |
|                   begin
 | |
|                     p1:=cloadnode.create(srsym,srsymtable);
 | |
|                   end;
 | |
| 
 | |
|                 syssym :
 | |
|                   begin
 | |
|                     p1:=statement_syssym(tsyssym(srsym).number);
 | |
|                   end;
 | |
| 
 | |
|                 typesym :
 | |
|                   begin
 | |
|                     htype.setsym(srsym);
 | |
|                     if not assigned(htype.def) then
 | |
|                      begin
 | |
|                        again:=false;
 | |
|                      end
 | |
|                     else
 | |
|                      begin
 | |
|                        if token=_LKLAMMER then
 | |
|                         begin
 | |
|                           consume(_LKLAMMER);
 | |
|                           p1:=comp_expr(true);
 | |
|                           consume(_RKLAMMER);
 | |
|                           p1:=ctypeconvnode.create_explicit(p1,htype);
 | |
|                         end
 | |
|                        else { not LKLAMMER }
 | |
|                         if (token=_POINT) and
 | |
|                            is_object(htype.def) then
 | |
|                          begin
 | |
|                            consume(_POINT);
 | |
|                            if assigned(current_procinfo) and
 | |
|                               assigned(current_procinfo.procdef._class) and
 | |
|                               not(getaddr) then
 | |
|                             begin
 | |
|                               if current_procinfo.procdef._class.is_related(tobjectdef(htype.def)) then
 | |
|                                begin
 | |
|                                  p1:=ctypenode.create(htype);
 | |
|                                  { search also in inherited methods }
 | |
|                                  srsym:=searchsym_in_class(tobjectdef(htype.def),pattern);
 | |
|                                  check_hints(srsym);
 | |
|                                  consume(_ID);
 | |
|                                  do_member_read(tobjectdef(htype.def),false,srsym,p1,again,[]);
 | |
|                                end
 | |
|                               else
 | |
|                                begin
 | |
|                                  Message(parser_e_no_super_class);
 | |
|                                  again:=false;
 | |
|                                end;
 | |
|                             end
 | |
|                            else
 | |
|                             begin
 | |
|                               { allows @TObject.Load }
 | |
|                               { also allows static methods and variables }
 | |
|                               p1:=ctypenode.create(htype);
 | |
|                               { TP allows also @TMenu.Load if Load is only }
 | |
|                               { defined in an anchestor class              }
 | |
|                               srsym:=search_class_member(tobjectdef(htype.def),pattern);
 | |
|                               check_hints(srsym);
 | |
|                               if not assigned(srsym) then
 | |
|                                Message1(sym_e_id_no_member,pattern)
 | |
|                               else if not(getaddr) and not(sp_static in srsym.symoptions) then
 | |
|                                Message(sym_e_only_static_in_static)
 | |
|                               else
 | |
|                                begin
 | |
|                                  consume(_ID);
 | |
|                                  do_member_read(tobjectdef(htype.def),getaddr,srsym,p1,again,[]);
 | |
|                                end;
 | |
|                             end;
 | |
|                          end
 | |
|                        else
 | |
|                         begin
 | |
|                           { class reference ? }
 | |
|                           if is_class(htype.def) then
 | |
|                            begin
 | |
|                              if getaddr and (token=_POINT) then
 | |
|                               begin
 | |
|                                 consume(_POINT);
 | |
|                                 { allows @Object.Method }
 | |
|                                 { also allows static methods and variables }
 | |
|                                 p1:=ctypenode.create(htype);
 | |
|                                 { TP allows also @TMenu.Load if Load is only }
 | |
|                                 { defined in an anchestor class              }
 | |
|                                 srsym:=search_class_member(tobjectdef(htype.def),pattern);
 | |
|                                 check_hints(srsym);
 | |
|                                 if not assigned(srsym) then
 | |
|                                  Message1(sym_e_id_no_member,pattern)
 | |
|                                 else
 | |
|                                  begin
 | |
|                                    consume(_ID);
 | |
|                                    do_member_read(tobjectdef(htype.def),getaddr,srsym,p1,again,[]);
 | |
|                                  end;
 | |
|                               end
 | |
|                              else
 | |
|                               begin
 | |
|                                 p1:=ctypenode.create(htype);
 | |
|                                 { For a type block we simply return only
 | |
|                                   the type. For all other blocks we return
 | |
|                                   a loadvmt node }
 | |
|                                 if (block_type<>bt_type) then
 | |
|                                   p1:=cloadvmtaddrnode.create(p1);
 | |
|                               end;
 | |
|                            end
 | |
|                           else
 | |
|                            p1:=ctypenode.create(htype);
 | |
|                         end;
 | |
|                      end;
 | |
|                   end;
 | |
| 
 | |
|                 enumsym :
 | |
|                   begin
 | |
|                     p1:=genenumnode(tenumsym(srsym));
 | |
|                   end;
 | |
| 
 | |
|                 constsym :
 | |
|                   begin
 | |
|                     case tconstsym(srsym).consttyp of
 | |
|                       constint :
 | |
|                         begin
 | |
|                           { do a very dirty trick to bootstrap this code }
 | |
|                           if (tconstsym(srsym).value.valueord>=-(int64(2147483647)+int64(1))) and
 | |
|                              (tconstsym(srsym).value.valueord<=2147483647) then
 | |
|                            p1:=cordconstnode.create(tconstsym(srsym).value.valueord,s32bittype,true)
 | |
|                           else if (tconstsym(srsym).value.valueord > maxlongint) and
 | |
|                                   (tconstsym(srsym).value.valueord <= int64(maxlongint)+int64(maxlongint)+1) then
 | |
|                            p1:=cordconstnode.create(tconstsym(srsym).value.valueord,u32bittype,true)
 | |
|                           else
 | |
|                            p1:=cordconstnode.create(tconstsym(srsym).value.valueord,cs64bittype,true);
 | |
|                         end;
 | |
|                       conststring :
 | |
|                         begin
 | |
|                           len:=tconstsym(srsym).value.len;
 | |
|                           if not(cs_ansistrings in aktlocalswitches) and (len>255) then
 | |
|                            len:=255;
 | |
|                           getmem(pc,len+1);
 | |
|                           move(pchar(tconstsym(srsym).value.valueptr)^,pc^,len);
 | |
|                           pc[len]:=#0;
 | |
|                           p1:=cstringconstnode.createpchar(pc,len);
 | |
|                         end;
 | |
|                       constchar :
 | |
|                         p1:=cordconstnode.create(tconstsym(srsym).value.valueord,cchartype,true);
 | |
|                       constreal :
 | |
|                         p1:=crealconstnode.create(pbestreal(tconstsym(srsym).value.valueptr)^,pbestrealtype^);
 | |
|                       constbool :
 | |
|                         p1:=cordconstnode.create(tconstsym(srsym).value.valueord,booltype,true);
 | |
|                       constset :
 | |
|                         p1:=csetconstnode.create(pconstset(tconstsym(srsym).value.valueptr),tconstsym(srsym).consttype);
 | |
|                       constord :
 | |
|                         p1:=cordconstnode.create(tconstsym(srsym).value.valueord,tconstsym(srsym).consttype,true);
 | |
|                       constpointer :
 | |
|                         p1:=cpointerconstnode.create(tconstsym(srsym).value.valueordptr,tconstsym(srsym).consttype);
 | |
|                       constnil :
 | |
|                         p1:=cnilnode.create;
 | |
|                       constresourcestring:
 | |
|                         begin
 | |
|                           p1:=cloadnode.create(srsym,srsymtable);
 | |
|                           do_resulttypepass(p1);
 | |
|                           p1.resulttype:=cansistringtype;
 | |
|                         end;
 | |
|                       constguid :
 | |
|                         p1:=cguidconstnode.create(pguid(tconstsym(srsym).value.valueptr)^);
 | |
|                     end;
 | |
|                   end;
 | |
| 
 | |
|                 procsym :
 | |
|                   begin
 | |
|                     { are we in a class method ? }
 | |
|                     possible_error:=(srsymtable.symtabletype<>withsymtable) and
 | |
|                                     (srsym.owner.symtabletype=objectsymtable) and
 | |
|                                     not(is_interface(tdef(srsym.owner.defowner))) and
 | |
|                                     assigned(current_procinfo) and
 | |
|                                     (po_classmethod in current_procinfo.procdef.procoptions);
 | |
|                     do_proc_call(srsym,srsymtable,nil,
 | |
|                                  (getaddr and not(token in [_CARET,_POINT])),
 | |
|                                  again,p1);
 | |
|                     { we need to know which procedure is called }
 | |
|                     if possible_error then
 | |
|                      begin
 | |
|                        do_resulttypepass(p1);
 | |
|                        if not(tcallnode(p1).procdefinition.proctypeoption=potype_constructor) and
 | |
|                           not(po_classmethod in tcallnode(p1).procdefinition.procoptions) then
 | |
|                          Message(parser_e_only_class_methods);
 | |
|                      end;
 | |
|                   end;
 | |
| 
 | |
|                 propertysym :
 | |
|                   begin
 | |
|                     { access to property in a method }
 | |
|                     { are we in a class method ? }
 | |
|                     if (srsymtable.symtabletype=objectsymtable) and
 | |
|                        assigned(current_procinfo) and
 | |
|                        (po_classmethod in current_procinfo.procdef.procoptions) then
 | |
|                      Message(parser_e_only_class_methods);
 | |
|                     { no method pointer }
 | |
|                     p1:=nil;
 | |
|                     handle_propertysym(srsym,srsymtable,p1);
 | |
|                   end;
 | |
| 
 | |
|                 labelsym :
 | |
|                   begin
 | |
|                     consume(_COLON);
 | |
|                     if tlabelsym(srsym).defined then
 | |
|                      Message(sym_e_label_already_defined);
 | |
|                     tlabelsym(srsym).defined:=true;
 | |
|                     p1:=clabelnode.create(tlabelsym(srsym),nil);
 | |
|                   end;
 | |
| 
 | |
|                 errorsym :
 | |
|                   begin
 | |
|                     p1:=cerrornode.create;
 | |
|                     if token=_LKLAMMER then
 | |
|                      begin
 | |
|                        consume(_LKLAMMER);
 | |
|                        parse_paras(false,false);
 | |
|                        consume(_RKLAMMER);
 | |
|                      end;
 | |
|                   end;
 | |
| 
 | |
|                 else
 | |
|                   begin
 | |
|                     p1:=cerrornode.create;
 | |
|                     Message(cg_e_illegal_expression);
 | |
|                   end;
 | |
|               end; { end case }
 | |
|             end;
 | |
|          end;
 | |
| 
 | |
|          {---------------------------------------------
 | |
|                          Factor_Read_Set
 | |
|          ---------------------------------------------}
 | |
| 
 | |
|          { Read a set between [] }
 | |
|          function factor_read_set:tnode;
 | |
|          var
 | |
|            p1,p2 : tnode;
 | |
|            lastp,
 | |
|            buildp : tarrayconstructornode;
 | |
|          begin
 | |
|            buildp:=nil;
 | |
|          { be sure that a least one arrayconstructn is used, also for an
 | |
|            empty [] }
 | |
|            if token=_RECKKLAMMER then
 | |
|             buildp:=carrayconstructornode.create(nil,buildp)
 | |
|            else
 | |
|             begin
 | |
|               while true do
 | |
|                begin
 | |
|                  p1:=comp_expr(true);
 | |
|                  if token=_POINTPOINT then
 | |
|                   begin
 | |
|                     consume(_POINTPOINT);
 | |
|                     p2:=comp_expr(true);
 | |
|                     p1:=carrayconstructorrangenode.create(p1,p2);
 | |
|                   end;
 | |
|                { insert at the end of the tree, to get the correct order }
 | |
|                  if not assigned(buildp) then
 | |
|                   begin
 | |
|                     buildp:=carrayconstructornode.create(p1,nil);
 | |
|                     lastp:=buildp;
 | |
|                   end
 | |
|                  else
 | |
|                   begin
 | |
|                     lastp.right:=carrayconstructornode.create(p1,nil);
 | |
|                     lastp:=tarrayconstructornode(lastp.right);
 | |
|                   end;
 | |
|                { there could be more elements }
 | |
|                  if token=_COMMA then
 | |
|                    consume(_COMMA)
 | |
|                  else
 | |
|                    break;
 | |
|                end;
 | |
|             end;
 | |
|            factor_read_set:=buildp;
 | |
|          end;
 | |
| 
 | |
| 
 | |
|          {---------------------------------------------
 | |
|                         PostFixOperators
 | |
|          ---------------------------------------------}
 | |
| 
 | |
|       procedure postfixoperators(var p1:tnode;var again:boolean);
 | |
| 
 | |
|         { tries to avoid syntax errors after invalid qualifiers }
 | |
|         procedure recoverconsume_postfixops;
 | |
| 
 | |
|           begin
 | |
|              while true do
 | |
|                begin
 | |
|                   case token of
 | |
|                      _CARET:
 | |
|                         consume(_CARET);
 | |
|                      _POINT:
 | |
|                         begin
 | |
|                            consume(_POINT);
 | |
|                            if token=_ID then
 | |
|                              consume(_ID);
 | |
|                         end;
 | |
|                      _LECKKLAMMER:
 | |
|                         begin
 | |
|                            consume(_LECKKLAMMER);
 | |
|                            repeat
 | |
|                              comp_expr(true);
 | |
|                              if token=_COMMA then
 | |
|                                consume(_COMMA)
 | |
|                              else
 | |
|                                break;
 | |
|                            until false;
 | |
|                            consume(_RECKKLAMMER);
 | |
|                         end
 | |
|                      else
 | |
|                         break;
 | |
|                   end;
 | |
|                end;
 | |
|           end;
 | |
| 
 | |
|         var
 | |
|            store_static : boolean;
 | |
|            protsym  : tpropertysym;
 | |
|            p2,p3 : tnode;
 | |
|            hsym  : tsym;
 | |
|            classh : tobjectdef;
 | |
|         begin
 | |
|           again:=true;
 | |
|           while again do
 | |
|            begin
 | |
|              { we need the resulttype }
 | |
|              do_resulttypepass(p1);
 | |
|              if codegenerror then
 | |
|               begin
 | |
|                 recoverconsume_postfixops;
 | |
|                 exit;
 | |
|               end;
 | |
|              { handle token }
 | |
|              case token of
 | |
|                _CARET:
 | |
|                   begin
 | |
|                     consume(_CARET);
 | |
|                     if (p1.resulttype.def.deftype<>pointerdef) then
 | |
|                       begin
 | |
|                          { ^ as binary operator is a problem!!!! (FK) }
 | |
|                          again:=false;
 | |
|                          Message(cg_e_invalid_qualifier);
 | |
|                          recoverconsume_postfixops;
 | |
|                          p1.destroy;
 | |
|                          p1:=cerrornode.create;
 | |
|                       end
 | |
|                     else
 | |
|                       begin
 | |
|                          p1:=cderefnode.create(p1);
 | |
|                       end;
 | |
|                   end;
 | |
| 
 | |
|                _LECKKLAMMER:
 | |
|                   begin
 | |
|                     if is_class_or_interface(p1.resulttype.def) then
 | |
|                       begin
 | |
|                         { default property }
 | |
|                         protsym:=search_default_property(tobjectdef(p1.resulttype.def));
 | |
|                         if not(assigned(protsym)) then
 | |
|                           begin
 | |
|                              p1.destroy;
 | |
|                              p1:=cerrornode.create;
 | |
|                              again:=false;
 | |
|                              message(parser_e_no_default_property_available);
 | |
|                           end
 | |
|                         else
 | |
|                           begin
 | |
|                             { The property symbol is referenced indirect }
 | |
|                             inc(protsym.refs);
 | |
|                             handle_propertysym(protsym,protsym.owner,p1);
 | |
|                           end;
 | |
|                       end
 | |
|                     else
 | |
|                       begin
 | |
|                         consume(_LECKKLAMMER);
 | |
|                         repeat
 | |
|                           case p1.resulttype.def.deftype of
 | |
|                             pointerdef:
 | |
|                               begin
 | |
|                                  { support delphi autoderef }
 | |
|                                  if (tpointerdef(p1.resulttype.def).pointertype.def.deftype=arraydef) and
 | |
|                                     (m_autoderef in aktmodeswitches) then
 | |
|                                   begin
 | |
|                                     p1:=cderefnode.create(p1);
 | |
|                                   end;
 | |
|                                  p2:=comp_expr(true);
 | |
|                                  p1:=cvecnode.create(p1,p2);
 | |
|                               end;
 | |
|                             stringdef :
 | |
|                               begin
 | |
|                                 p2:=comp_expr(true);
 | |
|                                 p1:=cvecnode.create(p1,p2);
 | |
|                               end;
 | |
|                             arraydef :
 | |
|                               begin
 | |
|                                 p2:=comp_expr(true);
 | |
|                               { support SEG:OFS for go32v2 Mem[] }
 | |
|                                 if (target_info.system in [system_i386_go32v2,system_i386_watcom]) and
 | |
|                                    (p1.nodetype=loadn) and
 | |
|                                    assigned(tloadnode(p1).symtableentry) and
 | |
|                                    assigned(tloadnode(p1).symtableentry.owner.name) and
 | |
|                                    (tloadnode(p1).symtableentry.owner.name^='SYSTEM') and
 | |
|                                    ((tloadnode(p1).symtableentry.name='MEM') or
 | |
|                                     (tloadnode(p1).symtableentry.name='MEMW') or
 | |
|                                     (tloadnode(p1).symtableentry.name='MEML')) then
 | |
|                                   begin
 | |
|                                     if (token=_COLON) then
 | |
|                                      begin
 | |
|                                        consume(_COLON);
 | |
|                                        p3:=caddnode.create(muln,cordconstnode.create($10,s32bittype,false),p2);
 | |
|                                        p2:=comp_expr(true);
 | |
|                                        p2:=caddnode.create(addn,p2,p3);
 | |
|                                        p1:=cvecnode.create(p1,p2);
 | |
|                                        include(tvecnode(p1).flags,nf_memseg);
 | |
|                                        include(tvecnode(p1).flags,nf_memindex);
 | |
|                                      end
 | |
|                                     else
 | |
|                                      begin
 | |
|                                        p1:=cvecnode.create(p1,p2);
 | |
|                                        include(tvecnode(p1).flags,nf_memindex);
 | |
|                                      end;
 | |
|                                   end
 | |
|                                 else
 | |
|                                   p1:=cvecnode.create(p1,p2);
 | |
|                               end;
 | |
|                             else
 | |
|                               begin
 | |
|                                 Message(cg_e_invalid_qualifier);
 | |
|                                 p1.destroy;
 | |
|                                 p1:=cerrornode.create;
 | |
|                                 comp_expr(true);
 | |
|                                 again:=false;
 | |
|                               end;
 | |
|                           end;
 | |
|                           do_resulttypepass(p1);
 | |
|                           if token=_COMMA then
 | |
|                             consume(_COMMA)
 | |
|                           else
 | |
|                             break;
 | |
|                         until false;
 | |
|                         consume(_RECKKLAMMER);
 | |
|                       end;
 | |
|                   end;
 | |
| 
 | |
|                _POINT :
 | |
|                   begin
 | |
|                     consume(_POINT);
 | |
|                     if (p1.resulttype.def.deftype=pointerdef) and
 | |
|                        (m_autoderef in aktmodeswitches) then
 | |
|                       begin
 | |
|                         p1:=cderefnode.create(p1);
 | |
|                         do_resulttypepass(p1);
 | |
|                       end;
 | |
|                     case p1.resulttype.def.deftype of
 | |
|                        recorddef:
 | |
|                          begin
 | |
|                             hsym:=tsym(trecorddef(p1.resulttype.def).symtable.search(pattern));
 | |
|                             check_hints(hsym);
 | |
|                             if assigned(hsym) and
 | |
|                                (hsym.typ=varsym) then
 | |
|                               p1:=csubscriptnode.create(hsym,p1)
 | |
|                             else
 | |
|                               begin
 | |
|                                 Message1(sym_e_illegal_field,pattern);
 | |
|                                 p1.destroy;
 | |
|                                 p1:=cerrornode.create;
 | |
|                               end;
 | |
|                             consume(_ID);
 | |
|                           end;
 | |
|                         variantdef:
 | |
|                           begin
 | |
|                           end;
 | |
|                         classrefdef:
 | |
|                           begin
 | |
|                              classh:=tobjectdef(tclassrefdef(p1.resulttype.def).pointertype.def);
 | |
|                              hsym:=searchsym_in_class(classh,pattern);
 | |
|                              check_hints(hsym);
 | |
|                              if hsym=nil then
 | |
|                               begin
 | |
|                                 Message1(sym_e_id_no_member,pattern);
 | |
|                                 p1.destroy;
 | |
|                                 p1:=cerrornode.create;
 | |
|                                 { try to clean up }
 | |
|                                 consume(_ID);
 | |
|                               end
 | |
|                              else
 | |
|                               begin
 | |
|                                 consume(_ID);
 | |
|                                 do_member_read(classh,getaddr,hsym,p1,again,[]);
 | |
|                               end;
 | |
|                            end;
 | |
| 
 | |
|                          objectdef:
 | |
|                            begin
 | |
|                               store_static:=allow_only_static;
 | |
|                               allow_only_static:=false;
 | |
|                               classh:=tobjectdef(p1.resulttype.def);
 | |
|                               hsym:=searchsym_in_class(classh,pattern);
 | |
|                               check_hints(hsym);
 | |
|                               allow_only_static:=store_static;
 | |
|                               if hsym=nil then
 | |
|                                 begin
 | |
|                                    Message1(sym_e_id_no_member,pattern);
 | |
|                                    p1.destroy;
 | |
|                                    p1:=cerrornode.create;
 | |
|                                    { try to clean up }
 | |
|                                    consume(_ID);
 | |
|                                 end
 | |
|                               else
 | |
|                                 begin
 | |
|                                    consume(_ID);
 | |
|                                    do_member_read(classh,getaddr,hsym,p1,again,[]);
 | |
|                                 end;
 | |
|                            end;
 | |
| 
 | |
|                          pointerdef:
 | |
|                            begin
 | |
|                              Message(cg_e_invalid_qualifier);
 | |
|                              if tpointerdef(p1.resulttype.def).pointertype.def.deftype in [recorddef,objectdef,classrefdef] then
 | |
|                               Message(parser_h_maybe_deref_caret_missing);
 | |
|                            end;
 | |
| 
 | |
|                          else
 | |
|                            begin
 | |
|                              Message(cg_e_invalid_qualifier);
 | |
|                              p1.destroy;
 | |
|                              p1:=cerrornode.create;
 | |
|                              consume(_ID);
 | |
|                            end;
 | |
|                     end;
 | |
|                   end;
 | |
| 
 | |
|                else
 | |
|                  begin
 | |
|                  { is this a procedure variable ? }
 | |
|                    if assigned(p1.resulttype.def) then
 | |
|                     begin
 | |
|                       if (p1.resulttype.def.deftype=procvardef) then
 | |
|                        begin
 | |
|                          if assigned(getprocvardef) and
 | |
|                             equal_defs(p1.resulttype.def,getprocvardef) then
 | |
|                            again:=false
 | |
|                          else
 | |
|                            if (token=_LKLAMMER) or
 | |
|                               ((tprocvardef(p1.resulttype.def).maxparacount=0) and
 | |
|                                (not((token in [_ASSIGNMENT,_UNEQUAL,_EQUAL]))) and
 | |
|                                (not afterassignment) and
 | |
|                                (not in_args)) then
 | |
|                              begin
 | |
|                                 if try_to_consume(_LKLAMMER) then
 | |
|                                   begin
 | |
|                                      p2:=parse_paras(false,false);
 | |
|                                      consume(_RKLAMMER);
 | |
|                                   end
 | |
|                                 else
 | |
|                                   p2:=nil;
 | |
|                                 p1:=ccallnode.create_procvar(p2,p1);
 | |
|                                 { proc():= is never possible }
 | |
|                                 if token=_ASSIGNMENT then
 | |
|                                  begin
 | |
|                                    Message(cg_e_illegal_expression);
 | |
|                                    p1.free;
 | |
|                                    p1:=cerrornode.create;
 | |
|                                    again:=false;
 | |
|                                  end;
 | |
|                              end
 | |
|                          else
 | |
|                            again:=false;
 | |
|                        end
 | |
|                       else
 | |
|                        again:=false;
 | |
|                     end
 | |
|                    else
 | |
|                     again:=false;
 | |
|                   end;
 | |
|              end;
 | |
|            end; { while again }
 | |
|         end;
 | |
| 
 | |
| 
 | |
|       {---------------------------------------------
 | |
|                       Factor (Main)
 | |
|       ---------------------------------------------}
 | |
| 
 | |
|       var
 | |
|          l        : longint;
 | |
|          card     : cardinal;
 | |
|          ic       : TConstExprInt;
 | |
|          oldp1,
 | |
|          p1       : tnode;
 | |
|          code     : integer;
 | |
|          again    : boolean;
 | |
|          sym      : tsym;
 | |
|          pd       : tprocdef;
 | |
|          classh   : tobjectdef;
 | |
|          d        : bestreal;
 | |
|          hs       : string;
 | |
|          htype    : ttype;
 | |
|          filepos  : tfileposinfo;
 | |
| 
 | |
|          {---------------------------------------------
 | |
|                            Helpers
 | |
|          ---------------------------------------------}
 | |
| 
 | |
|         procedure check_tokenpos;
 | |
|         begin
 | |
|           if (p1<>oldp1) then
 | |
|            begin
 | |
|              if assigned(p1) then
 | |
|               p1.set_tree_filepos(filepos);
 | |
|              oldp1:=p1;
 | |
|              filepos:=akttokenpos;
 | |
|            end;
 | |
|         end;
 | |
| 
 | |
|       begin
 | |
|         oldp1:=nil;
 | |
|         p1:=nil;
 | |
|         filepos:=akttokenpos;
 | |
|         again:=false;
 | |
|         if token=_ID then
 | |
|          begin
 | |
|            again:=true;
 | |
|            { Handle references to self }
 | |
|            if (idtoken=_SELF) and
 | |
|               not(block_type in [bt_const,bt_type]) and
 | |
|               assigned(current_procinfo) and
 | |
|               assigned(current_procinfo.procdef._class) then
 | |
|              begin
 | |
|                p1:=load_self_node;
 | |
|                consume(_ID);
 | |
|                again:=true;
 | |
|              end
 | |
|            else
 | |
|              factor_read_id(p1,again);
 | |
|            if again then
 | |
|             begin
 | |
|               check_tokenpos;
 | |
| 
 | |
|               { handle post fix operators }
 | |
|               postfixoperators(p1,again);
 | |
|             end;
 | |
|          end
 | |
|         else
 | |
|          case token of
 | |
|            _INHERITED :
 | |
|              begin
 | |
|                again:=true;
 | |
|                consume(_INHERITED);
 | |
|                if assigned(current_procinfo) and
 | |
|                   assigned(current_procinfo.procdef._class) then
 | |
|                 begin
 | |
|                   classh:=current_procinfo.procdef._class.childof;
 | |
|                   { if inherited; only then we need the method with
 | |
|                     the same name }
 | |
|                   if token in endtokens then
 | |
|                    begin
 | |
|                      hs:=current_procinfo.procdef.procsym.name;
 | |
|                      anon_inherited:=true;
 | |
|                      { For message methods we need to search using the message
 | |
|                        number or string }
 | |
|                      pd:=tprocsym(current_procinfo.procdef.procsym).first_procdef;
 | |
|                      if (po_msgint in pd.procoptions) then
 | |
|                       sym:=searchsym_in_class_by_msgint(classh,pd.messageinf.i)
 | |
|                      else
 | |
|                       if (po_msgstr in pd.procoptions) then
 | |
|                        sym:=searchsym_in_class_by_msgstr(classh,pd.messageinf.str)
 | |
|                      else
 | |
|                       sym:=searchsym_in_class(classh,hs);
 | |
|                    end
 | |
|                   else
 | |
|                    begin
 | |
|                      hs:=pattern;
 | |
|                      consume(_ID);
 | |
|                      anon_inherited:=false;
 | |
|                      sym:=searchsym_in_class(classh,hs);
 | |
|                    end;
 | |
|                   if assigned(sym) then
 | |
|                    begin
 | |
|                      check_hints(sym);
 | |
|                      { load the procdef from the inherited class and
 | |
|                        not from self }
 | |
|                      if sym.typ=procsym then
 | |
|                       begin
 | |
|                         htype.setdef(classh);
 | |
|                         if (po_classmethod in current_procinfo.procdef.procoptions) or
 | |
|                            (po_staticmethod in current_procinfo.procdef.procoptions) then
 | |
|                           htype.setdef(tclassrefdef.create(htype));
 | |
|                         p1:=ctypenode.create(htype);
 | |
|                       end;
 | |
|                      do_member_read(classh,false,sym,p1,again,[nf_inherited,nf_anon_inherited]);
 | |
|                    end
 | |
|                   else
 | |
|                    begin
 | |
|                      if anon_inherited then
 | |
|                       begin
 | |
|                         { For message methods we need to call DefaultHandler }
 | |
|                         if (po_msgint in pd.procoptions) or
 | |
|                            (po_msgstr in pd.procoptions) then
 | |
|                           begin
 | |
|                             sym:=searchsym_in_class(classh,'DEFAULTHANDLER');
 | |
|                             if not assigned(sym) or
 | |
|                                (sym.typ<>procsym) then
 | |
|                               internalerror(200303171);
 | |
|                             p1:=nil;
 | |
|                             do_proc_call(sym,sym.owner,classh,false,again,p1);
 | |
|                           end
 | |
|                         else
 | |
|                           begin
 | |
|                             { we need to ignore the inherited; }
 | |
|                             p1:=cnothingnode.create;
 | |
|                           end;
 | |
|                       end
 | |
|                      else
 | |
|                       begin
 | |
|                         Message1(sym_e_id_no_member,hs);
 | |
|                         p1:=cerrornode.create;
 | |
|                       end;
 | |
|                      again:=false;
 | |
|                    end;
 | |
|                   { turn auto inheriting off }
 | |
|                   anon_inherited:=false;
 | |
|                 end
 | |
|                else
 | |
|                  begin
 | |
|                     Message(parser_e_generic_methods_only_in_methods);
 | |
|                     again:=false;
 | |
|                     p1:=cerrornode.create;
 | |
|                  end;
 | |
|                postfixoperators(p1,again);
 | |
|              end;
 | |
| 
 | |
|            _INTCONST :
 | |
|              begin
 | |
|                { try cardinal first }
 | |
|                val(pattern,card,code);
 | |
|                if code<>0 then
 | |
|                  begin
 | |
|                    { then longint }
 | |
|                    valint(pattern,l,code);
 | |
|                    if code <> 0 then
 | |
|                      begin
 | |
|                        { then int64 }
 | |
|                        val(pattern,ic,code);
 | |
|                        if code<>0 then
 | |
|                          begin
 | |
|                             {finally float }
 | |
|                             val(pattern,d,code);
 | |
|                             if code<>0 then
 | |
|                              begin
 | |
|                                 Message(cg_e_invalid_integer);
 | |
|                                 consume(_INTCONST);
 | |
|                                 l:=1;
 | |
|                                 p1:=cordconstnode.create(l,s32bittype,true);
 | |
|                              end
 | |
|                             else
 | |
|                              begin
 | |
|                                 consume(_INTCONST);
 | |
|                                 p1:=crealconstnode.create(d,pbestrealtype^);
 | |
|                              end;
 | |
|                          end
 | |
|                        else
 | |
|                          begin
 | |
|                             consume(_INTCONST);
 | |
|                             p1:=cordconstnode.create(ic,cs64bittype,true);
 | |
|                          end
 | |
|                      end
 | |
|                    else
 | |
|                      begin
 | |
|                        consume(_INTCONST);
 | |
|                        p1:=cordconstnode.create(l,defaultordconsttype,true)
 | |
|                      end
 | |
|                  end
 | |
|                else
 | |
|                 begin
 | |
|                    consume(_INTCONST);
 | |
|                    { check whether the value isn't in the longint range as well }
 | |
|                    { (longint is easier to perform calculations with) (JM)      }
 | |
|                    if card <= $7fffffff then
 | |
|                      { no sign extension necessary, so not longint typecast (JM) }
 | |
|                      p1:=cordconstnode.create(card,s32bittype,true)
 | |
|                    else
 | |
|                      p1:=cordconstnode.create(card,u32bittype,true)
 | |
|                 end;
 | |
|              end;
 | |
| 
 | |
|            _REALNUMBER :
 | |
|              begin
 | |
|                val(pattern,d,code);
 | |
|                if code<>0 then
 | |
|                 begin
 | |
|                   Message(parser_e_error_in_real);
 | |
|                   d:=1.0;
 | |
|                 end;
 | |
|                consume(_REALNUMBER);
 | |
|                p1:=crealconstnode.create(d,pbestrealtype^);
 | |
|              end;
 | |
| 
 | |
|            _STRING :
 | |
|              begin
 | |
|                string_dec(htype);
 | |
|                { STRING can be also a type cast }
 | |
|                if token=_LKLAMMER then
 | |
|                 begin
 | |
|                   consume(_LKLAMMER);
 | |
|                   p1:=comp_expr(true);
 | |
|                   consume(_RKLAMMER);
 | |
|                   p1:=ctypeconvnode.create_explicit(p1,htype);
 | |
|                   { handle postfix operators here e.g. string(a)[10] }
 | |
|                   again:=true;
 | |
|                   postfixoperators(p1,again);
 | |
|                 end
 | |
|                else
 | |
|                 p1:=ctypenode.create(htype);
 | |
|              end;
 | |
| 
 | |
|            _FILE :
 | |
|              begin
 | |
|                htype:=cfiletype;
 | |
|                consume(_FILE);
 | |
|                { FILE can be also a type cast }
 | |
|                if token=_LKLAMMER then
 | |
|                 begin
 | |
|                   consume(_LKLAMMER);
 | |
|                   p1:=comp_expr(true);
 | |
|                   consume(_RKLAMMER);
 | |
|                   p1:=ctypeconvnode.create_explicit(p1,htype);
 | |
|                   { handle postfix operators here e.g. string(a)[10] }
 | |
|                   again:=true;
 | |
|                   postfixoperators(p1,again);
 | |
|                 end
 | |
|                else
 | |
|                 begin
 | |
|                   p1:=ctypenode.create(htype);
 | |
|                 end;
 | |
|              end;
 | |
| 
 | |
|            _CSTRING :
 | |
|              begin
 | |
|                p1:=cstringconstnode.createstr(pattern,st_default);
 | |
|                consume(_CSTRING);
 | |
|              end;
 | |
| 
 | |
|            _CCHAR :
 | |
|              begin
 | |
|                p1:=cordconstnode.create(ord(pattern[1]),cchartype,true);
 | |
|                consume(_CCHAR);
 | |
|              end;
 | |
| 
 | |
|            _CWSTRING:
 | |
|              begin
 | |
|                p1:=cstringconstnode.createwstr(patternw);
 | |
|                consume(_CWSTRING);
 | |
|              end;
 | |
| 
 | |
|            _CWCHAR:
 | |
|              begin
 | |
|                p1:=cordconstnode.create(ord(getcharwidestring(patternw,0)),cwidechartype,true);
 | |
|                consume(_CWCHAR);
 | |
|              end;
 | |
| 
 | |
|            _KLAMMERAFFE :
 | |
|              begin
 | |
|                consume(_KLAMMERAFFE);
 | |
|                got_addrn:=true;
 | |
|                { support both @<x> and @(<x>) }
 | |
|                if try_to_consume(_LKLAMMER) then
 | |
|                 begin
 | |
|                   p1:=factor(true);
 | |
|                   if token in [_CARET,_POINT,_LECKKLAMMER] then
 | |
|                    begin
 | |
|                      again:=true;
 | |
|                      postfixoperators(p1,again);
 | |
|                    end;
 | |
|                   consume(_RKLAMMER);
 | |
|                 end
 | |
|                else
 | |
|                 p1:=factor(true);
 | |
|                if token in [_CARET,_POINT,_LECKKLAMMER] then
 | |
|                 begin
 | |
|                   again:=true;
 | |
|                   postfixoperators(p1,again);
 | |
|                 end;
 | |
|                got_addrn:=false;
 | |
|                p1:=caddrnode.create(p1);
 | |
|                if cs_typed_addresses in aktlocalswitches then
 | |
|                  include(p1.flags,nf_typedaddr);
 | |
|                { Store the procvar that we are expecting, the
 | |
|                  addrn will use the information to find the correct
 | |
|                  procdef or it will return an error }
 | |
|                if assigned(getprocvardef) and
 | |
|                   (taddrnode(p1).left.nodetype = loadn) then
 | |
|                  taddrnode(p1).getprocvardef:=getprocvardef;
 | |
|              end;
 | |
| 
 | |
|            _LKLAMMER :
 | |
|              begin
 | |
|                consume(_LKLAMMER);
 | |
|                p1:=comp_expr(true);
 | |
|                consume(_RKLAMMER);
 | |
|                { it's not a good solution     }
 | |
|                { but (a+b)^ makes some problems  }
 | |
|                if token in [_CARET,_POINT,_LECKKLAMMER] then
 | |
|                 begin
 | |
|                   again:=true;
 | |
|                   postfixoperators(p1,again);
 | |
|                 end;
 | |
|              end;
 | |
| 
 | |
|            _LECKKLAMMER :
 | |
|              begin
 | |
|                consume(_LECKKLAMMER);
 | |
|                p1:=factor_read_set;
 | |
|                consume(_RECKKLAMMER);
 | |
|              end;
 | |
| 
 | |
|            _PLUS :
 | |
|              begin
 | |
|                consume(_PLUS);
 | |
|                p1:=factor(false);
 | |
|              end;
 | |
| 
 | |
|            _MINUS :
 | |
|              begin
 | |
|                consume(_MINUS);
 | |
|                if (token = _INTCONST) then
 | |
|                   begin
 | |
|                     { ugly hack, but necessary to be able to parse }
 | |
|                     { -9223372036854775808 as int64 (JM)           }
 | |
|                     pattern := '-'+pattern;
 | |
|                     p1:=sub_expr(oppower,false);
 | |
|                     {  -1 ** 4 should be - (1 ** 4) and not
 | |
|                        (-1) ** 4
 | |
|                        This was the reason of tw0869.pp test failure PM }
 | |
|                     if p1.nodetype=starstarn then
 | |
|                       begin
 | |
|                         if tbinarynode(p1).left.nodetype=ordconstn then
 | |
|                           begin
 | |
|                             tordconstnode(tbinarynode(p1).left).value:=-tordconstnode(tbinarynode(p1).left).value;
 | |
|                             p1:=cunaryminusnode.create(p1);
 | |
|                           end
 | |
|                         else if tbinarynode(p1).left.nodetype=realconstn then
 | |
|                           begin
 | |
|                             trealconstnode(tbinarynode(p1).left).value_real:=-trealconstnode(tbinarynode(p1).left).value_real;
 | |
|                             p1:=cunaryminusnode.create(p1);
 | |
|                           end
 | |
|                         else
 | |
|                           internalerror(20021029);
 | |
|                       end;
 | |
|                   end
 | |
|                else
 | |
|                  begin
 | |
|                    p1:=sub_expr(oppower,false);
 | |
|                    p1:=cunaryminusnode.create(p1);
 | |
|                  end;
 | |
|              end;
 | |
| 
 | |
|            _OP_NOT :
 | |
|              begin
 | |
|                consume(_OP_NOT);
 | |
|                p1:=factor(false);
 | |
|                p1:=cnotnode.create(p1);
 | |
|              end;
 | |
| 
 | |
|            _TRUE :
 | |
|              begin
 | |
|                consume(_TRUE);
 | |
|                p1:=cordconstnode.create(1,booltype,false);
 | |
|              end;
 | |
| 
 | |
|            _FALSE :
 | |
|              begin
 | |
|                consume(_FALSE);
 | |
|                p1:=cordconstnode.create(0,booltype,false);
 | |
|              end;
 | |
| 
 | |
|            _NIL :
 | |
|              begin
 | |
|                consume(_NIL);
 | |
|                p1:=cnilnode.create;
 | |
|                { It's really ugly code nil^, but delphi allows it }
 | |
|                if token in [_CARET] then
 | |
|                 begin
 | |
|                   again:=true;
 | |
|                   postfixoperators(p1,again);
 | |
|                 end;
 | |
|              end;
 | |
| 
 | |
|            else
 | |
|              begin
 | |
|                p1:=cerrornode.create;
 | |
|                consume(token);
 | |
|                Message(cg_e_illegal_expression);
 | |
|              end;
 | |
|         end;
 | |
| 
 | |
|         { generate error node if no node is created }
 | |
|         if not assigned(p1) then
 | |
|          begin
 | |
| {$ifdef EXTDEBUG}
 | |
|            Comment(V_Warning,'factor: p1=nil');
 | |
| {$endif}
 | |
|            p1:=cerrornode.create;
 | |
|          end;
 | |
| 
 | |
|         { get the resulttype for the node }
 | |
|         if (not assigned(p1.resulttype.def)) then
 | |
|          do_resulttypepass(p1);
 | |
| 
 | |
|         { tp7 procvar handling, but not if the next token
 | |
|           will be a := }
 | |
|         check_tp_procvar(p1);
 | |
| 
 | |
|         factor:=p1;
 | |
|         check_tokenpos;
 | |
|       end;
 | |
| {$ifdef fpc}
 | |
|   {$maxfpuregisters default}
 | |
| {$endif fpc}
 | |
| 
 | |
| {****************************************************************************
 | |
|                              Sub_Expr
 | |
| ****************************************************************************}
 | |
|    const
 | |
|       { Warning these stay be ordered !! }
 | |
|       operator_levels:array[Toperator_precedence] of set of Ttoken=
 | |
|          ([_LT,_LTE,_GT,_GTE,_EQUAL,_UNEQUAL,_OP_IN,_OP_IS],
 | |
|           [_PLUS,_MINUS,_OP_OR,_OP_XOR],
 | |
|           [_CARET,_SYMDIF,_STARSTAR,_STAR,_SLASH,
 | |
|            _OP_AS,_OP_AND,_OP_DIV,_OP_MOD,_OP_SHL,_OP_SHR],
 | |
|           [_STARSTAR] );
 | |
| 
 | |
|     function sub_expr(pred_level:Toperator_precedence;accept_equal : boolean):tnode;
 | |
|     {Reads a subexpression while the operators are of the current precedence
 | |
|      level, or any higher level. Replaces the old term, simpl_expr and
 | |
|      simpl2_expr.}
 | |
|       var
 | |
|         p1,p2   : tnode;
 | |
|         oldt    : Ttoken;
 | |
|         filepos : tfileposinfo;
 | |
|       begin
 | |
|         if pred_level=highest_precedence then
 | |
|           p1:=factor(false)
 | |
|         else
 | |
|           p1:=sub_expr(succ(pred_level),true);
 | |
|         repeat
 | |
|           if (token in operator_levels[pred_level]) and
 | |
|              ((token<>_EQUAL) or accept_equal) then
 | |
|            begin
 | |
|              oldt:=token;
 | |
|              filepos:=akttokenpos;
 | |
|              consume(token);
 | |
|              if pred_level=highest_precedence then
 | |
|                p2:=factor(false)
 | |
|              else
 | |
|                p2:=sub_expr(succ(pred_level),true);
 | |
|              case oldt of
 | |
|                _PLUS :
 | |
|                  p1:=caddnode.create(addn,p1,p2);
 | |
|                _MINUS :
 | |
|                  p1:=caddnode.create(subn,p1,p2);
 | |
|                _STAR :
 | |
|                  p1:=caddnode.create(muln,p1,p2);
 | |
|                _SLASH :
 | |
|                  p1:=caddnode.create(slashn,p1,p2);
 | |
|                _EQUAL :
 | |
|                  p1:=caddnode.create(equaln,p1,p2);
 | |
|                _GT :
 | |
|                  p1:=caddnode.create(gtn,p1,p2);
 | |
|                _LT :
 | |
|                  p1:=caddnode.create(ltn,p1,p2);
 | |
|                _GTE :
 | |
|                  p1:=caddnode.create(gten,p1,p2);
 | |
|                _LTE :
 | |
|                  p1:=caddnode.create(lten,p1,p2);
 | |
|                _SYMDIF :
 | |
|                  p1:=caddnode.create(symdifn,p1,p2);
 | |
|                _STARSTAR :
 | |
|                  p1:=caddnode.create(starstarn,p1,p2);
 | |
|                _OP_AS :
 | |
|                  p1:=casnode.create(p1,p2);
 | |
|                _OP_IN :
 | |
|                  p1:=cinnode.create(p1,p2);
 | |
|                _OP_IS :
 | |
|                  p1:=cisnode.create(p1,p2);
 | |
|                _OP_OR :
 | |
|                  p1:=caddnode.create(orn,p1,p2);
 | |
|                _OP_AND :
 | |
|                  p1:=caddnode.create(andn,p1,p2);
 | |
|                _OP_DIV :
 | |
|                  p1:=cmoddivnode.create(divn,p1,p2);
 | |
|                _OP_NOT :
 | |
|                  p1:=cnotnode.create(p1);
 | |
|                _OP_MOD :
 | |
|                  p1:=cmoddivnode.create(modn,p1,p2);
 | |
|                _OP_SHL :
 | |
|                  p1:=cshlshrnode.create(shln,p1,p2);
 | |
|                _OP_SHR :
 | |
|                  p1:=cshlshrnode.create(shrn,p1,p2);
 | |
|                _OP_XOR :
 | |
|                  p1:=caddnode.create(xorn,p1,p2);
 | |
|                _ASSIGNMENT :
 | |
|                  p1:=cassignmentnode.create(p1,p2);
 | |
|                _CARET :
 | |
|                  p1:=caddnode.create(caretn,p1,p2);
 | |
|                _UNEQUAL :
 | |
|                  p1:=caddnode.create(unequaln,p1,p2);
 | |
|              end;
 | |
|              p1.set_tree_filepos(filepos);
 | |
|            end
 | |
|           else
 | |
|            break;
 | |
|         until false;
 | |
|         sub_expr:=p1;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function comp_expr(accept_equal : boolean):tnode;
 | |
|       var
 | |
|          oldafterassignment : boolean;
 | |
|          p1 : tnode;
 | |
|       begin
 | |
|          oldafterassignment:=afterassignment;
 | |
|          afterassignment:=true;
 | |
|          p1:=sub_expr(opcompare,accept_equal);
 | |
|          { get the resulttype for this expression }
 | |
|          if not assigned(p1.resulttype.def) then
 | |
|           do_resulttypepass(p1);
 | |
|          afterassignment:=oldafterassignment;
 | |
|          comp_expr:=p1;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function expr : tnode;
 | |
| 
 | |
|       var
 | |
|          p1,p2 : tnode;
 | |
|          oldafterassignment : boolean;
 | |
|          oldp1 : tnode;
 | |
|          filepos : tfileposinfo;
 | |
| 
 | |
|       begin
 | |
|          oldafterassignment:=afterassignment;
 | |
|          p1:=sub_expr(opcompare,true);
 | |
|          { get the resulttype for this expression }
 | |
|          if not assigned(p1.resulttype.def) then
 | |
|           do_resulttypepass(p1);
 | |
|          filepos:=akttokenpos;
 | |
|          check_tp_procvar(p1);
 | |
|          if token in [_ASSIGNMENT,_PLUSASN,_MINUSASN,_STARASN,_SLASHASN] then
 | |
|            afterassignment:=true;
 | |
|          oldp1:=p1;
 | |
|          case token of
 | |
|            _POINTPOINT :
 | |
|              begin
 | |
|                 consume(_POINTPOINT);
 | |
|                 p2:=sub_expr(opcompare,true);
 | |
|                 p1:=crangenode.create(p1,p2);
 | |
|              end;
 | |
|            _ASSIGNMENT :
 | |
|              begin
 | |
|                 consume(_ASSIGNMENT);
 | |
|                 if (p1.resulttype.def.deftype=procvardef) then
 | |
|                   getprocvardef:=tprocvardef(p1.resulttype.def);
 | |
|                 p2:=sub_expr(opcompare,true);
 | |
|                 if assigned(getprocvardef) then
 | |
|                   handle_procvar(getprocvardef,p2);
 | |
|                 getprocvardef:=nil;
 | |
|                 p1:=cassignmentnode.create(p1,p2);
 | |
|              end;
 | |
|            _PLUSASN :
 | |
|              begin
 | |
|                consume(_PLUSASN);
 | |
|                p2:=sub_expr(opcompare,true);
 | |
|                p1:=cassignmentnode.create(p1,caddnode.create(addn,p1.getcopy,p2));
 | |
|             end;
 | |
|           _MINUSASN :
 | |
|             begin
 | |
|                consume(_MINUSASN);
 | |
|                p2:=sub_expr(opcompare,true);
 | |
|                p1:=cassignmentnode.create(p1,caddnode.create(subn,p1.getcopy,p2));
 | |
|             end;
 | |
|           _STARASN :
 | |
|             begin
 | |
|                consume(_STARASN  );
 | |
|                p2:=sub_expr(opcompare,true);
 | |
|                p1:=cassignmentnode.create(p1,caddnode.create(muln,p1.getcopy,p2));
 | |
|             end;
 | |
|           _SLASHASN :
 | |
|             begin
 | |
|                consume(_SLASHASN  );
 | |
|                p2:=sub_expr(opcompare,true);
 | |
|                p1:=cassignmentnode.create(p1,caddnode.create(slashn,p1.getcopy,p2));
 | |
|             end;
 | |
|          end;
 | |
|          { get the resulttype for this expression }
 | |
|          if not assigned(p1.resulttype.def) then
 | |
|           do_resulttypepass(p1);
 | |
|          afterassignment:=oldafterassignment;
 | |
|          if p1<>oldp1 then
 | |
|            p1.set_tree_filepos(filepos);
 | |
|          expr:=p1;
 | |
|       end;
 | |
| 
 | |
| {$ifdef int64funcresok}
 | |
|     function get_intconst:TConstExprInt;
 | |
| {$else int64funcresok}
 | |
|     function get_intconst:longint;
 | |
| {$endif int64funcresok}
 | |
|     {Reads an expression, tries to evalute it and check if it is an integer
 | |
|      constant. Then the constant is returned.}
 | |
|     var
 | |
|       p:tnode;
 | |
|     begin
 | |
|       p:=comp_expr(true);
 | |
|       if not codegenerror then
 | |
|        begin
 | |
|          if (p.nodetype<>ordconstn) or
 | |
|             not(is_integer(p.resulttype.def)) then
 | |
|           Message(cg_e_illegal_expression)
 | |
|          else
 | |
|           get_intconst:=tordconstnode(p).value;
 | |
|        end;
 | |
|       p.free;
 | |
|     end;
 | |
| 
 | |
| 
 | |
|     function get_stringconst:string;
 | |
|     {Reads an expression, tries to evaluate it and checks if it is a string
 | |
|      constant. Then the constant is returned.}
 | |
|     var
 | |
|       p:tnode;
 | |
|     begin
 | |
|       get_stringconst:='';
 | |
|       p:=comp_expr(true);
 | |
|       if p.nodetype<>stringconstn then
 | |
|         begin
 | |
|           if (p.nodetype=ordconstn) and is_char(p.resulttype.def) then
 | |
|             get_stringconst:=char(tordconstnode(p).value)
 | |
|           else
 | |
|             Message(cg_e_illegal_expression);
 | |
|         end
 | |
|       else
 | |
|         get_stringconst:=strpas(tstringconstnode(p).value_str);
 | |
|       p.free;
 | |
|     end;
 | |
| 
 | |
| end.
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.145  2003-12-29 17:19:35  jonas
 | |
|     * integrated hack from 1.0.x so we can parse low(int64) as int64 instead
 | |
|       of as double (in 1.0.x, it was necessary for low(longint))
 | |
| 
 | |
|   Revision 1.144  2003/12/08 22:35:28  peter
 | |
|     * again procvar fixes
 | |
| 
 | |
|   Revision 1.143  2003/11/29 16:19:54  peter
 | |
|     * Initialize() added
 | |
| 
 | |
|   Revision 1.142  2003/11/29 14:49:46  peter
 | |
|     * fix crash with exit() in a procedure
 | |
| 
 | |
|   Revision 1.141  2003/11/29 14:33:13  peter
 | |
|     * typed address only used for @ and addr() that are parsed
 | |
| 
 | |
|   Revision 1.140  2003/11/10 19:11:39  peter
 | |
|     * check paralength instead of assigned(left)
 | |
| 
 | |
|   Revision 1.139  2003/11/07 15:58:32  florian
 | |
|     * Florian's culmutative nr. 1; contains:
 | |
|       - invalid calling conventions for a certain cpu are rejected
 | |
|       - arm softfloat calling conventions
 | |
|       - -Sp for cpu dependend code generation
 | |
|       - several arm fixes
 | |
|       - remaining code for value open array paras on heap
 | |
| 
 | |
|   Revision 1.138  2003/11/06 15:54:32  peter
 | |
|     * fixed calling classmethod for other object from classmethod
 | |
| 
 | |
|   Revision 1.137  2003/11/04 16:42:13  peter
 | |
|     * assigned(proc()) does not change the calln to loadn
 | |
| 
 | |
|   Revision 1.136  2003/10/28 15:36:01  peter
 | |
|     * absolute to object field supported, fixes tb0458
 | |
| 
 | |
|   Revision 1.135  2003/10/09 15:20:56  peter
 | |
|     * self is not a token anymore. It is handled special when found
 | |
|       in a code block and when parsing an method
 | |
| 
 | |
|   Revision 1.134  2003/10/09 15:00:13  florian
 | |
|     * fixed constructor call in class methods
 | |
| 
 | |
|   Revision 1.133  2003/10/08 19:19:45  peter
 | |
|     * set_varstate cleanup
 | |
| 
 | |
|   Revision 1.132  2003/10/05 12:56:04  peter
 | |
|     * fix assigned(property)
 | |
| 
 | |
|   Revision 1.131  2003/10/02 21:15:31  peter
 | |
|     * protected visibility fixes
 | |
| 
 | |
|   Revision 1.130  2003/10/01 20:34:49  peter
 | |
|     * procinfo unit contains tprocinfo
 | |
|     * cginfo renamed to cgbase
 | |
|     * moved cgmessage to verbose
 | |
|     * fixed ppc and sparc compiles
 | |
| 
 | |
|   Revision 1.129  2003/09/23 17:56:05  peter
 | |
|     * locals and paras are allocated in the code generation
 | |
|     * tvarsym.localloc contains the location of para/local when
 | |
|       generating code for the current procedure
 | |
| 
 | |
|   Revision 1.128  2003/09/06 22:27:09  florian
 | |
|     * fixed web bug 2669
 | |
|     * cosmetic fix in printnode
 | |
|     * tobjectdef.gettypename implemented
 | |
| 
 | |
|   Revision 1.127  2003/09/05 17:41:12  florian
 | |
|     * merged Wiktor's Watcom patches in 1.1
 | |
| 
 | |
|   Revision 1.126  2003/08/23 22:29:51  peter
 | |
|     * fixed static class check for properties
 | |
| 
 | |
|   Revision 1.125  2003/08/23 18:41:52  peter
 | |
|     * allow typeof(self) in class methods
 | |
| 
 | |
|   Revision 1.124  2003/08/10 17:25:23  peter
 | |
|     * fixed some reported bugs
 | |
| 
 | |
|   Revision 1.123  2003/06/13 21:19:31  peter
 | |
|     * current_procdef removed, use current_procinfo.procdef instead
 | |
| 
 | |
|   Revision 1.122  2003/06/03 21:02:57  peter
 | |
|     * don't set nf_member when loaded from with symtable
 | |
|     * allow static variables in class methods
 | |
| 
 | |
|   Revision 1.121  2003/05/22 17:43:21  peter
 | |
|     * search defaulthandler only for message methods
 | |
| 
 | |
|   Revision 1.120  2003/05/15 18:58:53  peter
 | |
|     * removed selfpointer_offset, vmtpointer_offset
 | |
|     * tvarsym.adjusted_address
 | |
|     * address in localsymtable is now in the real direction
 | |
|     * removed some obsolete globals
 | |
| 
 | |
|   Revision 1.119  2003/05/13 20:54:39  peter
 | |
|     * ifdef'd code that checked for failed inherited constructors
 | |
| 
 | |
|   Revision 1.118  2003/05/13 19:14:41  peter
 | |
|     * failn removed
 | |
|     * inherited result code check moven to pexpr
 | |
| 
 | |
|   Revision 1.117  2003/05/11 21:37:03  peter
 | |
|     * moved implicit exception frame from ncgutil to psub
 | |
|     * constructor/destructor helpers moved from cobj/ncgutil to psub
 | |
| 
 | |
|   Revision 1.116  2003/05/11 14:45:12  peter
 | |
|     * tloadnode does not support objectsymtable,withsymtable anymore
 | |
|     * withnode cleanup
 | |
|     * direct with rewritten to use temprefnode
 | |
| 
 | |
|   Revision 1.115  2003/05/09 17:47:03  peter
 | |
|     * self moved to hidden parameter
 | |
|     * removed hdisposen,hnewn,selfn
 | |
| 
 | |
|   Revision 1.114  2003/05/01 07:59:42  florian
 | |
|     * introduced defaultordconsttype to decribe the default size of ordinal constants
 | |
|       on 64 bit CPUs it's equal to cs64bitdef while on 32 bit CPUs it's equal to s32bitdef
 | |
|     + added defines CPU32 and CPU64 for 32 bit and 64 bit CPUs
 | |
|     * int64s/qwords are allowed as for loop counter on 64 bit CPUs
 | |
| 
 | |
|   Revision 1.113  2003/04/27 11:21:33  peter
 | |
|     * aktprocdef renamed to current_procinfo.procdef
 | |
|     * procinfo renamed to current_procinfo
 | |
|     * procinfo will now be stored in current_module so it can be
 | |
|       cleaned up properly
 | |
|     * gen_main_procsym changed to create_main_proc and release_main_proc
 | |
|       to also generate a tprocinfo structure
 | |
|     * fixed unit implicit initfinal
 | |
| 
 | |
|   Revision 1.112  2003/04/27 07:29:50  peter
 | |
|     * current_procinfo.procdef cleanup, current_procdef is now always nil when parsing
 | |
|       a new procdef declaration
 | |
|     * aktprocsym removed
 | |
|     * lexlevel removed, use symtable.symtablelevel instead
 | |
|     * implicit init/final code uses the normal genentry/genexit
 | |
|     * funcret state checking updated for new funcret handling
 | |
| 
 | |
|   Revision 1.111  2003/04/26 00:33:07  peter
 | |
|     * vo_is_result flag added for the special RESULT symbol
 | |
| 
 | |
|   Revision 1.110  2003/04/25 20:59:33  peter
 | |
|     * removed funcretn,funcretsym, function result is now in varsym
 | |
|       and aliases for result and function name are added using absolutesym
 | |
|     * vs_hidden parameter for funcret passed in parameter
 | |
|     * vs_hidden fixes
 | |
|     * writenode changed to printnode and released from extdebug
 | |
|     * -vp option added to generate a tree.log with the nodetree
 | |
|     * nicer printnode for statements, callnode
 | |
| 
 | |
|   Revision 1.109  2003/04/23 10:13:55  peter
 | |
|     * firstaddr will check procvardef
 | |
| 
 | |
|   Revision 1.108  2003/04/22 23:50:23  peter
 | |
|     * firstpass uses expectloc
 | |
|     * checks if there are differences between the expectloc and
 | |
|       location.loc from secondpass in EXTDEBUG
 | |
| 
 | |
|   Revision 1.107  2003/04/11 15:49:01  peter
 | |
|     * default property also increased the reference count for the
 | |
|       property symbol
 | |
| 
 | |
|   Revision 1.106  2003/04/11 14:50:08  peter
 | |
|     * fix tw2454
 | |
| 
 | |
|   Revision 1.105  2003/03/27 17:44:13  peter
 | |
|     * fixed small mem leaks
 | |
| 
 | |
|   Revision 1.104  2003/03/17 18:55:30  peter
 | |
|     * allow more tokens instead of only semicolon after inherited
 | |
| 
 | |
|   Revision 1.103  2003/03/17 16:54:41  peter
 | |
|     * support DefaultHandler and anonymous inheritance fixed
 | |
|       for message methods
 | |
| 
 | |
|   Revision 1.102  2003/01/30 21:46:57  peter
 | |
|     * self fixes for static methods (merged)
 | |
| 
 | |
|   Revision 1.101  2003/01/16 22:12:22  peter
 | |
|     * Find the correct procvar to load when using @ in fpc mode
 | |
| 
 | |
|   Revision 1.100  2003/01/15 01:44:32  peter
 | |
|     * merged methodpointer fixes from 1.0.x
 | |
| 
 | |
|   Revision 1.98  2003/01/12 17:51:42  peter
 | |
|     * tp procvar handling fix for tb0448
 | |
| 
 | |
|   Revision 1.97  2003/01/05 22:44:14  peter
 | |
|     * remove a lot of code to support typen in loadn-procsym
 | |
| 
 | |
|   Revision 1.96  2002/12/11 22:40:36  peter
 | |
|     * assigned(procvar) fix for delphi mode, fixes tb0430
 | |
| 
 | |
|   Revision 1.95  2002/11/30 11:12:48  carl
 | |
|     + checking for symbols used with hint directives is done mostly in pexpr
 | |
|       only now
 | |
| 
 | |
|   Revision 1.94  2002/11/27 15:33:47  peter
 | |
|     * the never ending story of tp procvar hacks
 | |
| 
 | |
|   Revision 1.93  2002/11/26 22:58:24  peter
 | |
|     * fix for tw2178. When a ^ or . follows a procsym then the procsym
 | |
|       needs to be called
 | |
| 
 | |
|   Revision 1.92  2002/11/25 17:43:22  peter
 | |
|     * splitted defbase in defutil,symutil,defcmp
 | |
|     * merged isconvertable and is_equal into compare_defs(_ext)
 | |
|     * made operator search faster by walking the list only once
 | |
| 
 | |
|   Revision 1.91  2002/11/22 22:48:10  carl
 | |
|    * memory optimization with tconstsym (1.5%)
 | |
| 
 | |
|   Revision 1.90  2002/11/20 22:49:55  pierre
 | |
|    * commented check code tht was invalid in 1.1
 | |
| 
 | |
|   Revision 1.89  2002/11/18 18:34:41  peter
 | |
|     * fix crash with EXTDEBUG code
 | |
| 
 | |
|   Revision 1.88  2002/11/18 17:48:21  peter
 | |
|     * fix tw2209 (merged)
 | |
| 
 | |
|   Revision 1.87  2002/11/18 17:31:58  peter
 | |
|     * pass proccalloption to ret_in_xxx and push_xxx functions
 | |
| 
 | |
|   Revision 1.86  2002/10/05 00:48:57  peter
 | |
|     * support inherited; support for overload as it is handled by
 | |
|       delphi. This is only for delphi mode as it is working is
 | |
|       undocumented and hard to predict what is done
 | |
| 
 | |
|   Revision 1.85  2002/10/04 21:13:59  peter
 | |
|     * ignore vecn,subscriptn when checking for a procvar loadn
 | |
| 
 | |
|   Revision 1.84  2002/10/02 20:51:22  peter
 | |
|     * don't check interfaces for class methods
 | |
| 
 | |
|   Revision 1.83  2002/10/02 18:20:52  peter
 | |
|     * Copy() is now internal syssym that calls compilerprocs
 | |
| 
 | |
|   Revision 1.82  2002/09/30 07:00:48  florian
 | |
|     * fixes to common code to get the alpha compiler compiled applied
 | |
| 
 | |
|   Revision 1.81  2002/09/16 19:06:14  peter
 | |
|     * allow ^ after nil
 | |
| 
 | |
|   Revision 1.80  2002/09/07 15:25:07  peter
 | |
|     * old logs removed and tabs fixed
 | |
| 
 | |
|   Revision 1.79  2002/09/07 12:16:03  carl
 | |
|     * second part bug report 1996 fix, testrange in cordconstnode
 | |
|       only called if option is set (also make parsing a tiny faster)
 | |
| 
 | |
|   Revision 1.78  2002/09/03 16:26:27  daniel
 | |
|     * Make Tprocdef.defs protected
 | |
| 
 | |
|   Revision 1.77  2002/08/18 20:06:24  peter
 | |
|     * inlining is now also allowed in interface
 | |
|     * renamed write/load to ppuwrite/ppuload
 | |
|     * tnode storing in ppu
 | |
|     * nld,ncon,nbas are already updated for storing in ppu
 | |
| 
 | |
|   Revision 1.76  2002/08/17 09:23:39  florian
 | |
|     * first part of procinfo rewrite
 | |
| 
 | |
|   Revision 1.75  2002/08/01 16:37:47  jonas
 | |
|     - removed some superfluous "in_paras := true" statements
 | |
| 
 | |
|   Revision 1.74  2002/07/26 21:15:41  florian
 | |
|     * rewrote the system handling
 | |
| 
 | |
|   Revision 1.73  2002/07/23 09:51:23  daniel
 | |
|   * Tried to make Tprocsym.defs protected. I didn't succeed but the cleanups
 | |
|     are worth comitting.
 | |
| 
 | |
|   Revision 1.72  2002/07/20 11:57:55  florian
 | |
|     * types.pas renamed to defbase.pas because D6 contains a types
 | |
|       unit so this would conflicts if D6 programms are compiled
 | |
|     + Willamette/SSE2 instructions to assembler added
 | |
| 
 | |
|   Revision 1.71  2002/07/16 15:34:20  florian
 | |
|     * exit is now a syssym instead of a keyword
 | |
| 
 | |
|   Revision 1.70  2002/07/06 20:18:02  carl
 | |
|   * longstring declaration now gives parser error since its not supported!
 | |
| 
 | |
|   Revision 1.69  2002/06/12 15:46:14  jonas
 | |
|     * fixed web bug 1995
 | |
| 
 | |
|   Revision 1.68  2002/05/18 13:34:12  peter
 | |
|     * readded missing revisions
 | |
| 
 | |
|   Revision 1.67  2002/05/16 19:46:43  carl
 | |
|   + defines.inc -> fpcdefs.inc to avoid conflicts if compiling by hand
 | |
|   + try to fix temp allocation (still in ifdef)
 | |
|   + generic constructor calls
 | |
|   + start of tassembler / tmodulebase class cleanup
 | |
| 
 | |
|   Revision 1.65  2002/05/12 16:53:09  peter
 | |
|     * moved entry and exitcode to ncgutil and cgobj
 | |
|     * foreach gets extra argument for passing local data to the
 | |
|       iterator function
 | |
|     * -CR checks also class typecasts at runtime by changing them
 | |
|       into as
 | |
|     * fixed compiler to cycle with the -CR option
 | |
|     * fixed stabs with elf writer, finally the global variables can
 | |
|       be watched
 | |
|     * removed a lot of routines from cga unit and replaced them by
 | |
|       calls to cgobj
 | |
|     * u32bit-s32bit updates for and,or,xor nodes. When one element is
 | |
|       u32bit then the other is typecasted also to u32bit without giving
 | |
|       a rangecheck warning/error.
 | |
|     * fixed pascal calling method with reversing also the high tree in
 | |
|       the parast, detected by tcalcst3 test
 | |
| 
 | |
|   Revision 1.64  2002/04/23 19:16:34  peter
 | |
|     * add pinline unit that inserts compiler supported functions using
 | |
|       one or more statements
 | |
|     * moved finalize and setlength from ninl to pinline
 | |
| 
 | |
|   Revision 1.63  2002/04/21 19:02:05  peter
 | |
|     * removed newn and disposen nodes, the code is now directly
 | |
|       inlined from pexpr
 | |
|     * -an option that will write the secondpass nodes to the .s file, this
 | |
|       requires EXTDEBUG define to actually write the info
 | |
|     * fixed various internal errors and crashes due recent code changes
 | |
| 
 | |
|   Revision 1.62  2002/04/16 16:11:17  peter
 | |
|     * using inherited; without a parent having the same function
 | |
|       will do nothing like delphi
 | |
| 
 | |
|   Revision 1.61  2002/04/07 13:31:36  carl
 | |
|   + change unit use
 | |
| 
 | |
|   Revision 1.60  2002/04/01 20:57:13  jonas
 | |
|     * fixed web bug 1907
 | |
|     * fixed some other procvar related bugs (all related to accepting procvar
 | |
|         constructs with either too many or too little parameters)
 | |
|     (both merged, includes second typo fix of pexpr.pas)
 | |
| 
 | |
|   Revision 1.59  2002/03/31 20:26:35  jonas
 | |
|     + a_loadfpu_* and a_loadmm_* methods in tcg
 | |
|     * register allocation is now handled by a class and is mostly processor
 | |
|       independent (+rgobj.pas and i386/rgcpu.pas)
 | |
|     * temp allocation is now handled by a class (+tgobj.pas, -i386\tgcpu.pas)
 | |
|     * some small improvements and fixes to the optimizer
 | |
|     * some register allocation fixes
 | |
|     * some fpuvaroffset fixes in the unary minus node
 | |
|     * push/popusedregisters is now called rg.save/restoreusedregisters and
 | |
|       (for i386) uses temps instead of push/pop's when using -Op3 (that code is
 | |
|       also better optimizable)
 | |
|     * fixed and optimized register saving/restoring for new/dispose nodes
 | |
|     * LOC_FPU locations now also require their "register" field to be set to
 | |
|       R_ST, not R_ST0 (the latter is used for LOC_CFPUREGISTER locations only)
 | |
|     - list field removed of the tnode class because it's not used currently
 | |
|       and can cause hard-to-find bugs
 | |
| 
 | |
| }
 | 
