mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 04:11:27 +01:00 
			
		
		
		
	 4b82d30953
			
		
	
	
		4b82d30953
		
	
	
	
	
		
			
			+ Added read/write notifications of variables. These will be usefull for providing information for several optimizations. For example the value of the loop variable of a for loop does matter is the variable is read after the for loop, but if it's no longer used or written, it doesn't matter and this can be used to optimize the loop code generation.
		
			
				
	
	
		
			860 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			860 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     $Id$
 | |
|     Copyright (c) 2000-2002 by Florian Klaempfl
 | |
| 
 | |
|     This unit implements some basic nodes
 | |
| 
 | |
|     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 nbas;
 | |
| 
 | |
| {$i fpcdefs.inc}
 | |
| 
 | |
| interface
 | |
| 
 | |
|     uses
 | |
|        cpubase,
 | |
|        aasmbase,aasmtai,aasmcpu,
 | |
|        node,
 | |
|        symtype,symppu;
 | |
| 
 | |
|     type
 | |
|        tnothingnode = class(tnode)
 | |
|           constructor create;virtual;
 | |
|           function pass_1 : tnode;override;
 | |
|           function det_resulttype:tnode;override;
 | |
|        end;
 | |
|        tnothingnodeclass = class of tnothingnode;
 | |
| 
 | |
|        terrornode = class(tnode)
 | |
|           constructor create;virtual;
 | |
|           function pass_1 : tnode;override;
 | |
|           function det_resulttype:tnode;override;
 | |
|        end;
 | |
|        terrornodeclass = class of terrornode;
 | |
| 
 | |
|        tasmnode = class(tnode)
 | |
|           p_asm : taasmoutput;
 | |
|           constructor create(p : taasmoutput);virtual;
 | |
|           destructor destroy;override;
 | |
|           constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
 | |
|           procedure ppuwrite(ppufile:tcompilerppufile);override;
 | |
|           procedure derefimpl;override;
 | |
|           function getcopy : tnode;override;
 | |
|           function pass_1 : tnode;override;
 | |
|           function det_resulttype:tnode;override;
 | |
|           function docompare(p: tnode): boolean; override;
 | |
|        end;
 | |
|        tasmnodeclass = class of tasmnode;
 | |
| 
 | |
|        tstatementnode = class(tbinarynode)
 | |
|           constructor create(l,r : tnode);virtual;
 | |
|           function pass_1 : tnode;override;
 | |
|           function det_resulttype:tnode;override;
 | |
| {$ifdef extdebug}
 | |
|           procedure _dowrite;override;
 | |
| {$endif extdebug}
 | |
|        end;
 | |
|        tstatementnodeclass = class of tstatementnode;
 | |
| 
 | |
|        tblocknode = class(tunarynode)
 | |
|           constructor create(l : tnode);virtual;
 | |
|           function pass_1 : tnode;override;
 | |
|           function det_resulttype:tnode;override;
 | |
| {$ifdef state_tracking}
 | |
|           function track_state_pass(exec_known:boolean):boolean;override;
 | |
| {$endif state_tracking}
 | |
|        end;
 | |
|        tblocknodeclass = class of tblocknode;
 | |
| 
 | |
|        { to allow access to the location by temp references even after the temp has }
 | |
|        { already been disposed and to make sure the coherency between temps and     }
 | |
|        { temp references is kept after a getcopy                                    }
 | |
|        ptempinfo = ^ttempinfo;
 | |
|        ttempinfo = record
 | |
|          { set to the copy of a tempcreate pnode (if it gets copied) so that the }
 | |
|          { refs and deletenode can hook to this copy once they get copied too    }
 | |
|          hookoncopy : ptempinfo;
 | |
|          ref        : treference;
 | |
|          restype    : ttype;
 | |
|          valid      : boolean;
 | |
|        end;
 | |
| 
 | |
|        { a node which will create a (non)persistent temp of a given type with a given  }
 | |
|        { size (the size is separate to allow creating "void" temps with a custom size) }
 | |
|        ttempcreatenode = class(tnode)
 | |
