| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 | {    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}interfaceuses   node;    type       tenumflowcontrol = (         fc_exit,         fc_break,         fc_continue,         fc_inflowcontrol,         fc_gotolabel,         { in block that has an exception handler associated with it           (try..except, try..finally, exception block of try..except, ... }         fc_catching_exceptions,         { in try block of try..finally and target uses specific unwinding }         fc_unwind_exit,         fc_unwind_loop,         { the left side of an expression is already handled, so we are           not allowed to do ssl }         fc_lefthandled,         { in block where the exit statement jumps to an extra code instead of           immediately finishing execution of the current routine. }         fc_no_direct_exit);       tflowcontrol = set of tenumflowcontrol;    var       flowcontrol : tflowcontrol;{ produces the actual code }function do_secondpass(var p : tnode) : boolean;procedure secondpass(p : tnode);implementation   uses     cutils,     globtype,verbose,     globals,     aasmdata,     cgobj{$ifdef EXTDEBUG}     ,cgbase     ,aasmtai{$endif}     ;{*****************************************************************************                              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}             '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',             'objselectorn',             'objcprotocoln',             'specializen',             'finalizetemps'             );      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(p : tnode);      var         oldcodegenerror  : boolean;         oldlocalswitches : tlocalswitches;         oldpos    : tfileposinfo;         oldexecutionweight : longint;      begin         if not assigned(p) then          internalerror(200208221);         if not(nf_error in p.flags) then          begin            { The error flag takes precedence over the 'do not execute' flag,              as its assumed the node tree isn't tenable beyond this point }            if (nf_do_not_execute in p.flags) then              InternalError(2022112402);            oldcodegenerror:=codegenerror;            oldlocalswitches:=current_settings.localswitches;            oldpos:=current_filepos;            current_filepos:=p.fileinfo;            current_settings.localswitches:=p.localswitches;            codegenerror:=false;            oldexecutionweight:=cg.executionweight;            if assigned(p.optinfo) then              cg.executionweight:=min(p.optinfo^.executionweight,high(cg.executionweight))            else              cg.executionweight:=100;{$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                 begin                   if ((p.location.loc=loc_register) and (p.expectloc=loc_cregister))                      or ((p.location.loc=loc_fpuregister) and (p.expectloc=loc_cfpuregister)){$ifdef SUPPORT_MMX}                      or ((p.location.loc=loc_mmxregister) and (p.expectloc=loc_cmmxregister)){$endif SUPPORT_MMX}                      or ((p.location.loc=loc_reference) and (p.expectloc=loc_creference))                      or ((p.location.loc=loc_void) and (p.nodetype = calln)) then                     Comment(V_Note,'Location ('+tcgloc2str[p.location.loc]+') not equal to expectloc ('+tcgloc2str[p.expectloc]+'): '+nodetype2str[p.nodetype])                   else                     Comment(V_Warning,'Location ('+tcgloc2str[p.location.loc]+') not equal to expectloc ('+tcgloc2str[p.expectloc]+'): '+nodetype2str[p.nodetype]);                 end;               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;            cg.executionweight:=oldexecutionweight;          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.
 |