| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829 | {    Copyright (c) 2008 by Jonas Maebe    Whole program optimisation information collection base class    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 wpobase;{$i fpcdefs.inc}interfaceuses  globtype,  cclasses,  symtype;type  { the types of available whole program optimization }  twpotype = (wpo_devirtualization_context_insensitive,wpo_live_symbol_information);const  wpo2str: array[twpotype] of string[16] = ('devirtualization','symbol liveness');type  { ************************************************************************* }  { ******************** General base classes/interfaces ******************** }  { ************************************************************************* }  { interface to reading a section from a file with wpo info }  twposectionreaderintf = interface    ['{51BE3F89-C9C5-4965-9C83-AE7490C92E3E}']    function sectiongetnextline(out s: string): boolean;  end;  { interface to writing sections to a file with wpoinfo }  twposectionwriterintf = interface    ['{C056F0DD-62B1-4612-86C7-2D39944C4437}']    procedure startsection(const name: string);    procedure sectionputline(const s: string);  end;  { base class for wpo information stores }  { twpocomponentbase }  twpocomponentbase = class   public    constructor create; reintroduce; virtual;    { type of whole program optimization information collected/provided by      this class    }    class function getwpotype: twpotype; virtual; abstract;    { whole program optimizations for which this class generates information }    class function generatesinfoforwposwitches: twpoptimizerswitches; virtual; abstract;    { whole program optimizations performed by this class }    class function performswpoforswitches: twpoptimizerswitches; virtual; abstract;    { returns the name of the section parsed by this class }    class function sectionname: shortstring; virtual; abstract;    { checks whether the compiler options are compatible with this      optimization (default: don't check anything)    }    class procedure checkoptions; virtual;    { loads the information pertinent to this whole program optimization from      the current section being processed by reader    }    procedure loadfromwpofilesection(reader: twposectionreaderintf); virtual; abstract;    { stores the information of this component to a file in a format that can      be loaded again using loadfromwpofilesection()    }    procedure storewpofilesection(writer: twposectionwriterintf); virtual; abstract;    { extracts the information pertinent to this whole program optimization      from the current compiler state (loaded units, ...)    }    procedure constructfromcompilerstate; virtual; abstract;  end;  twpocomponentbaseclass = class of twpocomponentbase;  { forward declaration of overall wpo info manager class }  twpoinfomanagerbase = class;  { ************************************************************************* }  { ** Information created per unit for use during subsequent compilation *** }  { ************************************************************************* }  { information about called vmt entries for a class }  tcalledvmtentries = class   protected    { the class }    fobjdef: tdef;    fobjdefderef: tderef;    { the vmt entries }    fcalledentries: tbitset;   public    constructor create(_objdef: tdef; nentries: longint);    constructor ppuload(ppufile: tcompilerppufile);    destructor destroy; override;    procedure ppuwrite(ppufile: tcompilerppufile);    procedure buildderef;    procedure buildderefimpl;    procedure deref;    procedure derefimpl;    property objdef: tdef read fobjdef write fobjdef;    property objdefderef: tderef read fobjdefderef write fobjdefderef;    property calledentries: tbitset read fcalledentries write fcalledentries;  end;  { base class of information collected per unit. Still needs to be    generalised for different kinds of wpo information, currently specific    to devirtualization.  }  tunitwpoinfobase = class   protected    { created object types }    fcreatedobjtypes: tfpobjectlist;    { objectdefs pointed to by created classrefdefs }    fcreatedclassrefobjtypes: tfpobjectlist;    { objtypes potentially instantiated by fcreatedclassrefobjtypes      (objdectdefs pointed to by classrefdefs that are       passed as a regular parameter, loaded in a variable, ...       so they can end up in a classrefdef var and be instantiated)    }    fmaybecreatedbyclassrefdeftypes: tfpobjectlist;    { called virtual methods for all classes (hashed by mangled classname,      entries bitmaps indicating which vmt entries per class are called --      tcalledvmtentries)    }    fcalledvmtentries: tfphashlist;   public    constructor create; reintroduce; virtual;    destructor destroy; override;    property createdobjtypes: tfpobjectlist read fcreatedobjtypes;    property createdclassrefobjtypes: tfpobjectlist read fcreatedclassrefobjtypes;    property maybecreatedbyclassrefdeftypes: tfpobjectlist read fmaybecreatedbyclassrefdeftypes;    property calledvmtentries: tfphashlist read fcalledvmtentries;    procedure addcreatedobjtype(def: tdef);    procedure addcreatedobjtypeforclassref(def: tdef);    procedure addmaybecreatedbyclassref(def: tdef);    procedure addcalledvmtentry(def: tdef; index: longint);    { resets the "I've been registered with wpo" flags for all defs in the      above lists }    procedure resetdefs;  end;  { ************************************************************************* }  { **** Total information created for use during subsequent compilation **** }  { ************************************************************************* }  { class to create a file with wpo information }  { tavailablewpofilewriter }  twpofilewriter = class(tobject,twposectionwriterintf)   private    { array of class *instances* that wish to be written out to the      whole program optimization feedback file    }    fsectioncontents: tfpobjectlist;    ffilename: tcmdstr;    foutputfile: text;   public    constructor create(const fn: tcmdstr);    destructor destroy; override;    procedure writefile;    { starts a new section with name "name" }    procedure startsection(const name: string);    { writes s to the wpo file }    procedure sectionputline(const s: string);    { register a component instance that needs to be written      to the wpo feedback file    }    procedure registerwpocomponent(component: twpocomponentbase);  end;  { ************************************************************************* }  { ************ Information for use during current compilation ************* }  { ************************************************************************* }  { class to read a file with wpo information }  twpofilereader = class(tobject,twposectionreaderintf)   private    ffilename: tcmdstr;    flinenr: longint;    finputfile: text;    fcurline: string;    fusecurline: boolean;    { destination for the read information }    fdest: twpoinfomanagerbase;    function getnextnoncommentline(out s: string): boolean;   public     constructor create(const fn: tcmdstr; dest: twpoinfomanagerbase);     destructor destroy; override;     { processes the wpo info in the file }     procedure processfile;     { returns next line of the current section in s, and false if no more       lines in the current section     }     function sectiongetnextline(out s: string): boolean;  end;  { ************************************************************************* }  { ******* Specific kinds of whole program optimization components ********* }  { ************************************************************************* }  { method devirtualisation }  twpodevirtualisationhandler = class(twpocomponentbase)    { checks whether procdef (a procdef for a virtual method) can be replaced with      a static call when it's called as objdef.procdef, and if so returns the      mangled name in staticname.    }    function staticnameforcallingvirtualmethod(objdef, procdef: tdef; out staticname: string): boolean; virtual; abstract;    { checks whether procdef (a procdef for a virtual method) can be replaced with      a different procname in the vmt of objdef, and if so returns the new      mangledname in staticname    }    function staticnameforvmtentry(objdef, procdef: tdef; out staticname: string): boolean; virtual; abstract;  end;  twpodeadcodehandler = class(twpocomponentbase)    { checks whether a mangledname was removed as dead code from the final      binary (WARNING: must *not* be called for functions marked as inline,      since if all call sites are inlined, it won't appear in the final      binary but nevertheless is still necessary!)    }    function symbolinfinalbinary(const s: shortstring): boolean; virtual; abstract;  end;  { ************************************************************************* }  { ************ Collection of all instances of wpo components ************** }  { ************************************************************************* }  { class doing all the bookkeeping for everything  }  twpoinfomanagerbase = class   private    { array of classrefs of handler classes for the various kinds of whole      program optimizations that we support    }    fwpocomponents: tfphashlist;    freader: twpofilereader;    fwriter: twpofilewriter;   public    { instances of the various optimizers/information collectors (for      information used during this compilation)    }    wpoinfouse: array[twpotype] of twpocomponentbase;    { register a whole program optimization class type }    procedure registerwpocomponentclass(wpocomponent: twpocomponentbaseclass);    { get the program optimization class type that can parse the contents      of the section with name "secname" in the wpo feedback file    }    function gethandlerforsection(const secname: string): twpocomponentbaseclass;    { tell all instantiated wpo component classes to collect the information      from the global compiler state that they need (done at the very end of      the compilation process)    }    procedure extractwpoinfofromprogram;    { set the name of the feedback file from which all whole-program information      to be used during the current compilation will be read    }    procedure setwpoinputfile(const fn: tcmdstr);    { set the name of the feedback file to which all whole-program information      collected during the current compilation will be written    }    procedure setwpooutputfile(const fn: tcmdstr);    { check whether the specified wpo options (-FW/-Fw/-OW/-Ow) are complete      and sensical, and parse the wpo feedback file specified with      setwpoinputfile    }    procedure parseandcheckwpoinfo;    { routines accessing the optimizer information }    { 1) devirtualization at the symbol name level }    function can_be_devirtualized(objdef, procdef: tdef; out name: shortstring): boolean; virtual; abstract;    { 2) optimal replacement method name in vmt }    function optimized_name_for_vmt(objdef, procdef: tdef; out name: shortstring): boolean; virtual; abstract;    { 3) does a symbol appear in the final binary (i.e., not removed by dead code stripping/smart linking).        WARNING: do *not* call for inline functions/procedures/methods/...    }    function symbol_live(const name: shortstring): boolean; virtual; abstract;    constructor create; reintroduce;    destructor destroy; override;  end;  var    wpoinfomanager: twpoinfomanagerbase;implementation  uses    globals,    cutils,    sysutils,    symdef,    verbose;  { tcreatedwpoinfobase }  constructor tunitwpoinfobase.create;    begin      fcreatedobjtypes:=tfpobjectlist.create(false);      fcreatedclassrefobjtypes:=tfpobjectlist.create(false);      fmaybecreatedbyclassrefdeftypes:=tfpobjectlist.create(false);      fcalledvmtentries:=tfphashlist.create;    end;  destructor tunitwpoinfobase.destroy;    var      i: longint;    begin      { don't call resetdefs here, because the defs may have been freed        already }      fcreatedobjtypes.free;      fcreatedobjtypes:=nil;      fcreatedclassrefobjtypes.free;      fcreatedclassrefobjtypes:=nil;      fmaybecreatedbyclassrefdeftypes.free;      fmaybecreatedbyclassrefdeftypes:=nil;      { may not be assigned in case the info was loaded from a ppu and we        are not generating a wpo feedback file (see tunitwpoinfo.ppuload)      }      if assigned(fcalledvmtentries) then        begin          for i:=0 to fcalledvmtentries.count-1 do            tcalledvmtentries(fcalledvmtentries[i]).free;          fcalledvmtentries.free;          fcalledvmtentries:=nil;        end;      inherited destroy;    end;          procedure tunitwpoinfobase.resetdefs;    var      i: ptrint;    begin      if assigned(fcreatedobjtypes) then        for i:=0 to fcreatedobjtypes.count-1 do          tobjectdef(fcreatedobjtypes[i]).created_in_current_module:=false;      if assigned(fcreatedclassrefobjtypes) then        for i:=0 to fcreatedclassrefobjtypes.count-1 do          tobjectdef(fcreatedclassrefobjtypes[i]).classref_created_in_current_module:=false;      if assigned(fmaybecreatedbyclassrefdeftypes) then        for i:=0 to fmaybecreatedbyclassrefdeftypes.count-1 do          tobjectdef(fmaybecreatedbyclassrefdeftypes[i]).maybe_created_in_current_module:=false;    end;  procedure tunitwpoinfobase.addcreatedobjtype(def: tdef);    begin      fcreatedobjtypes.add(def);    end;  procedure tunitwpoinfobase.addcreatedobjtypeforclassref(def: tdef);    begin      fcreatedclassrefobjtypes.add(def);    end;  procedure tunitwpoinfobase.addmaybecreatedbyclassref(def: tdef);    begin      fmaybecreatedbyclassrefdeftypes.add(def);    end;  procedure tunitwpoinfobase.addcalledvmtentry(def: tdef; index: longint);    var      entries: tcalledvmtentries;      key: shortstring;    begin      key:=tobjectdef(def).vmt_mangledname;      entries:=tcalledvmtentries(fcalledvmtentries.find(key));      if not assigned(entries) then        begin          entries:=tcalledvmtentries.create(def,tobjectdef(def).vmtentries.count);          fcalledvmtentries.add(key,entries);        end;      entries.calledentries.include(index);    end;  { twpofilereader }  function twpofilereader.getnextnoncommentline(out s: string):    boolean;    begin      if (fusecurline) then        begin          s:=fcurline;          fusecurline:=false;          result:=true;          exit;        end;      repeat        readln(finputfile,s);        if (s='') and           eof(finputfile) then          begin            result:=false;            exit;          end;        inc(flinenr);      until (s='') or            (s[1]<>'#');      result:=true;    end;  constructor twpofilereader.create(const fn: tcmdstr; dest: twpoinfomanagerbase);    begin      if not FileExists(fn) or         { FileExists also returns true for directories }         DirectoryExists(fn) then        begin          cgmessage1(wpo_cant_find_file,fn);          exit;        end;      assign(finputfile,fn);      ffilename:=fn;      fdest:=dest;    end;  destructor twpofilereader.destroy;    begin      inherited destroy;    end;  procedure twpofilereader.processfile;    var      sectionhandler: twpocomponentbaseclass;      i: longint;      wpotype: twpotype;      s,      sectionname: string;    begin      cgmessage1(wpo_begin_processing,ffilename);      reset(finputfile);      flinenr:=0;      while getnextnoncommentline(s) do        begin          if (s='') then            continue;          { format: "% sectionname" }          if (s[1]<>'%') then            begin              cgmessage2(wpo_expected_section,tostr(flinenr),s);              break;            end;          for i:=2 to length(s) do            if (s[i]<>' ') then              break;          sectionname:=copy(s,i,255);          { find handler for section and process }          sectionhandler:=fdest.gethandlerforsection(sectionname);          if assigned(sectionhandler) then            begin              wpotype:=sectionhandler.getwpotype;              cgmessage2(wpo_found_section,sectionname,wpo2str[wpotype]);              { do we need this information? }              if ((sectionhandler.performswpoforswitches * init_settings.dowpoptimizerswitches) <> []) then                begin                  { did some other section already generate this type of information? }                  if assigned(fdest.wpoinfouse[wpotype]) then                    begin                      cgmessage2(wpo_duplicate_wpotype,wpo2str[wpotype],sectionname);                      fdest.wpoinfouse[wpotype].free;                    end;                  { process the section }                  fdest.wpoinfouse[wpotype]:=sectionhandler.create;                  twpocomponentbase(fdest.wpoinfouse[wpotype]).loadfromwpofilesection(self);                end              else                begin                  cgmessage1(wpo_skipping_unnecessary_section,sectionname);                  { skip the current section }                  while sectiongetnextline(s) do                    ;                end;            end          else            begin              cgmessage1(wpo_no_section_handler,sectionname);              { skip the current section }              while sectiongetnextline(s) do                ;            end;        end;      close(finputfile);      cgmessage1(wpo_end_processing,ffilename);    end;  function twpofilereader.sectiongetnextline(out s: string): boolean;    begin      result:=getnextnoncommentline(s);      if not result then        exit;      { start of new section? }      if (s<>'') and         (s[1]='%') then        begin          { keep read line for next call to getnextnoncommentline() }          fcurline:=s;          fusecurline:=true;          result:=false;        end;    end;  { twpocomponentbase }  constructor twpocomponentbase.create;    begin      { do nothing }    end;  class procedure twpocomponentbase.checkoptions;    begin      { do nothing }    end;  { twpofilewriter }  constructor twpofilewriter.create(const fn: tcmdstr);    begin      assign(foutputfile,fn);      ffilename:=fn;      fsectioncontents:=tfpobjectlist.create(true);    end;  destructor twpofilewriter.destroy;    begin      fsectioncontents.free;      inherited destroy;    end;  procedure twpofilewriter.writefile;    var      i: longint;    begin{$i-}      rewrite(foutputfile);{$i+}      if (ioresult <> 0) then        begin          cgmessage1(wpo_cant_create_feedback_file,ffilename);          exit;        end;      for i:=0 to fsectioncontents.count-1 do        twpocomponentbase(fsectioncontents[i]).storewpofilesection(self);      close(foutputfile);    end;  procedure twpofilewriter.startsection(const name: string);    begin      writeln(foutputfile,'% ',name);    end;  procedure twpofilewriter.sectionputline(const s: string);    begin      writeln(foutputfile,s);    end;  procedure twpofilewriter.registerwpocomponent(    component: twpocomponentbase);    begin      fsectioncontents.add(component);    end;{ twpoinfomanagerbase }  procedure twpoinfomanagerbase.registerwpocomponentclass(wpocomponent: twpocomponentbaseclass);    begin      fwpocomponents.add(wpocomponent.sectionname,wpocomponent);    end;  function twpoinfomanagerbase.gethandlerforsection(const secname: string      ): twpocomponentbaseclass;    begin      result:=twpocomponentbaseclass(fwpocomponents.find(secname));    end;  procedure twpoinfomanagerbase.setwpoinputfile(const fn: tcmdstr);    begin      freader:=twpofilereader.create(fn,self);    end;  procedure twpoinfomanagerbase.setwpooutputfile(const fn: tcmdstr);    begin      fwriter:=twpofilewriter.create(fn);    end;  procedure twpoinfomanagerbase.parseandcheckwpoinfo;    var      i: longint;    begin      { error if we don't have to optimize yet have an input feedback file }      if (init_settings.dowpoptimizerswitches=[]) and         assigned(freader) then        begin          cgmessage(wpo_input_without_info_use);          exit;        end;      { error if we have to optimize yet don't have an input feedback file }      if (init_settings.dowpoptimizerswitches<>[]) and         not assigned(freader) then        begin          cgmessage(wpo_no_input_specified);          exit;        end;      { if we have to generate wpo information, check that a file has been        specified and that we have something to write to it      }      if (init_settings.genwpoptimizerswitches<>[]) and         not assigned(fwriter) then        begin          cgmessage(wpo_no_output_specified);          exit;        end;      if (init_settings.genwpoptimizerswitches=[]) and         assigned(fwriter) then        begin          cgmessage(wpo_output_without_info_gen);          exit;        end;      { now read the input feedback file }      if assigned(freader) then        begin          freader.processfile;          freader.free;          freader:=nil;        end;      { and for each specified optimization check whether the input feedback        file contained the necessary information      }      if (([cs_wpo_devirtualize_calls,cs_wpo_optimize_vmts] * init_settings.dowpoptimizerswitches) <> []) and         not assigned(wpoinfouse[wpo_devirtualization_context_insensitive]) then        begin          cgmessage1(wpo_not_enough_info,wpo2str[wpo_devirtualization_context_insensitive]);          exit;        end;      if (cs_wpo_symbol_liveness in init_settings.dowpoptimizerswitches) and         not assigned(wpoinfouse[wpo_live_symbol_information]) then        begin          cgmessage1(wpo_not_enough_info,wpo2str[wpo_live_symbol_information]);          exit;        end;      { perform pre-checking to ensure there are no known incompatibilities between        the selected optimizations and other switches      }      for i:=0 to fwpocomponents.count-1 do        if (twpocomponentbaseclass(fwpocomponents[i]).generatesinfoforwposwitches*init_settings.genwpoptimizerswitches)<>[] then          twpocomponentbaseclass(fwpocomponents[i]).checkoptions    end;  procedure twpoinfomanagerbase.extractwpoinfofromprogram;    var      i: longint;      info: twpocomponentbase;    begin      { if don't have to write anything, fwriter has not been created }      if not assigned(fwriter) then        exit;      { let all wpo components gather the necessary info from the compiler state }      for i:=0 to fwpocomponents.count-1 do        if (twpocomponentbaseclass(fwpocomponents[i]).generatesinfoforwposwitches*current_settings.genwpoptimizerswitches)<>[] then          begin            info:=twpocomponentbaseclass(fwpocomponents[i]).create;            info.constructfromcompilerstate;            fwriter.registerwpocomponent(info);          end;      { and write their info to disk }      fwriter.writefile;      fwriter.free;      fwriter:=nil;    end;  constructor twpoinfomanagerbase.create;    begin      inherited create;      fwpocomponents:=tfphashlist.create;    end;  destructor twpoinfomanagerbase.destroy;    var      i: twpotype;    begin      freader.free;      freader:=nil;      fwriter.free;      fwriter:=nil;      fwpocomponents.free;      fwpocomponents:=nil;      for i:=low(wpoinfouse) to high(wpoinfouse) do        if assigned(wpoinfouse[i]) then          wpoinfouse[i].free;      inherited destroy;    end;  { tcalledvmtentries }  constructor tcalledvmtentries.create(_objdef: tdef; nentries: longint);    begin      objdef:=_objdef;      calledentries:=tbitset.create(nentries);    end;  constructor tcalledvmtentries.ppuload(ppufile: tcompilerppufile);    var      len: longint;    begin      ppufile.getderef(fobjdefderef);      len:=ppufile.getlongint;      calledentries:=tbitset.create_bytesize(len);      if (len <> calledentries.datasize) then        internalerror(2009060301);      ppufile.readdata(calledentries.data^,len);    end;  destructor tcalledvmtentries.destroy;    begin      fcalledentries.free;      inherited destroy;    end;  procedure tcalledvmtentries.ppuwrite(ppufile: tcompilerppufile);    begin      ppufile.putderef(objdefderef);      ppufile.putlongint(calledentries.datasize);      ppufile.putdata(calledentries.data^,calledentries.datasize);    end;  procedure tcalledvmtentries.buildderef;    begin      objdefderef.build(objdef);    end;  procedure tcalledvmtentries.buildderefimpl;    begin    end;  procedure tcalledvmtentries.deref;    begin      objdef:=tdef(objdefderef.resolve);    end;  procedure tcalledvmtentries.derefimpl;    begin    end;end.
 |