|           size: longint;
 | |
|           tempinfo: ptempinfo;
 | |
|           persistent: boolean;
 | |
|           { * persistent temps are used in manually written code where the temp }
 | |
|           { be usable among different statements and where you can manually say }
 | |
|           { when the temp has to be freed (using a ttempdeletenode)             }
 | |
|           { * non-persistent temps are mostly used in typeconversion helpers,   }
 | |
|           { where the node that receives the temp becomes responsible for       }
 | |
|           { freeing it. In this last case, you should use only one reference    }
 | |
|           { to it and *not* generate a ttempdeletenode                          }
 | |
|           constructor create(const _restype: ttype; _size: longint; _persistent: boolean); virtual;
 | |
|           function getcopy: tnode; override;
 | |
|           function pass_1 : tnode; override;
 | |
|           function det_resulttype: tnode; override;
 | |
|           function docompare(p: tnode): boolean; override;
 | |
|         end;
 | |
|        ttempcreatenodeclass = class of ttempcreatenode;
 | |
| 
 | |
|         { a node which is a reference to a certain temp }
 | |
|         ttemprefnode = class(tnode)
 | |
|         {$ifdef var_notification}
 | |
|           writeaccess:boolean;
 | |
|         {$endif}
 | |
|           constructor create(const temp: ttempcreatenode); virtual;
 | |
|           constructor create_offset(const temp: ttempcreatenode;aoffset:longint);
 | |
|           function getcopy: tnode; override;
 | |
|           function pass_1 : tnode; override;
 | |
|           function det_resulttype : tnode; override;
 | |
|         {$ifdef var_notification}
 | |
|           procedure mark_write;override;
 | |
|         {$endif}
 | |
|           function docompare(p: tnode): boolean; override;
 | |
|          protected
 | |
|           tempinfo: ptempinfo;
 | |
|           offset : longint;
 | |
|         end;
 | |
|        ttemprefnodeclass = class of ttemprefnode;
 | |
| 
 | |
|         { a node which removes a temp }
 | |
|         ttempdeletenode = class(tnode)
 | |
|           constructor create(const temp: ttempcreatenode);
 | |
|           { this will convert the persistant temp to a normal temp
 | |
|             for returning to the other nodes }
 | |
|           constructor create_normal_temp(const temp: ttempcreatenode);
 | |
|           function getcopy: tnode; override;
 | |
|           function pass_1: tnode; override;
 | |
|           function det_resulttype: tnode; override;
 | |
|           function docompare(p: tnode): boolean; override;
 | |
|           destructor destroy; override;
 | |
|          protected
 | |
|           tempinfo: ptempinfo;
 | |
|           release_to_normal : boolean;
 | |
|         end;
 | |
|        ttempdeletenodeclass = class of ttempdeletenode;
 | |
| 
 | |
|     var
 | |
|        cnothingnode : tnothingnodeclass;
 | |
|        cerrornode : terrornodeclass;
 | |
|        casmnode : tasmnodeclass;
 | |
|        cstatementnode : tstatementnodeclass;
 | |
|        cblocknode : tblocknodeclass;
 | |
|        ctempcreatenode : ttempcreatenodeclass;
 | |
|        ctemprefnode : ttemprefnodeclass;
 | |
|        ctempdeletenode : ttempdeletenodeclass;
 | |
| 
 | |
|        { Create a blocknode and statement node for multiple statements
 | |
|          generated internally by the parser }
 | |
|        function  internalstatements(var laststatement:tstatementnode):tblocknode;
 | |
|        procedure addstatement(var laststatement:tstatementnode;n:tnode);
 | |
| 
 | |
| 
 | |
| implementation
 | |
| 
 | |
|     uses
 | |
|       cutils,
 | |
|       verbose,globals,globtype,systems,
 | |
|       symconst,symdef,symsym,defbase,
 | |
|       pass_1,
 | |
|       nld,ncal,nflw,rgobj,cgbase
 | |
|       ;
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                                      Helpers
 | |
| *****************************************************************************}
 | |
| 
 | |
