mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-31 16:51:35 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			171 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     Copyright (c) 2002-2009 by Florian Klaempfl and David Zhang
 | |
| 
 | |
|     This unit contains the CPU specific part of tprocinfo
 | |
| 
 | |
|     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 cpupi;
 | |
| 
 | |
| {$i fpcdefs.inc}
 | |
| 
 | |
| interface
 | |
| 
 | |
|   uses
 | |
|     cutils,
 | |
|     globtype,symdef,
 | |
|     procinfo,cpuinfo,cpupara,
 | |
|     psub,aasmdata,cgutils;
 | |
| 
 | |
|   type
 | |
| 
 | |
|     { TMIPSProcInfo }
 | |
| 
 | |
|     TMIPSProcInfo=class(tcgprocinfo)
 | |
|       intregstart,
 | |
|       floatregstart : aint;
 | |
|       intregssave,
 | |
|       floatregssave : byte;
 | |
|       register_used : tparasupregsused;
 | |
|       computed_local_size : longint;
 | |
|       save_gp_ref: treference;
 | |
|       //intparareg,
 | |
|       //parasize : longint;
 | |
|       constructor create(aparent:tprocinfo);override;
 | |
|       function calc_stackframe_size:longint;override;
 | |
|       procedure set_first_temp_offset;override;
 | |
|       procedure allocate_got_register(list:tasmlist);override;
 | |
|       procedure postprocess_code;override;
 | |
|     end;
 | |
| 
 | |
|    { Used by Stabs debug info generator }
 | |
| 
 | |
|    function mips_extra_offset(procdef : tprocdef) : longint;
 | |
| 
 | |
| implementation
 | |
| 
 | |
|     uses
 | |
|       systems,globals,verbose,
 | |
|       cpubase,cgbase,cgobj,
 | |
|       tgobj,paramgr,symconst,symcpu,aasmcpu;
 | |
| 
 | |
|     constructor TMIPSProcInfo.create(aparent: tprocinfo);
 | |
|       begin
 | |
|         inherited create(aparent);
 | |
|         if (cs_generate_stackframes in current_settings.localswitches) or
 | |
|            not (cs_opt_stackframe in current_settings.optimizerswitches) then
 | |
|           include(flags,pi_needs_stackframe);
 | |
| 
 | |
|         floatregssave:=12; { f20-f31 }
 | |
|         intregssave:=10;   { r16-r23,r30,r31 }
 | |
|         computed_local_size:=-1;
 | |
|         { pi_needs_got is not yet set correctly
 | |
|           so include it always if creating PIC code }
 | |
|         if (cs_create_pic in current_settings.moduleswitches) then
 | |
|           begin
 | |
|             include(flags, pi_needs_got);
 | |
|             got:=NR_GP;
 | |
|           end
 | |
|         else
 | |
|           got:=NR_NO;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure TMIPSProcInfo.set_first_temp_offset;
 | |
|       begin
 | |
|         { MIPS stack frame is always "optimized" }
 | |
|         framepointer:=NR_STACK_POINTER_REG;
 | |
|         tg.direction:=1;
 | |
| 
 | |
|         { We allocate enough space to save all registers because we can't determine
 | |
|           the necessary space because the used registers aren't known before
 | |
|           secondpass is run. }
 | |
| 
 | |
|         { will call _mcount if profiling }
 | |
|         if (cs_profile in current_settings.moduleswitches) and
 | |
|            not (po_nostackframe in procdef.procoptions) then
 | |
|           include(flags,pi_do_call);
 | |
| 
 | |
|         { Fixes the case when there are calls done by low-level means
 | |
|           (cg.a_call_name) but no child callnode. !!For assembler procedure
 | |
|           there is no clean way to determine what it calls, unless it is
 | |
|           also declared as nostackframe and everything is managed manually. }
 | |
|         if (pi_do_call in flags) or
 | |
|            ((pi_is_assembler in flags) and not (po_nostackframe in procdef.procoptions)) then
 | |
|           begin
 | |
|             include(flags,pi_do_call);   // for pi_is_assembler case
 | |
|             allocate_push_parasize(mips_nb_used_registers*sizeof(aint));
 | |
|           end;
 | |
| 
 | |
|         if not (po_nostackframe in procdef.procoptions) then
 | |
|           tg.setfirsttemp(Align(maxpushedparasize+
 | |
|             floatregssave*sizeof(aint)+intregssave*sizeof(aint)
 | |
|             ,max(current_settings.alignment.localalignmin,8)))
 | |
|         else
 | |
|           tg.setfirsttemp(align(maxpushedparasize,max(current_settings.alignment.localalignmin,8)));
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure TMIPSProcInfo.allocate_got_register(list:tasmlist);
 | |
|       begin
 | |
|         if (cs_create_pic in current_settings.moduleswitches) then
 | |
|           begin
 | |
|             if (pi_do_call in flags) then
 | |
|               include(flags,pi_needs_got);
 | |
|             if (pi_needs_got in flags) and
 | |
|                not (po_nostackframe in procdef.procoptions) then
 | |
|               tg.gettemp(list,sizeof(aint),sizeof(aint),tt_noreuse,save_gp_ref);
 | |
|           end;
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function TMIPSProcInfo.calc_stackframe_size:longint;
 | |
|       begin
 | |
|         result:=maxpushedparasize;
 | |
|         floatregstart:=result;
 | |
|         inc(result,floatregssave*4);
 | |
|         intregstart:=result;
 | |
|         //inc(result,intregssave*4);
 | |
|         result:=Align(tg.lasttemp,max(current_settings.alignment.localalignmin,8));
 | |
|         if computed_local_size=-1 then
 | |
|           begin
 | |
|             computed_local_size:=result;
 | |
|             tcpuprocdef(procdef).total_local_size:=result;
 | |
|           end
 | |
|         else if computed_local_size <> result then
 | |
|           Comment(V_Error,'TMIPSProcInfo.calc_stackframe_size result changed');
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     procedure TMIPSProcInfo.postprocess_code;
 | |
|       begin
 | |
|         fixup_jmps(aktproccode);
 | |
|       end;
 | |
| 
 | |
| 
 | |
|     function mips_extra_offset(procdef : tprocdef) : longint;
 | |
|       begin
 | |
|         if procdef=nil then
 | |
|           mips_extra_offset:=0
 | |
|         else
 | |
|           mips_extra_offset:=tcpuprocdef(procdef).total_local_size;
 | |
|       end;
 | |
| 
 | |
| begin
 | |
|   cprocinfo:=TMIPSProcInfo;
 | |
| end.
 | 
