| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775 | {    Copyright (c) 1998-2002 by Florian Klaempfl    Generates nodes for routines that need compiler support    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 pinline;{$i fpcdefs.inc}interface    uses      symtype,      node,      globals,      cpuinfo;    function new_dispose_statement(is_new:boolean) : tnode;    function new_function : tnode;    function inline_setlength : tnode;    function inline_initialize : tnode;    function inline_finalize : tnode;    function inline_copy : tnode;implementation    uses       { common }       cutils,       { global }       globtype,tokens,verbose,       systems,       { symtable }       symbase,symconst,symdef,symsym,symtable,defutil,       { pass 1 }       pass_1,htypechk,       nmat,nadd,ncal,nmem,nset,ncnv,ninl,ncon,nld,nflw,nbas,nutils,       { parser }       scanner,       pbase,pexpr,       { codegen }       cgbase       ;    function new_dispose_statement(is_new:boolean) : tnode;      var        newstatement : tstatementnode;        temp         : ttempcreatenode;        para         : tcallparanode;        p,p2     : tnode;        again    : boolean; { dummy for do_proc_call }        destructorname : TIDString;        sym      : tsym;        classh   : tobjectdef;        callflag : tcallnodeflag;        destructorpos,        storepos : tfileposinfo;      begin        consume(_LKLAMMER);        p:=comp_expr(true);        { calc return type }        if is_new then          set_varstate(p,vs_written,[])        else          set_varstate(p,vs_readwritten,[vsf_must_be_valid]);        if (m_mac in current_settings.modeswitches) and           is_class(p.resultdef) then          begin            classh:=tobjectdef(p.resultdef);            { make sure we call ObjPas.TObject.Create/Free and not a random }            { create/free method in a macpas descendent object (since those }            { are not supposed to be called automatically when you call     }            { new/dispose)                                                  }            while assigned(classh.childof) do              classh := classh.childof;            if is_new then              begin                sym:=search_class_member(classh,'CREATE');                p2 := cloadvmtaddrnode.create(ctypenode.create(p.resultdef));              end            else              begin                sym:=search_class_member(classh,'FREE');                p2 := p;             end;            if not(assigned(sym)) then              begin                 p.free;                 if is_new then                   p2.free;                 new_dispose_statement := cerrornode.create;                 consume_all_until(_RKLAMMER);                 consume(_RKLAMMER);                 exit;              end;            do_member_read(classh,false,sym,p2,again,[]);            { we need the real called method }            do_typecheckpass(p2);            if (p2.nodetype=calln) and               assigned(tcallnode(p2).procdefinition) then              begin                if is_new then                  begin                    if (tcallnode(p2).procdefinition.proctypeoption<>potype_constructor) then                      Message(parser_e_expr_have_to_be_constructor_call);                    p2.resultdef:=p.resultdef;                    p2:=cassignmentnode.create(p,p2);                    typecheckpass(p2);                  end                else                  begin                   { Free is not a destructor                    if (tcallnode(p2).procdefinition.proctypeoption<>potype_destructor) then                      Message(parser_e_expr_have_to_be_destructor_call);                   }                  end              end            else              internalerror(2005061202);            new_dispose_statement := p2;          end        { constructor,destructor specified }        else if not(m_mac in current_settings.modeswitches) and                try_to_consume(_COMMA) then          begin            { extended syntax of new and dispose }            { function styled new is handled in factor }            { destructors have no parameters }            destructorname:=pattern;            destructorpos:=current_tokenpos;            consume(_ID);            if (p.resultdef.typ<>pointerdef) then              begin                 Message1(type_e_pointer_type_expected,p.resultdef.typename);                 p.free;                 p:=factor(false);                 p.free;                 consume(_RKLAMMER);                 new_dispose_statement:=cerrornode.create;                 exit;              end;            { first parameter must be an object or class }            if tpointerdef(p.resultdef).pointeddef.typ<>objectdef then              begin                 Message(parser_e_pointer_to_class_expected);                 p.free;                 new_dispose_statement:=factor(false);                 consume_all_until(_RKLAMMER);                 consume(_RKLAMMER);                 exit;              end;            { check, if the first parameter is a pointer to a _class_ }            classh:=tobjectdef(tpointerdef(p.resultdef).pointeddef);            if is_class(classh) then              begin                 Message(parser_e_no_new_or_dispose_for_classes);                 new_dispose_statement:=factor(false);                 consume_all_until(_RKLAMMER);                 consume(_RKLAMMER);                 exit;              end;            { search cons-/destructor, also in parent classes }            storepos:=current_tokenpos;            current_tokenpos:=destructorpos;            sym:=search_class_member(classh,destructorname);            current_tokenpos:=storepos;            { the second parameter of new/dispose must be a call }            { to a cons-/destructor                              }            if (not assigned(sym)) or (sym.typ<>procsym) then              begin                 if is_new then                  Message(parser_e_expr_have_to_be_constructor_call)                 else                  Message(parser_e_expr_have_to_be_destructor_call);                 p.free;                 new_dispose_statement:=cerrornode.create;              end            else              begin                { For new(var,constructor) we need to take a copy because                  p is also used in the assignmentn below }                if is_new then                  begin                    p2:=cderefnode.create(p.getcopy);                    include(p2.flags,nf_no_checkpointer);                  end                else                  p2:=cderefnode.create(p);                do_typecheckpass(p2);                if is_new then                  callflag:=cnf_new_call                else                  callflag:=cnf_dispose_call;                if is_new then                  do_member_read(classh,false,sym,p2,again,[callflag])                else                  begin                    if not(m_fpc in current_settings.modeswitches) then                      do_member_read(classh,false,sym,p2,again,[callflag])                    else                      begin                        p2:=ccallnode.create(nil,tprocsym(sym),sym.owner,p2,[callflag]);                        { support dispose(p,done()); }                        if try_to_consume(_LKLAMMER) then                          begin                            if not try_to_consume(_RKLAMMER) then                              begin                                Message(parser_e_no_paras_for_destructor);                                consume_all_until(_RKLAMMER);                                consume(_RKLAMMER);                              end;                          end;                      end;                  end;                { we need the real called method }                do_typecheckpass(p2);                if (p2.nodetype=calln) and                   assigned(tcallnode(p2).procdefinition) then                  begin                    if is_new then                     begin                       if (tcallnode(p2).procdefinition.proctypeoption<>potype_constructor) then                         Message(parser_e_expr_have_to_be_constructor_call);                       p2.resultdef:=p.resultdef;                       p2:=cassignmentnode.create(p,p2);                     end                    else                     begin                       if (tcallnode(p2).procdefinition.proctypeoption<>potype_destructor) then                         Message(parser_e_expr_have_to_be_destructor_call);                     end;                  end                else                  begin                    if is_new then                      CGMessage(parser_e_expr_have_to_be_constructor_call)                    else                      CGMessage(parser_e_expr_have_to_be_destructor_call);                  end;                result:=p2;              end;          end        else          begin             if (p.resultdef.typ<>pointerdef) then               Begin                  Message1(type_e_pointer_type_expected,p.resultdef.typename);                  new_dispose_statement:=cerrornode.create;               end             else               begin                  if (tpointerdef(p.resultdef).pointeddef.typ=objectdef) and                     (oo_has_vmt in tobjectdef(tpointerdef(p.resultdef).pointeddef).objectoptions) then                    Message(parser_w_use_extended_syntax_for_objects);                  if (tpointerdef(p.resultdef).pointeddef.typ=orddef) and                     (torddef(tpointerdef(p.resultdef).pointeddef).ordtype=uvoid) then                    begin                      if (m_tp7 in current_settings.modeswitches) or                         (m_delphi in current_settings.modeswitches) then                       Message(parser_w_no_new_dispose_on_void_pointers)                      else                       Message(parser_e_no_new_dispose_on_void_pointers);                    end;                  { create statements with call to getmem+initialize or                    finalize+freemem }                  new_dispose_statement:=internalstatements(newstatement);                  if is_new then                   begin                     { create temp for result }                     temp := ctempcreatenode.create(p.resultdef,p.resultdef.size,tt_persistent,true);                     addstatement(newstatement,temp);                     { create call to fpc_getmem }                     para := ccallparanode.create(cordconstnode.create                         (tpointerdef(p.resultdef).pointeddef.size,s32inttype,true),nil);                     addstatement(newstatement,cassignmentnode.create(                         ctemprefnode.create(temp),                         ccallnode.createintern('fpc_getmem',para)));                     { create call to fpc_initialize }                     if tpointerdef(p.resultdef).pointeddef.needs_inittable then                       addstatement(newstatement,initialize_data_node(cderefnode.create(ctemprefnode.create(temp))));                     { copy the temp to the destination }                     addstatement(newstatement,cassignmentnode.create(                         p,                         ctemprefnode.create(temp)));                     { release temp }                     addstatement(newstatement,ctempdeletenode.create(temp));                   end                  else                   begin                     { create call to fpc_finalize }                     if tpointerdef(p.resultdef).pointeddef.needs_inittable then                       addstatement(newstatement,finalize_data_node(cderefnode.create(p.getcopy)));                     { create call to fpc_freemem }                     para := ccallparanode.create(p,nil);                     addstatement(newstatement,ccallnode.createintern('fpc_freemem',para));                   end;               end;          end;        consume(_RKLAMMER);      end;    function new_function : tnode;      var        newstatement : tstatementnode;        newblock     : tblocknode;        temp         : ttempcreatenode;        para         : tcallparanode;        p1,p2  : tnode;        classh : tobjectdef;        srsym    : tsym;        srsymtable : TSymtable;        again  : boolean; { dummy for do_proc_call }      begin        consume(_LKLAMMER);        p1:=factor(false);        if p1.nodetype<>typen then         begin           Message(type_e_type_id_expected);           consume_all_until(_RKLAMMER);           consume(_RKLAMMER);           p1.destroy;           new_function:=cerrornode.create;           exit;         end;        if (p1.resultdef.typ<>pointerdef) then         begin           Message1(type_e_pointer_type_expected,p1.resultdef.typename);           consume_all_until(_RKLAMMER);           consume(_RKLAMMER);           p1.destroy;           new_function:=cerrornode.create;           exit;         end;        if try_to_consume(_RKLAMMER) then          begin            if (tpointerdef(p1.resultdef).pointeddef.typ=objectdef) and               (oo_has_vmt in tobjectdef(tpointerdef(p1.resultdef).pointeddef).objectoptions)  then              Message(parser_w_use_extended_syntax_for_objects);            { create statements with call to getmem+initialize }            newblock:=internalstatements(newstatement);            { create temp for result }            temp := ctempcreatenode.create(p1.resultdef,p1.resultdef.size,tt_persistent,true);            addstatement(newstatement,temp);            { create call to fpc_getmem }            para := ccallparanode.create(cordconstnode.create                (tpointerdef(p1.resultdef).pointeddef.size,s32inttype,true),nil);            addstatement(newstatement,cassignmentnode.create(                ctemprefnode.create(temp),                ccallnode.createintern('fpc_getmem',para)));            { create call to fpc_initialize }            if tpointerdef(p1.resultdef).pointeddef.needs_inittable then             begin               para := ccallparanode.create(caddrnode.create_internal(crttinode.create                          (tstoreddef(tpointerdef(p1.resultdef).pointeddef),initrtti)),                       ccallparanode.create(ctemprefnode.create                          (temp),nil));               addstatement(newstatement,ccallnode.createintern('fpc_initialize',para));             end;            { the last statement should return the value as              location and type, this is done be referencing the              temp and converting it first from a persistent temp to              normal temp }            addstatement(newstatement,ctempdeletenode.create_normal_temp(temp));            addstatement(newstatement,ctemprefnode.create(temp));            p1.destroy;            p1:=newblock;          end        else          begin            consume(_COMMA);            if tpointerdef(p1.resultdef).pointeddef.typ<>objectdef then             begin               Message(parser_e_pointer_to_class_expected);               consume_all_until(_RKLAMMER);               consume(_RKLAMMER);               p1.destroy;               new_function:=cerrornode.create;               exit;             end;            classh:=tobjectdef(tpointerdef(p1.resultdef).pointeddef);            { use the objectdef for loading the VMT }            p2:=p1;            p1:=ctypenode.create(tpointerdef(p1.resultdef).pointeddef);            do_typecheckpass(p1);            { search the constructor also in the symbol tables of              the parents }            afterassignment:=false;            searchsym_in_class(classh,nil,pattern,srsym,srsymtable);            consume(_ID);            do_member_read(classh,false,srsym,p1,again,[cnf_new_call]);            { we need to know which procedure is called }            do_typecheckpass(p1);            if not(                   (p1.nodetype=calln) and                   assigned(tcallnode(p1).procdefinition) and                   (tcallnode(p1).procdefinition.proctypeoption=potype_constructor)                  ) then              Message(parser_e_expr_have_to_be_constructor_call);            { constructors return boolean, update resultdef to return              the pointer to the object }            p1.resultdef:=p2.resultdef;            p2.free;            consume(_RKLAMMER);          end;        new_function:=p1;      end;    function inline_setlength : tnode;      var        paras   : tnode;        npara,        ppn     : tcallparanode;        dims,        counter : integer;        isarray : boolean;        def     : tdef;        destppn : tnode;        newstatement : tstatementnode;        temp    : ttempcreatenode;        newblock : tnode;      begin        { for easy exiting if something goes wrong }        result := cerrornode.create;        consume(_LKLAMMER);        paras:=parse_paras(false,false,_RKLAMMER);        consume(_RKLAMMER);        if not assigned(paras) then         begin           CGMessage1(parser_e_wrong_parameter_size,'SetLength');           exit;         end;        dims:=0;        if assigned(paras) then         begin           { check type of lengths }           ppn:=tcallparanode(paras);           while assigned(ppn.right) do            begin              set_varstate(ppn.left,vs_read,[vsf_must_be_valid]);              inserttypeconv(ppn.left,sinttype);              inc(dims);              ppn:=tcallparanode(ppn.right);            end;         end;        if dims=0 then         begin           CGMessage1(parser_e_wrong_parameter_size,'SetLength');           paras.free;           exit;         end;        { last param must be var }        destppn:=ppn.left;        inc(parsing_para_level);        valid_for_var(destppn,true);        set_varstate(destppn,vs_written,[]);        dec(parsing_para_level);        { first param must be a string or dynamic array ...}        isarray:=is_dynamic_array(destppn.resultdef);        if not((destppn.resultdef.typ=stringdef) or               isarray) then         begin           CGMessage(type_e_mismatch);           paras.free;           exit;         end;        { only dynamic arrays accept more dimensions }        if (dims>1) then         begin           if (not isarray) then            CGMessage(type_e_mismatch)           else            begin              { check if the amount of dimensions is valid }              def := tarraydef(destppn.resultdef).elementdef;              counter:=dims;              while counter > 1 do                begin                  if not(is_dynamic_array(def)) then                    begin                      CGMessage1(parser_e_wrong_parameter_size,'SetLength');                      break;                    end;                  dec(counter);                  def := tarraydef(def).elementdef;                end;            end;         end;        if isarray then         begin            { create statements with call initialize the arguments and              call fpc_dynarr_setlength }            newblock:=internalstatements(newstatement);            { get temp for array of lengths }            temp := ctempcreatenode.create(sinttype,dims*sinttype.size,tt_persistent,false);            addstatement(newstatement,temp);            { load array of lengths }            ppn:=tcallparanode(paras);            counter:=0;            while assigned(ppn.right) do             begin               addstatement(newstatement,cassignmentnode.create(                   ctemprefnode.create_offset(temp,counter*sinttype.size),                   ppn.left));               ppn.left:=nil;               inc(counter);               ppn:=tcallparanode(ppn.right);             end;            { destppn is also reused }            ppn.left:=nil;            { create call to fpc_dynarr_setlength }            npara:=ccallparanode.create(caddrnode.create_internal                      (ctemprefnode.create(temp)),                   ccallparanode.create(cordconstnode.create                      (counter,s32inttype,true),                   ccallparanode.create(caddrnode.create_internal                      (crttinode.create(tstoreddef(destppn.resultdef),initrtti)),                   ccallparanode.create(ctypeconvnode.create_internal(destppn,voidpointertype),nil))));            addstatement(newstatement,ccallnode.createintern('fpc_dynarray_setlength',npara));            addstatement(newstatement,ctempdeletenode.create(temp));            { we don't need original the callparanodes tree }            paras.free;         end        else         begin            { we can reuse the supplied parameters }            newblock:=ccallnode.createintern(               'fpc_'+tstringdef(destppn.resultdef).stringtypname+'_setlength',paras);         end;        result.free;        result:=newblock;      end;    function inline_initialize : tnode;      var        newblock,        paras   : tnode;        ppn     : tcallparanode;      begin        { for easy exiting if something goes wrong }        result := cerrornode.create;        consume(_LKLAMMER);        paras:=parse_paras(false,false,_RKLAMMER);        consume(_RKLAMMER);        if not assigned(paras) then         begin           CGMessage1(parser_e_wrong_parameter_size,'Initialize');           exit;         end;        ppn:=tcallparanode(paras);        { 2 arguments? }        if assigned(ppn.right) then         begin           CGMessage1(parser_e_wrong_parameter_size,'Initialize');           paras.free;           exit;         end;        newblock:=initialize_data_node(ppn.left);        ppn.left:=nil;        paras.free;        result.free;        result:=newblock;      end;    function inline_finalize : tnode;      var        newblock,        paras   : tnode;        npara,        destppn,        ppn     : tcallparanode;      begin        { for easy exiting if something goes wrong }        result := cerrornode.create;        consume(_LKLAMMER);        paras:=parse_paras(false,false,_RKLAMMER);        consume(_RKLAMMER);        if not assigned(paras) then         begin           CGMessage1(parser_e_wrong_parameter_size,'Finalize');           exit;         end;        ppn:=tcallparanode(paras);        { 2 arguments? }        if assigned(ppn.right) then         begin           destppn:=tcallparanode(ppn.right);           { 3 arguments is invalid }           if assigned(destppn.right) then            begin              CGMessage1(parser_e_wrong_parameter_size,'Finalize');              paras.free;              exit;            end;           { create call to fpc_finalize_array }           npara:=ccallparanode.create(cordconstnode.create                     (destppn.left.resultdef.size,s32inttype,true),                  ccallparanode.create(ctypeconvnode.create                     (ppn.left,s32inttype),                  ccallparanode.create(caddrnode.create_internal                     (crttinode.create(tstoreddef(destppn.left.resultdef),initrtti)),                  ccallparanode.create(caddrnode.create_internal                     (destppn.left),nil))));           newblock:=ccallnode.createintern('fpc_finalize_array',npara);           destppn.left:=nil;           ppn.left:=nil;         end        else         begin           newblock:=finalize_data_node(ppn.left);           ppn.left:=nil;         end;        paras.free;        result.free;        result:=newblock;      end;    function inline_copy : tnode;      var        copynode,        lowppn,        highppn,        npara,        paras   : tnode;        ppn     : tcallparanode;        paradef : tdef;        counter : integer;      begin        { for easy exiting if something goes wrong }        result := cerrornode.create;        consume(_LKLAMMER);        paras:=parse_paras(false,false,_RKLAMMER);        consume(_RKLAMMER);        if not assigned(paras) then         begin           CGMessage1(parser_e_wrong_parameter_size,'Copy');           exit;         end;        { determine copy function to use based on the first argument,          also count the number of arguments in this loop }        counter:=1;        ppn:=tcallparanode(paras);        while assigned(ppn.right) do         begin           inc(counter);           ppn:=tcallparanode(ppn.right);         end;        paradef:=ppn.left.resultdef;        if is_ansistring(paradef) or           (is_chararray(paradef) and            (paradef.size>255)) or           ((cs_ansistrings in current_settings.localswitches) and            is_pchar(paradef)) then          copynode:=ccallnode.createintern('fpc_ansistr_copy',paras)        else         if is_widestring(paradef) or            is_widechararray(paradef) or            is_pwidechar(paradef) then           copynode:=ccallnode.createintern('fpc_widestr_copy',paras)        else         if is_char(paradef) then           copynode:=ccallnode.createintern('fpc_char_copy',paras)        else         if is_dynamic_array(paradef) then          begin            { Only allow 1 or 3 arguments }            if (counter<>1) and (counter<>3) then             begin               CGMessage1(parser_e_wrong_parameter_size,'Copy');               exit;             end;            { create statements with call }            if (counter=3) then             begin               highppn:=tcallparanode(paras).left.getcopy;               lowppn:=tcallparanode(tcallparanode(paras).right).left.getcopy;             end            else             begin               { use special -1,-1 argument to copy the whole array }               highppn:=cordconstnode.create(-1,s32inttype,false);               lowppn:=cordconstnode.create(-1,s32inttype,false);             end;            { create call to fpc_dynarray_copy }            npara:=ccallparanode.create(highppn,                   ccallparanode.create(lowppn,                   ccallparanode.create(caddrnode.create_internal                      (crttinode.create(tstoreddef(ppn.left.resultdef),initrtti)),                   ccallparanode.create                      (ctypeconvnode.create_internal(ppn.left,voidpointertype),nil))));            copynode:=ccallnode.createinternres('fpc_dynarray_copy',npara,ppn.left.resultdef);            ppn.left:=nil;            paras.free;          end        else         begin           { generic fallback that will give an error if a wrong             type is passed }           copynode:=ccallnode.createintern('fpc_shortstr_copy',paras)         end;        result.free;        result:=copynode;      end;end.
 |