mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 04:31:30 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			233 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			233 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     Copyright (c) 1998-2002 by Florian Klaempfl
 | |
| 
 | |
|     This unit handles the codegeneration pass
 | |
| 
 | |
|     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 pass_2;
 | |
| 
 | |
| {$i fpcdefs.inc}
 | |
| 
 | |
| interface
 | |
| 
 | |
| uses
 | |
|    node;
 | |
| 
 | |
|     type
 | |
|        tenumflowcontrol = (
 | |
|          fc_exit,
 | |
|          fc_break,
 | |
|          fc_continue,
 | |
|          fc_inflowcontrol,
 | |
|          fc_gotolabel,
 | |
|          { the left side of an expression is already handled, so we are
 | |
|            not allowed to do ssl }
 | |
|          fc_lefthandled);
 | |
| 
 | |
|        tflowcontrol = set of tenumflowcontrol;
 | |
| 
 | |
|     var
 | |
|        flowcontrol : tflowcontrol;
 | |
| 
 | |
| { produces the actual code }
 | |
| function do_secondpass(var p : tnode) : boolean;
 | |
| procedure secondpass(var p : tnode);
 | |
| 
 | |
| 
 | |
| implementation
 | |
| 
 | |
|    uses
 | |
| {$ifdef EXTDEBUG}
 | |
|      cutils,
 | |
| {$endif}
 | |
|      globtype,systems,verbose,
 | |
|      globals,
 | |
|      paramgr,
 | |
|      aasmtai,aasmdata,
 | |
|      cgbase,
 | |
|      nflw,cgobj;
 | |
| 
 | |
| {*****************************************************************************
 | |
|                               SecondPass
 | |
| *****************************************************************************}
 | |
| 
 | |
| {$ifdef EXTDEBUG}
 | |
|      var
 | |
|        secondprefix : string;
 | |
| 
 | |
|      procedure logsecond(ht:tnodetype; entry: boolean);
 | |
|        const
 | |
|          secondnames: array[tnodetype] of string[13] =
 | |
|             ('<emptynode>',
 | |
|              'add-addn',  {addn}
 | |
|              'add-muln',  {muln}
 | |
|              'add-subn',  {subn}
 | |
|              'moddiv-divn',      {divn}
 | |
|              'add-symdifn',      {symdifn}
 | |
|              'moddiv-modn',      {modn}
 | |
|              'assignment',  {assignn}
 | |
|              'load',        {loadn}
 | |
|              'nothing-range',     {range}
 | |
|              'add-ltn',  {ltn}
 | |
|              'add-lten',  {lten}
 | |
|              'add-gtn',  {gtn}
 | |
|              'add-gten',  {gten}
 | |
|              'add-equaln',  {equaln}
 | |
|              'add-unequaln',  {unequaln}
 | |
|              'in',    {inn}
 | |
|              'add-orn',  {orn}
 | |
|              'add-xorn',  {xorn}
 | |
|              'shlshr-shrn',      {shrn}
 | |
|              'shlshr-shln',      {shln}
 | |
|              'add-slashn',  {slashn}
 | |
|              'add-andn',  {andn}
 | |
|              'subscriptn',  {subscriptn}
 | |
|              'deref',       {derefn}
 | |
|              'addr',        {addrn}
 | |
|              'ordconst',    {ordconstn}
 | |
|              'typeconv',    {typeconvn}
 | |
|              'calln',       {calln}
 | |
|              'noth-callpar',{callparan}
 | |
|              'realconst',   {realconstn}
 | |
|              'unaryminus',  {unaryminusn}
 | |
|              'unaryplus',   {unaryplusn}
 | |
|              'asm',         {asmn}
 | |
|              'vecn',        {vecn}
 | |
|              'pointerconst',{pointerconstn}
 | |
|              'stringconst', {stringconstn}
 | |
|              'not',         {notn}
 | |
|              'inline',      {inlinen}
 | |
|              'niln',        {niln}
 | |
|              'error',       {errorn}
 | |
|              'nothing-typen',     {typen}
 | |
|              'setelement',  {setelementn}
 | |
|              'setconst',    {setconstn}
 | |
|              'blockn',      {blockn}
 | |
|              'statement',   {statementn}
 | |
|              'ifn',         {ifn}
 | |
|              'breakn',      {breakn}
 | |
|              'continuen',   {continuen}
 | |
|              'while_repeat', {whilerepeatn}
 | |
|              'for',         {forn}
 | |
|              'exitn',       {exitn}
 | |
|              'with',        {withn}
 | |
|              'case',        {casen}
 | |
|              'label',       {labeln}
 | |
|              'goto',        {goton}
 | |
|              'tryexcept',   {tryexceptn}
 | |
|              'raise',       {raisen}
 | |
|              'tryfinally',  {tryfinallyn}
 | |
|              'on',    {onn}
 | |
|              'is',    {isn}
 | |
|              'as',    {asn}
 | |
|              'add-starstar',  {starstarn}
 | |
|              'arrayconstruc', {arrayconstructn}
 | |
|              'noth-arrcnstr',     {arrayconstructrangen}
 | |
|              'tempcreaten',
 | |
|              'temprefn',
 | |
|              'tempdeleten',
 | |
|              'addoptn',
 | |
|              'nothing-nothg',     {nothingn}
 | |
|              'loadvmt',      {loadvmtn}
 | |
|              'guidconstn',
 | |
|              'rttin',
 | |
|              'loadparentfpn',
 | |
|              'dataconstn',
 | |
|              'objselectorn',
 | |
|              'objcprotocoln'
 | |
|              );
 | |
