mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-11-04 14:39:36 +01:00 
			
		
		
		
	- complete $WARN switch parsing and add a stab to change message state - reimplement way of message hiding by -vm switch (by Dmitry Boyarintsev) (note: $WARN switch does not work at the moment since state handling is not yet implemented) git-svn-id: trunk@14809 -
		
			
				
	
	
		
			347 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			347 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
{
 | 
						|
    Copyright (c) 1998-2002 by Peter Vreman
 | 
						|
 | 
						|
    This unit implements the parsing of the switches like $I-
 | 
						|
 | 
						|
    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 switches;
 | 
						|
 | 
						|
{$i fpcdefs.inc}
 | 
						|
 | 
						|
interface
 | 
						|
 | 
						|
uses
 | 
						|
  globtype;
 | 
						|
 | 
						|
procedure HandleSwitch(switch,state:char);
 | 
						|
function CheckSwitch(switch,state:char):boolean;
 | 
						|
 | 
						|
procedure recordpendingverbosityswitch(sw: char; state: char);
 | 
						|
procedure recordpendingmessagestate(msg: longint; state: tmsgstate);
 | 
						|
procedure recordpendinglocalswitch(sw: tlocalswitch; state: char);
 | 
						|
procedure recordpendinglocalfullswitch(const switches: tlocalswitches);
 | 
						|
procedure recordpendingverbosityfullswitch(verbosity: longint);
 | 
						|
procedure recordpendingcallingswitch(const str: shortstring);
 | 
						|
procedure flushpendingswitchesstate;
 | 
						|
 | 
						|
implementation
 | 
						|
uses
 | 
						|
  systems,cpuinfo,
 | 
						|
  globals,verbose,comphook,
 | 
						|
  fmodule;
 | 
						|
 | 
						|
{****************************************************************************
 | 
						|
                          Main Switches Parsing
 | 
						|
****************************************************************************}
 | 
						|
 | 
						|
type
 | 
						|
  TSwitchType=(ignoredsw,localsw,modulesw,globalsw,illegalsw,unsupportedsw,alignsw,optimizersw,packenumsw,pentiumfdivsw);
 | 
						|
  SwitchRec=record
 | 
						|
    typesw : TSwitchType;
 | 
						|
    setsw  : byte;
 | 
						|
  end;
 | 
						|
  SwitchRecTable = array['A'..'Z'] of SwitchRec;
 | 
						|
 | 
						|
const
 | 
						|
  turboSwitchTable: SwitchRecTable =(
 | 
						|
   {A} (typesw:alignsw; setsw:ord(cs_localnone)),
 | 
						|
   {B} (typesw:localsw; setsw:ord(cs_full_boolean_eval)),
 | 
						|
   {C} (typesw:localsw; setsw:ord(cs_do_assertion)),
 | 
						|
   {D} (typesw:modulesw; setsw:ord(cs_debuginfo)),
 | 
						|
   {E} (typesw:modulesw; setsw:ord(cs_fp_emulation)),
 | 
						|
   {F} (typesw:ignoredsw; setsw:ord(cs_localnone)),
 | 
						|
   {G} (typesw:ignoredsw; setsw:ord(cs_localnone)),
 | 
						|
   {H} (typesw:localsw; setsw:ord(cs_ansistrings)),
 | 
						|
   {I} (typesw:localsw; setsw:ord(cs_check_io)),
 | 
						|
   {J} (typesw:localsw; setsw:ord(cs_typed_const_writable)),
 | 
						|
   {K} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {L} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {M} (typesw:localsw; setsw:ord(cs_generate_rtti)),
 | 
						|
   {N} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {O} (typesw:optimizersw; setsw:ord(cs_opt_none)),
 | 
						|
   {P} (typesw:modulesw; setsw:ord(cs_openstring)),
 | 
						|
   {Q} (typesw:localsw; setsw:ord(cs_check_overflow)),
 | 
						|
   {R} (typesw:localsw; setsw:ord(cs_check_range)),
 | 
						|
   {S} (typesw:localsw; setsw:ord(cs_check_stack)),
 | 
						|
   {T} (typesw:localsw; setsw:ord(cs_typed_addresses)),
 | 
						|
   {U} (typesw:pentiumfdivsw; setsw:ord(cs_localnone)),
 | 
						|
   {V} (typesw:localsw; setsw:ord(cs_strict_var_strings)),
 | 
						|
   {W} (typesw:localsw; setsw:ord(cs_generate_stackframes)),
 | 
						|
   {X} (typesw:modulesw; setsw:ord(cs_extsyntax)),
 | 
						|
   {Y} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {Z} (typesw:packenumsw; setsw:ord(cs_localnone))
 | 
						|
    );
 | 
						|
 | 
						|
 | 
						|
  macSwitchTable: SwitchRecTable =(
 | 
						|
   {A} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {B} (typesw:localsw; setsw:ord(cs_full_boolean_eval)),
 | 
						|
   {C} (typesw:localsw; setsw:ord(cs_do_assertion)),
 | 
						|
   {D} (typesw:modulesw; setsw:ord(cs_debuginfo)),
 | 
						|
   {E} (typesw:modulesw; setsw:ord(cs_fp_emulation)),
 | 
						|
   {F} (typesw:ignoredsw; setsw:ord(cs_localnone)),
 | 
						|
   {G} (typesw:ignoredsw; setsw:ord(cs_localnone)),
 | 
						|
   {H} (typesw:localsw; setsw:ord(cs_ansistrings)),
 | 
						|
   {I} (typesw:localsw; setsw:ord(cs_check_io)),
 | 
						|
   {J} (typesw:localsw; setsw:ord(cs_external_var)),
 | 
						|
   {K} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {L} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {M} (typesw:localsw; setsw:ord(cs_generate_rtti)),
 | 
						|
   {N} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {O} (typesw:optimizersw; setsw:ord(cs_opt_none)),
 | 
						|
   {P} (typesw:modulesw; setsw:ord(cs_openstring)),
 | 
						|
   {Q} (typesw:localsw; setsw:ord(cs_check_overflow)),
 | 
						|
   {R} (typesw:localsw; setsw:ord(cs_check_range)),
 | 
						|
   {S} (typesw:localsw; setsw:ord(cs_check_stack)),
 | 
						|
   {T} (typesw:localsw; setsw:ord(cs_typed_addresses)),
 | 
						|
   {U} (typesw:illegalsw; setsw:ord(cs_localnone)),
 | 
						|
   {V} (typesw:localsw; setsw:ord(cs_strict_var_strings)),
 | 
						|
   {W} (typesw:localsw; setsw:ord(cs_generate_stackframes)),
 | 
						|
   {X} (typesw:modulesw; setsw:ord(cs_extsyntax)),
 | 
						|
   {Y} (typesw:unsupportedsw; setsw:ord(cs_localnone)),
 | 
						|
   {Z} (typesw:localsw; setsw:ord(cs_externally_visible))
 | 
						|
    );
 | 
						|
 | 
						|
procedure HandleSwitch(switch,state:char);
 | 
						|
 | 
						|
var
 | 
						|
  switchTablePtr: ^SwitchRecTable;
 | 
						|
 | 
						|
begin
 | 
						|
  switch:=upcase(switch);
 | 
						|
{ Is the Switch in the letters ? }
 | 
						|
  if not ((switch in ['A'..'Z']) and (state in ['-','+'])) then
 | 
						|
   begin
 | 
						|
     Message(scan_w_illegal_switch);
 | 
						|
     exit;
 | 
						|
   end;
 | 
						|
 | 
						|
{ Select switch table }
 | 
						|
  if m_mac in current_settings.modeswitches  then
 | 
						|
    switchTablePtr:= @macSwitchTable
 | 
						|
  else
 | 
						|
    switchTablePtr:= @turboSwitchTable;
 | 
						|
 | 
						|
{ Handle the switch }
 | 
						|
   with switchTablePtr^[switch] do
 | 
						|
   begin
 | 
						|
     case typesw of
 | 
						|
       alignsw:
 | 
						|
         if state='+' then
 | 
						|
           current_settings.packrecords:=4
 | 
						|
         else
 | 
						|
           current_settings.packrecords:=1;
 | 
						|
       optimizersw :
 | 
						|
         begin
 | 
						|
           if state='+' then
 | 
						|
             current_settings.optimizerswitches:=level2optimizerswitches
 | 
						|
           else
 | 
						|
             current_settings.optimizerswitches:=[];
 | 
						|
         end;
 | 
						|
       ignoredsw :
 | 
						|
         Message1(scan_n_ignored_switch,'$'+switch);
 | 
						|
       illegalsw :
 | 
						|
         Message1(scan_w_illegal_switch,'$'+switch);
 | 
						|
       unsupportedsw :
 | 
						|
         Message1(scan_w_unsupported_switch,'$'+switch);
 | 
						|
       localsw :
 | 
						|
         recordpendinglocalswitch(tlocalswitch(setsw),state);
 | 
						|
       modulesw :
 | 
						|
         begin
 | 
						|
           if current_module.in_global then
 | 
						|
            begin
 | 
						|
{$ifndef cpufpemu}
 | 
						|
              if tmoduleswitch(setsw)=cs_fp_emulation then
 | 
						|
                begin
 | 
						|
                  Message1(scan_w_unsupported_switch_by_target,'$'+switch);
 | 
						|
                end
 | 
						|
              else
 | 
						|
{$endif cpufpemu}
 | 
						|
                begin
 | 
						|
                  if state='+' then
 | 
						|
                    include(current_settings.moduleswitches,tmoduleswitch(setsw))
 | 
						|
                  else
 | 
						|
                    begin
 | 
						|
                      { Turning off debuginfo when lineinfo is requested
 | 
						|
                        is not possible }
 | 
						|
                      if not((cs_use_lineinfo in current_settings.globalswitches) and
 | 
						|
                             (tmoduleswitch(setsw)=cs_debuginfo)) then
 | 
						|
                        exclude(current_settings.moduleswitches,tmoduleswitch(setsw));
 | 
						|
                    end;
 | 
						|
                end;
 | 
						|
            end
 | 
						|
           else
 | 
						|
            Message(scan_w_switch_is_global);
 | 
						|
         end;
 | 
						|
       globalsw :
 | 
						|
         begin
 | 
						|
           if current_module.in_global and (current_module=main_module) then
 | 
						|
            begin
 | 
						|
              if state='+' then
 | 
						|
               include(current_settings.globalswitches,tglobalswitch(setsw))
 | 
						|
              else
 | 
						|
               exclude(current_settings.globalswitches,tglobalswitch(setsw));
 | 
						|
            end
 | 
						|
           else
 | 
						|
            Message(scan_w_switch_is_global);
 | 
						|
         end;
 | 
						|
       packenumsw:
 | 
						|
         begin
 | 
						|
           if state='-' then
 | 
						|
             current_settings.packenum:=1
 | 
						|
           else
 | 
						|
             current_settings.packenum:=4;
 | 
						|
         end;
 | 
						|
       pentiumfdivsw:
 | 
						|
         begin
 | 
						|
           { Switch u- means pentium-safe fdiv off -> fpc default. We don't }
 | 
						|
           { support u+                                                     }
 | 
						|
           if state='+' then
 | 
						|
             Message1(scan_w_unsupported_switch,'$'+switch);
 | 
						|
         end;
 | 
						|
     end;
 | 
						|
   end;
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
function CheckSwitch(switch,state:char):boolean;
 | 
						|
 | 
						|
var
 | 
						|
  found : boolean;
 | 
						|
  switchTablePtr: ^SwitchRecTable;
 | 
						|
 | 
						|
begin
 | 
						|
  switch:=upcase(switch);
 | 
						|
{ Is the Switch in the letters ? }
 | 
						|
  if not ((switch in ['A'..'Z']) and (state in ['-','+'])) then
 | 
						|
   begin
 | 
						|
     Message(scan_w_illegal_switch);
 | 
						|
     CheckSwitch:=false;
 | 
						|
     exit;
 | 
						|
   end;
 | 
						|
 | 
						|
{ Select switch table }
 | 
						|
  if m_mac in current_settings.modeswitches then
 | 
						|
    switchTablePtr:= @macSwitchTable
 | 
						|
  else
 | 
						|
    switchTablePtr:= @turboSwitchTable;
 | 
						|
 | 
						|
{ Check the switch }
 | 
						|
   with switchTablePtr^[switch] do
 | 
						|
   begin
 | 
						|
     case typesw of
 | 
						|
      localsw : found:=(tlocalswitch(setsw) in current_settings.localswitches);
 | 
						|
     modulesw : found:=(tmoduleswitch(setsw) in current_settings.moduleswitches);
 | 
						|
     globalsw : found:=(tglobalswitch(setsw) in current_settings.globalswitches);
 | 
						|
     packenumsw : found := (current_settings.packenum = 4);
 | 
						|
     else
 | 
						|
      found:=false;
 | 
						|
     end;
 | 
						|
     if state='-' then
 | 
						|
      found:=not found;
 | 
						|
     CheckSwitch:=found;
 | 
						|
   end;
 | 
						|
end;
 | 
						|
 | 
						|
 | 
						|
procedure recordpendingverbosityswitch(sw: char; state: char);
 | 
						|
  begin
 | 
						|
    pendingstate.nextverbositystr:=pendingstate.nextverbositystr+sw+state;
 | 
						|
  end;
 | 
						|
 | 
						|
procedure recordpendingmessagestate(msg: longint; state: tmsgstate);
 | 
						|
  begin
 | 
						|
    { todo }
 | 
						|
  end;
 | 
						|
 | 
						|
procedure recordpendinglocalswitch(sw: tlocalswitch; state: char);
 | 
						|
  begin
 | 
						|
    if not pendingstate.localswitcheschanged then
 | 
						|
       pendingstate.nextlocalswitches:=current_settings.localswitches;
 | 
						|
    if state='-' then
 | 
						|
      exclude(pendingstate.nextlocalswitches,sw)
 | 
						|
    else if state='+' then
 | 
						|
      include(pendingstate.nextlocalswitches,sw)
 | 
						|
    else { state = '*' }
 | 
						|
      begin
 | 
						|
        if sw in init_settings.localswitches then
 | 
						|
         include(pendingstate.nextlocalswitches,sw)
 | 
						|
        else
 | 
						|
         exclude(pendingstate.nextlocalswitches,sw);
 | 
						|
      end;
 | 
						|
    pendingstate.localswitcheschanged:=true;
 | 
						|
  end;
 | 
						|
 | 
						|
 | 
						|
procedure recordpendinglocalfullswitch(const switches: tlocalswitches);
 | 
						|
  begin
 | 
						|
    pendingstate.nextlocalswitches:=switches;
 | 
						|
    pendingstate.localswitcheschanged:=true;
 | 
						|
  end;
 | 
						|
 | 
						|
 | 
						|
procedure recordpendingverbosityfullswitch(verbosity: longint);
 | 
						|
  begin
 | 
						|
    pendingstate.nextverbositystr:='';
 | 
						|
    pendingstate.nextverbosityfullswitch:=verbosity;
 | 
						|
    pendingstate.verbosityfullswitched:=true;
 | 
						|
  end;
 | 
						|
 | 
						|
procedure recordpendingcallingswitch(const str: shortstring);
 | 
						|
  begin
 | 
						|
    pendingstate.nextcallingstr:=str;
 | 
						|
  end;
 | 
						|
 | 
						|
 | 
						|
procedure flushpendingswitchesstate;
 | 
						|
  var
 | 
						|
    tmpproccal: tproccalloption;
 | 
						|
  begin
 | 
						|
    { process pending localswitches (range checking, etc) }
 | 
						|
    if pendingstate.localswitcheschanged then
 | 
						|
      begin
 | 
						|
        current_settings.localswitches:=pendingstate.nextlocalswitches;
 | 
						|
        pendingstate.localswitcheschanged:=false;
 | 
						|
      end;
 | 
						|
    { process pending verbosity changes (warnings on, etc) }
 | 
						|
    if pendingstate.verbosityfullswitched then
 | 
						|
      begin
 | 
						|
        status.verbosity:=pendingstate.nextverbosityfullswitch;
 | 
						|
        pendingstate.verbosityfullswitched:=false;
 | 
						|
      end;
 | 
						|
    if pendingstate.nextverbositystr<>'' then
 | 
						|
      begin
 | 
						|
        setverbosity(pendingstate.nextverbositystr);
 | 
						|
        pendingstate.nextverbositystr:='';
 | 
						|
      end;
 | 
						|
    { process pending calling convention changes (calling x) }
 | 
						|
    if pendingstate.nextcallingstr<>'' then
 | 
						|
      begin
 | 
						|
        if not SetAktProcCall(pendingstate.nextcallingstr,tmpproccal) then
 | 
						|
          Message1(parser_w_unknown_proc_directive_ignored,pendingstate.nextcallingstr)
 | 
						|
        else if not(tmpproccal in supported_calling_conventions) then
 | 
						|
          Message1(parser_e_illegal_calling_convention,pendingstate.nextcallingstr)
 | 
						|
        else
 | 
						|
          current_settings.defproccall:=tmpproccal;
 | 
						|
        pendingstate.nextcallingstr:='';
 | 
						|
      end;
 | 
						|
  end;
 | 
						|
 | 
						|
 | 
						|
end.
 |