|     function internalstatements(var laststatement:tstatementnode):tblocknode;
 | |
|       begin
 | |
|         { create dummy initial statement }
 | |
|         laststatement := cstatementnode.create(nil,cnothingnode.create);
 | |
|         internalstatements := cblocknode.create(laststatement);
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure addstatement(var laststatement:tstatementnode;n:tnode);
 | |
|       begin
 | |
|         if assigned(laststatement.left) then
 | |
|          internalerror(200204201);
 | |
|         laststatement.left:=cstatementnode.create(nil,n);
 | |
|         laststatement:=tstatementnode(laststatement.left);
 | |
|       end;
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                              TFIRSTNOTHING
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor tnothingnode.create;
 | |
|       begin
 | |
|          inherited create(nothingn);
 | |
|       end;
 | |
| 
 | |
|     function tnothingnode.det_resulttype:tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          resulttype:=voidtype;
 | |
|       end;
 | |
| 
 | |
|     function tnothingnode.pass_1 : tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|       end;
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                              TFIRSTERROR
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor terrornode.create;
 | |
| 
 | |
|       begin
 | |
|          inherited create(errorn);
 | |
|       end;
 | |
| 
 | |
|     function terrornode.det_resulttype:tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          include(flags,nf_error);
 | |
|          codegenerror:=true;
 | |
|          resulttype:=generrortype;
 | |
|       end;
 | |
| 
 | |
|     function terrornode.pass_1 : tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          codegenerror:=true;
 | |
|       end;
 | |
| 
 | |
