| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271 | {    $Id$    Copyright (c) 1998-2002 by Florian Klaempfl    Generate 680x0 assembler for math nodes    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 n68kmat;{$i fpcdefs.inc}interface    uses      node,nmat,ncgmat,cpubase,cgbase;    type      tm68knotnode = class(tnotnode)         procedure pass_2;override;      end;      tm68kmoddivnode = class(tcgmoddivnode)         procedure emit_div_reg_reg(signed: boolean;denum,num : tregister);override;         procedure emit_mod_reg_reg(signed: boolean;denum,num : tregister);override;      end;implementation    uses      globtype,systems,      cutils,verbose,globals,      symconst,symdef,aasmbase,aasmtai,aasmcpu,      pass_1,pass_2,      ncon,      cpuinfo,paramgr,defutil,parabase,      tgobj,ncgutil,cgobj,cgutils,rgobj,rgcpu,cgcpu,cg64f32;{*****************************************************************************                               TM68KNOTNODE*****************************************************************************}    procedure tm68knotnode.pass_2;      var         hl : tasmlabel;         opsize : tcgsize;      begin         opsize:=def_cgsize(resulttype.def);         if is_boolean(resulttype.def) then          begin            { the second pass could change the location of left }            { if it is a register variable, so we've to do      }            { this before the case statement                    }            if left.location.loc<>LOC_JUMP then             secondpass(left);            case left.location.loc of              LOC_JUMP :                begin                  location_reset(location,LOC_JUMP,OS_NO);                  hl:=truelabel;                  truelabel:=falselabel;                  falselabel:=hl;                  secondpass(left);                  maketojumpbool(exprasmlist,left,lr_load_regvars);                  hl:=truelabel;                  truelabel:=falselabel;                  falselabel:=hl;                end;              LOC_FLAGS :                begin                  location_copy(location,left.location);//                  location_release(exprasmlist,left.location);                  inverse_flags(location.resflags);                end;              LOC_CONSTANT,              LOC_REGISTER,              LOC_CREGISTER,              LOC_REFERENCE,              LOC_CREFERENCE :                begin                  location_force_reg(exprasmlist,left.location,def_cgsize(resulttype.def),true);                  exprasmlist.concat(taicpu.op_reg(A_TST,tcgsize2opsize[opsize],left.location.register));//                  location_release(exprasmlist,left.location);                  location_reset(location,LOC_FLAGS,OS_NO);                  location.resflags:=F_E;                end;             else                internalerror(200203224);            end;          end         else if is_64bitint(left.resulttype.def) then           begin              secondpass(left);              location_copy(location,left.location);              location_force_reg(exprasmlist,location,OS_64,false);              cg64.a_op64_loc_reg(exprasmlist,OP_NOT,location,                joinreg64(location.register64.reglo,location.register64.reghi));           end         else          begin             secondpass(left);             location_force_reg(exprasmlist,left.location,def_cgsize(left.resulttype.def),false);             location_copy(location,left.location);             if location.loc=LOC_CREGISTER then              location.register := cg.getintregister(exprasmlist,opsize);             { perform the NOT operation }             cg.a_op_reg_reg(exprasmlist,OP_NOT,opsize,location.register,left.location.register);          end;      end;{*****************************************************************************                               TM68KMODDIVNODE*****************************************************************************}  procedure tm68kmoddivnode.emit_div_reg_reg(signed: boolean;denum,num : tregister);   var     continuelabel : tasmlabel;     reg_d0,reg_d1 : tregister;     paraloc1 : tcgpara;   begin     { no RTL call, so inline a zero denominator verification }     if aktoptprocessor <> MC68000 then       begin         { verify if denominator is zero }         objectlibrary.getlabel(continuelabel);         { compare against zero, if not zero continue }         cg.a_cmp_const_reg_label(exprasmlist,OS_S32,OC_NE,0,denum,continuelabel);//	 paraloc1.init;//         cg.a_param_const(exprasmlist,OS_S32,200,paramanager.getintparaloc(pocall_default,1,paraloc1));         cg.a_call_name(exprasmlist,'FPC_HANDLEERROR');         cg.a_label(exprasmlist, continuelabel);         if signed then            exprasmlist.concat(taicpu.op_reg_reg(A_DIVS,S_L,denum,num))         else            exprasmlist.concat(taicpu.op_reg_reg(A_DIVU,S_L,denum,num));         { result should be in denuminator }         cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,num,denum);       end     else       begin         { On MC68000/68010 mw must pass through RTL routines }         reg_d0:=NR_D0;         cg.getcpuregister(exprasmlist,NR_D0);         reg_d1:=NR_D1;         cg.getcpuregister(exprasmlist,NR_D1);         { put numerator in d0 }         cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,num,reg_d0);         { put denum in D1 }         cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,denum,reg_d1);         if signed then             cg.a_call_name(exprasmlist,'FPC_DIV_LONGINT')         else             cg.a_call_name(exprasmlist,'FPC_DIV_CARDINAL');        cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,reg_d0,denum);        cg.ungetcpuregister(exprasmlist,reg_d0);        cg.ungetcpuregister(exprasmlist,reg_d1);       end;   end;  procedure tm68kmoddivnode.emit_mod_reg_reg(signed: boolean;denum,num : tregister);      var tmpreg : tregister;          continuelabel : tasmlabel;          signlabel : tasmlabel;          reg_d0,reg_d1 : tregister;    begin     { no RTL call, so inline a zero denominator verification }     if aktoptprocessor <> MC68000 then       begin         { verify if denominator is zero }         objectlibrary.getlabel(continuelabel);         { compare against zero, if not zero continue }         cg.a_cmp_const_reg_label(exprasmlist,OS_S32,OC_NE,0,denum,continuelabel);//         cg.a_param_const(exprasmlist, OS_S32,200,paramanager.getintparaloc(pocall_default,1));         cg.a_call_name(exprasmlist,'FPC_HANDLEERROR');         cg.a_label(exprasmlist, continuelabel);         tmpreg:=cg.getintregister(exprasmlist,OS_INT);         { we have to prepare the high register with the  }         { correct sign. i.e we clear it, check if the low dword reg }         { which will participate in the division is signed, if so we}         { we extend the sign to the high doword register by inverting }         { all the bits.                                             }         exprasmlist.concat(taicpu.op_reg(A_CLR,S_L,tmpreg));         objectlibrary.getlabel(signlabel);         exprasmlist.concat(taicpu.op_reg(A_TST,S_L,tmpreg));         cg.a_cmp_const_reg_label(exprasmlist,OS_S32,OC_A,0,tmpreg,signlabel);         { its a negative value, therefore change sign }         cg.a_label(exprasmlist,signlabel);         { tmpreg:num / denum }         if signed then           exprasmlist.concat(taicpu.op_reg_reg_reg(A_DIVSL,S_L,denum,tmpreg,num))         else           exprasmlist.concat(taicpu.op_reg_reg_reg(A_DIVUL,S_L,denum,tmpreg,num));         { remainder in tmpreg }         cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,tmpreg,denum);         cg.ungetcpuregister(exprasmlist,tmpreg);       end     else       begin         { On MC68000/68010 mw must pass through RTL routines }         Reg_d0:=NR_D0;         cg.getcpuregister(exprasmlist,NR_D0);         Reg_d1:=NR_D1;         cg.getcpuregister(exprasmlist,NR_D1);         { put numerator in d0 }         cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,num,Reg_D0);         { put denum in D1 }         cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,denum,Reg_D1);         if signed then             cg.a_call_name(exprasmlist,'FPC_MOD_LONGINT')         else             cg.a_call_name(exprasmlist,'FPC_MOD_CARDINAL');        cg.a_load_reg_reg(exprasmlist,OS_INT,OS_INT,Reg_D0,denum);        cg.ungetcpuregister(exprasmlist,Reg_D0);        cg.ungetcpuregister(exprasmlist,Reg_D1);       end;    end;begin   cnotnode:=tm68knotnode;   cmoddivnode:=tm68kmoddivnode;end.{  $Log$  Revision 1.12  2005-01-08 04:11:26  karoly    * made m68k to compile again  Revision 1.11  2004/10/31 21:45:03  peter    * generic tlocation    * move tlocation to cgutils  Revision 1.10  2004/09/25 14:23:54  peter    * ungetregister is now only used for cpuregisters, renamed to      ungetcpuregister    * renamed (get|unget)explicitregister(s) to ..cpuregister    * removed location-release/reference_release  Revision 1.9  2004/06/20 08:55:31  florian    * logs truncated  Revision 1.8  2004/04/25 21:26:16  florian    * some m68k stuff fixed}
 |