|       var
 | |
|         p: pchar;
 | |
|       begin
 | |
|         if entry then
 | |
|           begin
 | |
|             secondprefix:=secondprefix+' ';
 | |
|             p := strpnew(secondprefix+'second '+secondnames[ht]+' (entry)')
 | |
|           end
 | |
|         else
 | |
|           begin
 | |
|             p := strpnew(secondprefix+'second '+secondnames[ht]+' (exit)');
 | |
|             delete(secondprefix,length(secondprefix),1);
 | |
|           end;
 | |
|         current_asmdata.CurrAsmList.concat(tai_comment.create(p));
 | |
|       end;
 | |
| {$endif EXTDEBUG}
 | |
| 
 | |
|      procedure secondpass(var p : tnode);
 | |
|       var
 | |
|          oldcodegenerror  : boolean;
 | |
|          oldlocalswitches : tlocalswitches;
 | |
|          oldpos    : tfileposinfo;
 | |
|       begin
 | |
|          if not assigned(p) then
 | |
|           internalerror(200208221);
 | |
|          if not(nf_error in p.flags) then
 | |
|           begin
 | |
|             oldcodegenerror:=codegenerror;
 | |
|             oldlocalswitches:=current_settings.localswitches;
 | |
|             oldpos:=current_filepos;
 | |
|             current_filepos:=p.fileinfo;
 | |
|             current_settings.localswitches:=p.localswitches;
 | |
|             codegenerror:=false;
 | |
| {$ifdef EXTDEBUG}
 | |
|             if (p.expectloc=LOC_INVALID) then
 | |
|               Comment(V_Warning,'ExpectLoc is not set before secondpass: '+nodetype2str[p.nodetype]);
 | |
|             if (p.location.loc<>LOC_INVALID) then
 | |
|               Comment(V_Warning,'Location.Loc is already set before secondpass: '+nodetype2str[p.nodetype]);
 | |
|             if (cs_asm_nodes in current_settings.globalswitches) then
 | |
|               logsecond(p.nodetype,true);
 | |
| {$endif EXTDEBUG}
 | |
|             p.pass_generate_code;
 | |
| {$ifdef EXTDEBUG}
 | |
|             if (cs_asm_nodes in current_settings.globalswitches) then
 | |
|               logsecond(p.nodetype,false);
 | |
|             if (not codegenerror) then
 | |
|              begin
 | |
|                if (p.location.loc<>p.expectloc) then
 | |
|                  Comment(V_Warning,'Location not equal to expectloc: '+nodetype2str[p.nodetype]);
 | |
|                if (p.location.loc=LOC_INVALID) then
 | |
|                  Comment(V_Warning,'Location not set in secondpass: '+nodetype2str[p.nodetype]);
 | |
|              end;
 | |
| {$endif EXTDEBUG}
 | |
|             if codegenerror then
 | |
|               include(p.flags,nf_error);
 | |
|             codegenerror:=codegenerror or oldcodegenerror;
 | |
|             current_settings.localswitches:=oldlocalswitches;
 | |
|             current_filepos:=oldpos;
 | |
|           end
 | |
|          else
 | |
|            codegenerror:=true;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function do_secondpass(var p : tnode) : boolean;
 | |
|       begin
 | |
|          { current_asmdata.CurrAsmList must be empty }
 | |
|          if not current_asmdata.CurrAsmList.empty then
 | |
|            internalerror(200405201);
 | |
| 
 | |
|          { clear errors before starting }
 | |
|          codegenerror:=false;
 | |
|          if not(nf_error in p.flags) then
 | |
|            secondpass(p);
 | |
|          do_secondpass:=codegenerror;
 | |
|       end;
 | |
| 
 | |
| 
 | |
| end.
 | 