| {*****************************************************************************
 | |
|                             TSTATEMENTNODE
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor tstatementnode.create(l,r : tnode);
 | |
| 
 | |
|       begin
 | |
|          inherited create(statementn,l,r);
 | |
|       end;
 | |
| 
 | |
|     function tstatementnode.det_resulttype:tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          resulttype:=voidtype;
 | |
| 
 | |
|          { right is the statement itself calln assignn or a complex one }
 | |
|          resulttypepass(right);
 | |
|          if (not (cs_extsyntax in aktmoduleswitches)) and
 | |
|             assigned(right.resulttype.def) and
 | |
|             not((right.nodetype=calln) and
 | |
|                 (tcallnode(right).procdefinition.proctypeoption=potype_constructor)) and
 | |
|             not(is_void(right.resulttype.def)) then
 | |
|            CGMessage(cg_e_illegal_expression);
 | |
|          if codegenerror then
 | |
|            exit;
 | |
| 
 | |
|          { left is the next in the list }
 | |
|          resulttypepass(left);
 | |
|          if codegenerror then
 | |
|            exit;
 | |
|       end;
 | |
| 
 | |
|     function tstatementnode.pass_1 : tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          { no temps over several statements }
 | |
|          rg.cleartempgen;
 | |
|          { right is the statement itself calln assignn or a complex one }
 | |
|          firstpass(right);
 | |
|          if codegenerror then
 | |
|            exit;
 | |
|          location.loc:=right.location.loc;
 | |
|          registers32:=right.registers32;
 | |
|          registersfpu:=right.registersfpu;
 | |
| {$ifdef SUPPORT_MMX}
 | |
|          registersmmx:=right.registersmmx;
 | |
| {$endif SUPPORT_MMX}
 | |
|          { left is the next in the list }
 | |
|          firstpass(left);
 | |
|          if codegenerror then
 | |
|            exit;
 | |
|          if right.registers32>registers32 then
 | |
|            registers32:=right.registers32;
 | |
|          if right.registersfpu>registersfpu then
 | |
|            registersfpu:=right.registersfpu;
 | |
| {$ifdef SUPPORT_MMX}
 | |
|          if right.registersmmx>registersmmx then
 | |
|            registersmmx:=right.registersmmx;
 | |
| {$endif}
 | |
|       end;
 | |
| 
 | |
| {$ifdef extdebug}
 | |
|     procedure tstatementnode._dowrite;
 | |
| 
 | |
|       begin
 | |
|          { can't use inherited dowrite, because that will use the
 | |
|            binary which we don't want for statements }
 | |
|          dowritenodetype;
 | |
|          writeln(',');
 | |
|          { write the statement }
 | |
|          writenodeindention:=writenodeindention+'    ';
 | |
|          writenode(right);
 | |
|          writeln(')');
 | |
|          delete(writenodeindention,1,4);
 | |
|          { go on with the next statement }
 | |
|          writenode(left);
 | |
|       end;
 | |
| {$endif}
 | |
| 
 | |
| {*****************************************************************************
 | |
|                              TBLOCKNODE
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor tblocknode.create(l : tnode);
 | |
| 
 | |
|       begin
 | |
|          inherited create(blockn,l);
 | |
|       end;
 | |
| 
 | |
|     function tblocknode.det_resulttype:tnode;
 | |
|       var
 | |
|          hp : tstatementnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          resulttype:=voidtype;
 | |
| 
 | |
|          hp:=tstatementnode(left);
 | |
|          while assigned(hp) do
 | |
|            begin
 | |
|               if assigned(hp.right) then
 | |
|                 begin
 | |
|                    codegenerror:=false;
 | |
|                    resulttypepass(hp.right);
 | |
|                    if (not (cs_extsyntax in aktmoduleswitches)) and
 | |
|                       assigned(hp.right.resulttype.def) and
 | |
|                       not((hp.right.nodetype=calln) and
 | |
|                           (tcallnode(hp.right).procdefinition.proctypeoption=potype_constructor)) and
 | |
|                       not(is_void(hp.right.resulttype.def)) then
 | |
|                      CGMessage(cg_e_illegal_expression);
 | |
|                    { the resulttype of the block is the last type that is
 | |
|                      returned. Normally this is a voidtype. But when the
 | |
|                      compiler inserts a block of multiple statements then the
 | |
|                      last entry can return a value }
 | |
|                    resulttype:=hp.right.resulttype;
 | |
|                 end;
 | |
|               hp:=tstatementnode(hp.left);
 | |
|            end;
 | |
|       end;
 | |
| 
 | |
|     function tblocknode.pass_1 : tnode;
 | |
|       var
 | |
|          hp : tstatementnode;
 | |
|          count : longint;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          count:=0;
 | |
|          hp:=tstatementnode(left);
 | |
|          while assigned(hp) do
 | |
|            begin
 | |
|               if cs_regalloc in aktglobalswitches then
 | |
|                 begin
 | |
|                    { node transformations }
 | |
| 
 | |
|                    { concat function result to exit }
 | |
|                    { this is wrong for string or other complex
 | |
|                      result types !!! }
 | |
|                    if {ret_in_acc(aktprocdef.rettype.def) and }
 | |
|                       (is_ordinal(aktprocdef.rettype.def) or
 | |
|                        is_smallset(aktprocdef.rettype.def)) and
 | |
|                       assigned(hp.left) and
 | |
|                       assigned(tstatementnode(hp.left).right) and
 | |
|                       (tstatementnode(hp.left).right.nodetype=exitn) and
 | |
|                       (hp.right.nodetype=assignn) and
 | |
|                       { !!!! this tbinarynode should be tassignmentnode }
 | |
|                       (tbinarynode(hp.right).left.nodetype=funcretn) then
 | |
|                       begin
 | |
|                          if assigned(texitnode(tstatementnode(hp.left).right).left) then
 | |
|                            CGMessage(cg_n_inefficient_code)
 | |
|                          else
 | |
|                            begin
 | |
|                               texitnode(tstatementnode(hp.left).right).left:=tassignmentnode(hp.right).right;
 | |
|                               tassignmentnode(hp.right).right:=nil;
 | |
|                               hp.right.free;
 | |
|                               hp.right:=nil;
 | |
|                            end;
 | |
|                       end
 | |
|                    { warning if unreachable code occurs and elimate this }
 | |
|                    else if (hp.right.nodetype in
 | |
|                      [exitn,breakn,continuen,goton]) and
 | |
|                      { statement node (JM) }
 | |
|                      assigned(hp.left) and
 | |
|                      { kind of statement! (JM) }
 | |
|                      assigned(tstatementnode(hp.left).right) and
 | |
|                      (tstatementnode(hp.left).right.nodetype<>labeln) then
 | |
|                      begin
 | |
|                         { use correct line number }
 | |
|                         aktfilepos:=hp.left.fileinfo;
 | |
|                         hp.left.free;
 | |
|                         hp.left:=nil;
 | |
|                         CGMessage(cg_w_unreachable_code);
 | |
|                         { old lines }
 | |
|                         aktfilepos:=hp.right.fileinfo;
 | |
|                      end;
 | |
|                 end;
 | |
|               if assigned(hp.right) then
 | |
|                 begin
 | |
|                    rg.cleartempgen;
 | |
|                    codegenerror:=false;
 | |
|                    firstpass(hp.right);
 | |
| 
 | |
|                    hp.registers32:=hp.right.registers32;
 | |
|                    hp.registersfpu:=hp.right.registersfpu;
 | |
| {$ifdef SUPPORT_MMX}
 | |
|                    hp.registersmmx:=hp.right.registersmmx;
 | |
| {$endif SUPPORT_MMX}
 | |
|                 end
 | |
|               else
 | |
|                 hp.registers32:=0;
 | |
| 
 | |
|               if hp.registers32>registers32 then
 | |
|                 registers32:=hp.registers32;
 | |
|               if hp.registersfpu>registersfpu then
 | |
|                 registersfpu:=hp.registersfpu;
 | |
| {$ifdef SUPPORT_MMX}
 | |
|               if hp.registersmmx>registersmmx then
 | |
|                 registersmmx:=hp.registersmmx;
 | |
| {$endif}
 | |
|               location.loc:=hp.location.loc;
 | |
|               inc(count);
 | |
|               hp:=tstatementnode(hp.left);
 | |
|            end;
 | |
|       end;
 | |
| 
 | |
| {$ifdef state_tracking}
 | |
|       function Tblocknode.track_state_pass(exec_known:boolean):boolean;
 | |
| 
 | |
|       var hp:Tstatementnode;
 | |
| 
 | |
|       begin
 | |
|         track_state_pass:=false;
 | |
|         hp:=Tstatementnode(left);
 | |
|         while assigned(hp) do
 | |
|             begin
 | |
|                 if hp.right.track_state_pass(exec_known) then
 | |
|                     track_state_pass:=true;
 | |
|                 hp:=Tstatementnode(hp.left);
 | |
|             end;
 | |
|       end;
 | |
| {$endif state_tracking}
 | |
| 
 | |
| {*****************************************************************************
 | |
|                              TASMNODE
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor tasmnode.create(p : taasmoutput);
 | |
| 
 | |
|       begin
 | |
|          inherited create(asmn);
 | |
|          p_asm:=p;
 | |
|       end;
 | |
| 
 | |
|     destructor tasmnode.destroy;
 | |
|       begin
 | |
|         if assigned(p_asm) then
 | |
|          p_asm.free;
 | |
|         inherited destroy;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     constructor tasmnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
 | |
|       var
 | |
|         hp : tai;
 | |
|       begin
 | |
|         inherited ppuload(t,ppufile);
 | |
|         p_asm:=taasmoutput.create;
 | |
|         repeat
 | |
|           hp:=ppuloadai(ppufile);
 | |
|           if hp=nil then
 | |
|            break;
 | |
|           p_asm.concat(hp);
 | |
|         until false;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure tasmnode.ppuwrite(ppufile:tcompilerppufile);
 | |
|       var
 | |
|         hp : tai;
 | |
|       begin
 | |
|         inherited ppuwrite(ppufile);
 | |
|         hp:=tai(p_asm.first);
 | |
|         while assigned(hp) do
 | |
|          begin
 | |
|            ppuwriteai(ppufile,hp);
 | |
|            hp:=tai(hp.next);
 | |
|          end;
 | |
|         { end is marked by a nil }
 | |
|         ppuwriteai(ppufile,nil);
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure tasmnode.derefimpl;
 | |
|       var
 | |
|         hp : tai;
 | |
|       begin
 | |
|         inherited derefimpl;
 | |
|         hp:=tai(p_asm.first);
 | |
|         while assigned(hp) do
 | |
|          begin
 | |
|            hp.derefimpl;
 | |
|            hp:=tai(hp.next);
 | |
|          end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function tasmnode.getcopy: tnode;
 | |
|       var
 | |
|         n: tasmnode;
 | |
|       begin
 | |
|         n := tasmnode(inherited getcopy);
 | |
|         if assigned(p_asm) then
 | |
|           begin
 | |
|             n.p_asm:=taasmoutput.create;
 | |
|             n.p_asm.concatlistcopy(p_asm);
 | |
|           end
 | |
|         else n.p_asm := nil;
 | |
|         getcopy := n;
 | |
|       end;
 | |
| 
 | |
|     function tasmnode.det_resulttype:tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          resulttype:=voidtype;
 | |
|       end;
 | |
| 
 | |
|     function tasmnode.pass_1 : tnode;
 | |
|       begin
 | |
|          result:=nil;
 | |
|          procinfo.flags:=procinfo.flags or pi_uses_asm;
 | |
|       end;
 | |
| 
 | |
|     function tasmnode.docompare(p: tnode): boolean;
 | |
|       begin
 | |
|         { comparing of asmlists is not implemented (JM) }
 | |
|         docompare := false;
 | |
|       end;
 | |
| 
 | |
| {*****************************************************************************
 | |
|                           TEMPCREATENODE
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor ttempcreatenode.create(const _restype: ttype; _size: longint; _persistent: boolean);
 | |
|       begin
 | |
|         inherited create(tempcreaten);
 | |
|         size := _size;
 | |
|         new(tempinfo);
 | |
|         fillchar(tempinfo^,sizeof(tempinfo^),0);
 | |
|         tempinfo^.restype := _restype;
 | |
|         persistent := _persistent;
 | |
|       end;
 | |
| 
 | |
|     function ttempcreatenode.getcopy: tnode;
 | |
|       var
 | |
|         n: ttempcreatenode;
 | |
|       begin
 | |
|         n := ttempcreatenode(inherited getcopy);
 | |
|         n.size := size;
 | |
| 
 | |
|         new(n.tempinfo);
 | |
|         fillchar(n.tempinfo^,sizeof(n.tempinfo^),0);
 | |
|         n.tempinfo^.restype := tempinfo^.restype;
 | |
| 
 | |
|         { signal the temprefs that the temp they point to has been copied, }
 | |
|         { so that if the refs get copied as well, they can hook themselves }
 | |
|         { to the copy of the temp                                          }
 | |
|         tempinfo^.hookoncopy := n.tempinfo;
 | |
| 
 | |
|         result := n;
 | |
|       end;
 | |
| 
 | |
|     function ttempcreatenode.pass_1 : tnode;
 | |
|       begin
 | |
|         result := nil;
 | |
|       end;
 | |
| 
 | |
|     function ttempcreatenode.det_resulttype: tnode;
 | |
|       begin
 | |
|         result := nil;
 | |
|         { a tempcreatenode doesn't have a resulttype, only temprefnodes do }
 | |
|         resulttype := voidtype;
 | |
|       end;
 | |
| 
 | |
|     function ttempcreatenode.docompare(p: tnode): boolean;
 | |
|       begin
 | |
|         result :=
 | |
|           inherited docompare(p) and
 | |
|           (ttempcreatenode(p).size = size) and
 | |
|           is_equal(ttempcreatenode(p).tempinfo^.restype.def,tempinfo^.restype.def);
 | |
|       end;
 | |
| 
 | |
| {*****************************************************************************
 | |
|                              TEMPREFNODE
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor ttemprefnode.create(const temp: ttempcreatenode);
 | |
|       begin
 | |
|         inherited create(temprefn);
 | |
|         tempinfo := temp.tempinfo;
 | |
|         offset:=0;
 | |
|       end;
 | |
| 
 | |
|     constructor ttemprefnode.create_offset(const temp: ttempcreatenode;aoffset:longint);
 | |
|       begin
 | |
|         self.create(temp);
 | |
|         offset := aoffset;
 | |
|       end;
 | |
| 
 | |
|     function ttemprefnode.getcopy: tnode;
 | |
|       var
 | |
|         n: ttemprefnode;
 | |
|       begin
 | |
|         n := ttemprefnode(inherited getcopy);
 | |
| 
 | |
|         if assigned(tempinfo^.hookoncopy) then
 | |
|           { if the temp has been copied, assume it becomes a new }
 | |
|           { temp which has to be hooked by the copied reference  }
 | |
|           begin
 | |
|             { hook the ref to the copied temp }
 | |
|             n.tempinfo := tempinfo^.hookoncopy;
 | |
|           end
 | |
|         else
 | |
|           { if the temp we refer to hasn't been copied, assume }
 | |
|           { we're just a new reference to that temp            }
 | |
|           begin
 | |
|             n.tempinfo := tempinfo;
 | |
|           end;
 | |
| 
 | |
|         result := n;
 | |
|       end;
 | |
| 
 | |
|     function ttemprefnode.pass_1 : tnode;
 | |
|       begin
 | |
|         location.loc:=LOC_REFERENCE;
 | |
|         result := nil;
 | |
|       end;
 | |
| 
 | |
|     function ttemprefnode.det_resulttype: tnode;
 | |
|       begin
 | |
|         { check if the temp is already resulttype passed }
 | |
|         if not assigned(tempinfo^.restype.def) then
 | |
|           internalerror(200108233);
 | |
|         result := nil;
 | |
|         resulttype := tempinfo^.restype;
 | |
|       end;
 | |
| 
 | |
|     function ttemprefnode.docompare(p: tnode): boolean;
 | |
|       begin
 | |
|         result :=
 | |
|           inherited docompare(p) and
 | |
|           (ttemprefnode(p).tempinfo = tempinfo);
 | |
|       end;
 | |
| 
 | |
| {$ifdef var_notification}
 | |
|     procedure Ttemprefnode.mark_write;
 | |
| 
 | |
|     begin
 | |
|       writeaccess:=true;
 | |
|     end;
 | |
| {$endif}
 | |
| 
 | |
| 
 | |
| {*****************************************************************************
 | |
|                              TEMPDELETENODE
 | |
| *****************************************************************************}
 | |
| 
 | |
|     constructor ttempdeletenode.create(const temp: ttempcreatenode);
 | |
|       begin
 | |
|         inherited create(temprefn);
 | |
|         tempinfo := temp.tempinfo;
 | |
|         release_to_normal := false;
 | |
|         if not temp.persistent then
 | |
|           internalerror(200204211);
 | |
|       end;
 | |
| 
 | |
|     constructor ttempdeletenode.create_normal_temp(const temp: ttempcreatenode);
 | |
|       begin
 | |
|         inherited create(tempdeleten);
 | |
|         tempinfo := temp.tempinfo;
 | |
|         release_to_normal := true;
 | |
|       end;
 | |
| 
 | |
|     function ttempdeletenode.getcopy: tnode;
 | |
|       var
 | |
|         n: ttempdeletenode;
 | |
|       begin
 | |
|         n := ttempdeletenode(inherited getcopy);
 | |
| 
 | |
|         if assigned(tempinfo^.hookoncopy) then
 | |
|           { if the temp has been copied, assume it becomes a new }
 | |
|           { temp which has to be hooked by the copied deletenode }
 | |
|           begin
 | |
|             { hook the tempdeletenode to the copied temp }
 | |
|             n.tempinfo := tempinfo^.hookoncopy;
 | |
|           end
 | |
|         else
 | |
|           { if the temp we refer to hasn't been copied, we have a }
 | |
|           { problem since that means we now have two delete nodes }
 | |
|           { for one temp                                          }
 | |
|           internalerror(200108234);
 | |
|         result := n;
 | |
|       end;
 | |
| 
 | |
|     function ttempdeletenode.pass_1 : tnode;
 | |
|       begin
 | |
|         result := nil;
 | |
|       end;
 | |
| 
 | |
|     function ttempdeletenode.det_resulttype: tnode;
 | |
|       begin
 | |
|         result := nil;
 | |
|         resulttype := voidtype;
 | |
|       end;
 | |
| 
 | |
|     function ttempdeletenode.docompare(p: tnode): boolean;
 | |
|       begin
 | |
|         result :=
 | |
|           inherited docompare(p) and
 | |
|           (ttemprefnode(p).tempinfo = tempinfo);
 | |
|       end;
 | |
| 
 | |
|     destructor ttempdeletenode.destroy;
 | |
|       begin
 | |
|         dispose(tempinfo);
 | |
|       end;
 | |
| 
 | |
| begin
 | |
|    cnothingnode:=tnothingnode;
 | |
|    cerrornode:=terrornode;
 | |
|    casmnode:=tasmnode;
 | |
|    cstatementnode:=tstatementnode;
 | |
|    cblocknode:=tblocknode;
 | |
|    ctempcreatenode:=ttempcreatenode;
 | |
|    ctemprefnode:=ttemprefnode;
 | |
|    ctempdeletenode:=ttempdeletenode;
 | |
| end.
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.35  2002-09-01 08:01:16  daniel
 | |
|    * Removed sets from Tcallnode.det_resulttype
 | |
|    + Added read/write notifications of variables. These will be usefull
 | |
|      for providing information for several optimizations. For example
 | |
|      the value of the loop variable of a for loop does matter is the
 | |
|      variable is read after the for loop, but if it's no longer used
 | |
|      or written, it doesn't matter and this can be used to optimize
 | |
|      the loop code generation.
 | |
| 
 | |
|   Revision 1.34  2002/08/18 20:06:23  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.33  2002/08/17 22:09:44  florian
 | |
|     * result type handling in tcgcal.pass_2 overhauled
 | |
|     * better tnode.dowrite
 | |
|     * some ppc stuff fixed
 | |
| 
 | |
|   Revision 1.32  2002/08/17 09:23:34  florian
 | |
|     * first part of procinfo rewrite
 | |
| 
 | |
|   Revision 1.31  2002/08/15 19:10:35  peter
 | |
|     * first things tai,tnode storing in ppu
 | |
| 
 | |
|   Revision 1.30  2002/07/20 11:57:53  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.29  2002/07/19 11:41:35  daniel
 | |
|   * State tracker work
 | |
|   * The whilen and repeatn are now completely unified into whilerepeatn. This
 | |
|     allows the state tracker to change while nodes automatically into
 | |
|     repeat nodes.
 | |
|   * Resulttypepass improvements to the notn. 'not not a' is optimized away and
 | |
|     'not(a>b)' is optimized into 'a<=b'.
 | |
|   * Resulttypepass improvements to the whilerepeatn. 'while not a' is optimized
 | |
|     by removing the notn and later switchting the true and falselabels. The
 | |
|     same is done with 'repeat until not a'.
 | |
| 
 | |
|   Revision 1.28  2002/07/14 18:00:43  daniel
 | |
|   + Added the beginning of a state tracker. This will track the values of
 | |
|     variables through procedures and optimize things away.
 | |
| 
 | |
|   Revision 1.27  2002/07/01 18:46:22  peter
 | |
|     * internal linker
 | |
|     * reorganized aasm layer
 | |
| 
 | |
|   Revision 1.26  2002/06/24 12:43:00  jonas
 | |
|     * fixed errors found with new -CR code from Peter when cycling with -O2p3r
 | |
| 
 | |
|   Revision 1.25  2002/05/18 13:34:09  peter
 | |
|     * readded missing revisions
 | |
| 
 | |
|   Revision 1.24  2002/05/16 19:46:37  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.22  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.21  2002/04/21 19:02:03  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.20  2002/04/04 19:05:57  peter
 | |
|     * removed unused units
 | |
|     * use tlocation.size in cg.a_*loc*() routines
 | |
| 
 | |
|   Revision 1.19  2002/03/31 20:26:33  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
 | |
| 
 | |
| }
 |