| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423 | {    $Id$    Copyright (c) 1998-2002 by Florian Klaempfl    Reads inline Powerpc assembler and writes the lines direct to the output    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. ****************************************************************************}{  This unit reads PowerPC inline assembler and writes the lines direct to the output file.}unit radirect;{$i fpcdefs.inc}interface    uses      node;     function assemble : tnode;  implementation    uses       { common }       cutils,       { global }       globals,verbose,       systems,       { aasm }       aasmbase,aasmtai,aasmcpu,       { symtable }       symconst,symbase,symtype,symsym,symtable,defutil,       { pass 1 }       nbas,       { parser }       scanner,       { codegen }       cgbase,       { constants }       agppcgas,       cpubase       ;    { checks if a string identifies a register }    { it should be optimized                   }    function is_register(const s : string) : boolean;      var         r : toldregister;      begin         is_register:=false;         if length(s)>5 then           exit;         for r:=low(gas_reg2str) to high(gas_reg2str) do           if gas_reg2str[r]=s then              begin                 is_register:=true;                 exit;              end;      end;    function assemble : tnode;      var         retstr,s,hs : string;         c : char;         ende : boolean;         srsym,sym : tsym;         srsymtable : tsymtable;         code : TAAsmoutput;         framereg : tregister;         i,l : longint;       procedure writeasmline;         var           i : longint;         begin           i:=length(s);           while (i>0) and (s[i] in [' ',#9]) do            dec(i);           s[0]:=chr(i);           if s<>'' then            code.concat(Tai_direct.Create(strpnew(s)));            { consider it set function set if the offset was loaded }           if assigned(current_procinfo.procdef.funcretsym) and              (pos(retstr,upper(s))>0) then             tvarsym(current_procinfo.procdef.funcretsym).varstate:=vs_assigned;           s:='';         end;     begin       ende:=false;       framereg.enum:=R_INTREGISTER;       framereg.number:=NR_STACK_POINTER_REG;       convert_register_to_enum(framereg);       s:='';       if assigned(current_procinfo.procdef.funcretsym) and          is_fpu(current_procinfo.procdef.rettype.def) then         tvarsym(current_procinfo.procdef.funcretsym).varstate:=vs_assigned;       { !!!!!       if (not is_void(current_procinfo.procdef.rettype.def)) then         retstr:=upper(tostr(tvarsym(current_procinfo.procdef.funcretsym).adjusted_address)+'('+gas_reg2str[procinfo^.framepointer]+')')       else       }         retstr:='';       c:=current_scanner.asmgetchar;       code:=TAAsmoutput.Create;       while not(ende) do         begin            { wrong placement            current_scanner.gettokenpos; }            case c of              'A'..'Z','a'..'z','_':                begin                   current_scanner.gettokenpos;                   i:=0;                   hs:='';                   while ((ord(c)>=ord('A')) and (ord(c)<=ord('Z')))                      or ((ord(c)>=ord('a')) and (ord(c)<=ord('z')))                      or ((ord(c)>=ord('0')) and (ord(c)<=ord('9')))                      or (c='_') do                     begin                        inc(i);                        hs[i]:=c;                        c:=current_scanner.asmgetchar;                     end;                   hs[0]:=chr(i);                   if upper(hs)='END' then                      ende:=true                   else                      begin                         if c=':' then                           begin                             searchsym(upper(hs),srsym,srsymtable);                             if srsym<>nil then                               if (srsym.typ = labelsym) then                                 Begin                                    hs:=tlabelsym(srsym).lab.name;                                    tlabelsym(srsym).lab.is_set:=true;                                 end                               else                                 Message(asmr_w_using_defined_as_local);                           end                         else                           { access to local variables }                           if assigned(current_procinfo.procdef) then                             begin                                { I don't know yet, what the ppc port requires }                                { we'll see how things settle down             }                                { is the last written character an special }                                { char ?                                   }                                { !!!                                if (s[length(s)]='%') and                                   ret_in_acc(current_procinfo.procdef.rettype.def) and                                   ((pos('AX',upper(hs))>0) or                                   (pos('AL',upper(hs))>0)) then                                  tfuncretsym(current_procinfo.procdef.funcretsym).funcretstate:=vs_assigned;                                }                                if ((s[length(s)]<>'0') or (hs[1]<>'x')) and not(is_register(hs)) then                                  begin                                     if assigned(current_procinfo.procdef.localst) and                                        (current_procinfo.procdef.localst.symtablelevel >= normal_function_level) then                                       sym:=tsym(current_procinfo.procdef.localst.search(upper(hs)))                                     else                                       sym:=nil;                                     if assigned(sym) then                                       begin                                          if (sym.typ=labelsym) then                                            Begin                                               hs:=tlabelsym(sym).lab.name;                                            end                                          else if sym.typ=varsym then                                            begin                                               if (vo_is_external in tvarsym(sym).varoptions) then                                                 hs:=tvarsym(sym).mangledname                                               else                                                 begin                                                    if (tvarsym(sym).reg.enum<>R_NO) then// until new regallocator stuff settles down//                                                      hs:=gas_reg2str[procinfo.framepointer.enum]                                                      hs:=gas_reg2str[framereg.enum]                                                    else                                                      hs:=tostr(tvarsym(sym).address)+//                                                        '('+gas_reg2str[procinfo.framepointer.enum]+')';                                                        '('+gas_reg2str[framereg.enum]+')';                                                 end;                                            end                                          else                                          { call to local function }                                          if (sym.typ=procsym) and (pos('BL',upper(s))>0) then                                            hs:=tprocsym(sym).first_procdef.mangledname;                                       end                                     else                                       begin                                          if assigned(current_procinfo.procdef.parast) then                                            sym:=tsym(current_procinfo.procdef.parast.search(upper(hs)))                                          else                                            sym:=nil;                                          if assigned(sym) then                                            begin                                               if sym.typ=varsym then                                                 begin                                                    l:=tvarsym(sym).address;                                                    { set offset }                                                    inc(l,current_procinfo.procdef.parast.address_fixup);//                                                    hs:=tostr(l)+'('+gas_reg2str[procinfo.framepointer.enum]+')';                                                    hs:=tostr(l)+'('+gas_reg2str[framereg.enum]+')';                                                    if pos(',',s) > 0 then                                                      tvarsym(sym).varstate:=vs_used;                                                 end;                                            end                                          { I added that but it creates a problem in line.ppi                                          because there is a local label wbuffer and                                          a static variable WBUFFER ...                                          what would you decide, florian ?}                                          else                                            begin                                               searchsym(upper(hs),sym,srsymtable);                                               if assigned(sym) and (sym.owner.symtabletype in [globalsymtable,staticsymtable]) then                                                 begin                                                   case sym.typ of                                                     constsym :                                                       begin                                                         inc(tconstsym(sym).refs);                                                         case tconstsym(sym).consttyp of                                                           constint,constchar,constbool :                                                             hs:=tostr(tconstsym(sym).value.valueord);                                                           constpointer :                                                             hs:=tostr(tconstsym(sym).value.valueordptr);                                                           else                                                             Message(asmr_e_wrong_sym_type);                                                         end;                                                       end;                                                     varsym :                                                       begin                                                         Message2(asmr_h_direct_global_to_mangled,hs,tvarsym(sym).mangledname);                                                         hs:=tvarsym(sym).mangledname;                                                         inc(tvarsym(sym).refs);                                                       end;                                                     typedconstsym :                                                       begin                                                         Message2(asmr_h_direct_global_to_mangled,hs,ttypedconstsym(sym).mangledname);                                                         hs:=ttypedconstsym(sym).mangledname;                                                       end;                                                     procsym :                                                       begin                                                         { procs can be called or the address can be loaded }                                                         if (pos('BL',upper(s))>0) {or (pos('LEA',upper(s))>0))}  then                                                          begin                                                            if Tprocsym(sym).procdef_count>1 then                                                              Message1(asmr_w_direct_global_is_overloaded_func,hs);                                                            Message2(asmr_h_direct_global_to_mangled,hs,tprocsym(sym).first_procdef.mangledname);                                                            hs:=tprocsym(sym).first_procdef.mangledname;                                                          end;                                                       end;                                                     else                                                       Message(asmr_e_wrong_sym_type);                                                   end;                                                 end{$ifdef dummy}                                               else if upper(hs)='__SELF' then                                                 begin                                                    if assigned(procinfo^._class) then                                                      hs:=tostr(procinfo^.selfpointer_offset)+                                                          '('+gas_reg2str[procinfo^.framepointer]+')'                                                    else                                                     Message(asmr_e_cannot_use_SELF_outside_a_method);                                                 end                                               else if upper(hs)='__RESULT' then                                                 begin                                                    if (not is_void(current_procinfo.procdef.rettype.def)) then                                                      hs:=retstr                                                    else                                                      Message(asmr_e_void_function);                                                 end                                               { implement old stack/frame pointer access for nested procedures }                                               {!!!!                                               else if upper(hs)='__OLDSP' then                                                 begin                                                    { complicate to check there }                                                    { we do it: }                                                    if lexlevel>normal_function_level then                                                      hs:=tostr(procinfo^.framepointer_offset)+                                                        '('+gas_reg2str[procinfo^.framepointer]+')'                                                    else                                                      Message(asmr_e_cannot_use_OLDEBP_outside_nested_procedure);                                                 end;                                               }                                               end;{$endif dummy}                                            end;                                       end;                                  end;                             end;                         s:=s+hs;                      end;                end;              '{',';',#10,#13:                begin                   if pos(retstr,s) > 0 then                     tvarsym(current_procinfo.procdef.funcretsym).varstate:=vs_assigned;                   writeasmline;                   c:=current_scanner.asmgetchar;                end;              #26:                Message(scan_f_end_of_file);              else                begin                  current_scanner.gettokenpos;                  inc(byte(s[0]));                  s[length(s)]:=c;                  c:=current_scanner.asmgetchar;                end;            end;         end;       writeasmline;       assemble:=casmnode.create(code);     end;{*****************************************************************************                                     Initialize*****************************************************************************}const  asmmode_ppc_direct_info : tasmmodeinfo =          (            id    : asmmode_direct;            idtxt : 'DIRECT'          );initialization  RegisterAsmMode(asmmode_ppc_direct_info);end.{  $Log$  Revision 1.16  2003-06-13 21:19:32  peter    * current_procdef removed, use current_procinfo.procdef instead  Revision 1.15  2003/06/02 21:42:05  jonas    * function results can now also be regvars    - removed tprocinfo.return_offset, never use it again since it's invalid      if the result is a regvar  Revision 1.14  2003/05/30 23:57:08  peter    * more sparc cleanup    * accumulator removed, splitted in function_return_reg (called) and      function_result_reg (caller)  Revision 1.13  2003/04/27 11:21:36  peter    * aktprocdef renamed to current_procinfo.procdef    * procinfo renamed to current_procinfo    * procinfo will now be stored in current_module so it can be      cleaned up properly    * gen_main_procsym changed to create_main_proc and release_main_proc      to also generate a tprocinfo structure    * fixed unit implicit initfinal  Revision 1.12  2003/04/27 07:48:05  peter    * updated for removed lexlevel  Revision 1.11  2003/04/25 21:05:22  florian    * fixed tfuncretsym stuff in powerpc specific part  Revision 1.10  2003/04/24 12:05:53  florian    * symbols which are register identifiers aren't resolved anymore  Revision 1.9  2003/04/23 22:18:01  peter    * fixes to get rtl compiled  Revision 1.8  2003/03/22 18:00:27  jonas    * fixes for new regallocator  Revision 1.7  2003/01/08 18:43:58  daniel   * Tregister changed into a record  Revision 1.6  2002/11/25 17:43:28  peter    * splitted defbase in defutil,symutil,defcmp    * merged isconvertable and is_equal into compare_defs(_ext)    * made operator search faster by walking the list only once  Revision 1.5  2002/09/03 19:04:18  daniel    * Fixed PowerPC & M68000 compilation  Revision 1.4  2002/09/03 16:26:28  daniel    * Make Tprocdef.defs protected  Revision 1.3  2002/08/31 15:59:31  florian    + HEAP* stuff must be generated for Linux/PPC as well    + direct assembler reader searches now global and static symtables as well  Revision 1.2  2002/08/18 21:36:42  florian    + handling of local variables in direct reader implemented  Revision 1.1  2002/08/10 14:52:52  carl    + moved target_cpu_string to cpuinfo    * renamed asmmode enum.    * assembler reader has now less ifdef's    * move from nppcmem.pas -> ncgmem.pas vec. node.  Revision 1.2  2002/07/28 20:45:23  florian    + added direct assembler reader for PowerPC  Revision 1.1  2002/07/11 14:41:34  florian    * start of the new generic parameter handling}
 |