mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-29 00:41:46 +01:00 
			
		
		
		
	 eb066d3b32
			
		
	
	
		eb066d3b32
		
	
	
	
	
		
			
			before g_stackframe_entry (still have to fix macos) * compilation fixes (cycle doesn't work yet though)
		
			
				
	
	
		
			198 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     $Id$
 | |
|     Copyright (c) 2002 by Florian Klaempfl
 | |
| 
 | |
|     Implements the PowerPC specific part of call 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 bymethodpointer
 | |
|     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 nppccal;
 | |
| 
 | |
| {$i fpcdefs.inc}
 | |
| 
 | |
| interface
 | |
| 
 | |
|     uses
 | |
|       symdef,node,ncal,ncgcal;
 | |
| 
 | |
|     type
 | |
|        tppccallnode = class(tcgcallnode)
 | |
|           function pass_1 : tnode;override;
 | |
|           procedure push_framepointer;override;
 | |
|        end;
 | |
| 
 | |
| implementation
 | |
| 
 | |
|     uses
 | |
|       globtype,systems,
 | |
|       cutils,verbose,globals,
 | |
|       symconst,symbase,symsym,symtable,defutil,paramgr,
 | |
| {$ifdef GDB}
 | |
|   {$ifdef delphi}
 | |
|       sysutils,
 | |
|   {$else}
 | |
|       strings,
 | |
|   {$endif}
 | |
|       gdb,
 | |
| {$endif GDB}
 | |
|       cginfo,cgbase,pass_2,
 | |
|       cpuinfo,cpubase,aasmbase,aasmtai,aasmcpu,
 | |
|       nmem,nld,ncnv,
 | |
|       ncgutil,cgobj,tgobj,regvars,rgobj,rgcpu,cg64f32,cgcpu,cpupi;
 | |
| 
 | |
|   function tppccallnode.pass_1 : tnode;
 | |
| 
 | |
|     begin
 | |
|        result:=inherited pass_1;
 | |
|        if assigned(result) then
 | |
|          exit;
 | |
|        if procdefinition is tprocdef then
 | |
|          begin
 | |
|             if tprocdef(procdefinition).parast.datasize>tppcprocinfo(current_procinfo).maxpushedparasize then
 | |
|               tppcprocinfo(current_procinfo).maxpushedparasize:=tprocdef(procdefinition).parast.datasize;
 | |
|          end
 | |
|        else
 | |
|          begin
 | |
|             {!!!!}
 | |
|          end;
 | |
|     end;
 | |
| 
 | |
|   procedure tppccallnode.push_framepointer;
 | |
|     var
 | |
|        href : treference;
 | |
|        hregister1,hregister2 : tregister;
 | |
|        i : longint;
 | |
|     begin
 | |
|        if current_procdef.parast.symtablelevel=(tprocdef(procdefinition).parast.symtablelevel) then
 | |
|          begin
 | |
|             { pass the same framepointer as the current procedure got }
 | |
|             hregister2.enum:=R_INTREGISTER;
 | |
|             hregister2.number:=NR_R11;
 | |
|             reference_reset_base(href,current_procinfo.framepointer,current_procinfo.framepointer_offset);
 | |
|             cg.a_load_ref_reg(exprasmlist,OS_ADDR,OS_ADDR,href,hregister2);
 | |
|             { it must be adjusted! }
 | |
|          end
 | |
|          { this is only true if the difference is one !!
 | |
|            but it cannot be more !! }
 | |
|        else if (current_procdef.parast.symtablelevel=(tprocdef(procdefinition).parast.symtablelevel)-1) then
 | |
|          begin
 | |
|             { pass the same framepointer as the current procedure got }
 | |
|             hregister1.enum:=R_INTREGISTER;
 | |
|             hregister1.number:=NR_R1;
 | |
|             hregister2.enum:=R_INTREGISTER;
 | |
|             hregister2.number:=NR_R11;
 | |
|             cg.a_load_reg_reg(exprasmlist,OS_32,OS_32,hregister1,hregister2);
 | |
|          end
 | |
|        else if (current_procdef.parast.symtablelevel>(tprocdef(procdefinition).parast.symtablelevel)) then
 | |
|          begin
 | |
|             hregister1:=rg.getregisterint(exprasmlist,OS_ADDR);
 | |
|             reference_reset_base(href,current_procinfo.framepointer,current_procinfo.framepointer_offset);
 | |
|             cg.a_load_ref_reg(exprasmlist,OS_ADDR,OS_ADDR,href,hregister1);
 | |
|             { the previous frame pointer is always saved at }
 | |
|             { previous_framepointer+12 (in the link area)   }
 | |
|             reference_reset_base(href,hregister1,12);
 | |
|             i:=current_procdef.parast.symtablelevel-1;
 | |
|             while (i>tprocdef(procdefinition).parast.symtablelevel) do
 | |
|               begin
 | |
|                  cg.a_load_ref_reg(exprasmlist,OS_ADDR,OS_ADDR,href,hregister1);
 | |
|                  dec(i);
 | |
|               end;
 | |
|             hregister2.enum:=R_INTREGISTER;
 | |
|             hregister2.number:=NR_R11;
 | |
|             cg.a_load_reg_reg(exprasmlist,OS_ADDR,OS_ADDR,hregister1,hregister2);
 | |
|             rg.ungetregisterint(exprasmlist,hregister1);
 | |
|          end
 | |
|        else
 | |
|          internalerror(2002081303);
 | |
|     end;
 | |
| 
 | |
| begin
 | |
|    ccallnode:=tppccallnode;
 | |
| end.
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.17  2003-06-04 11:58:58  jonas
 | |
|     * calculate localsize also in g_return_from_proc since it's now called
 | |
|       before g_stackframe_entry (still have to fix macos)
 | |
|     * compilation fixes (cycle doesn't work yet though)
 | |
| 
 | |
|   Revision 1.16  2003/05/25 14:32:42  jonas
 | |
|     * fixed register numbering bug
 | |
| 
 | |
|   Revision 1.15  2003/05/24 11:47:27  jonas
 | |
|     * fixed framepointer storage: it's now always stored at r1+12, which is
 | |
|       a place in the link area reserved for compiler use.
 | |
| 
 | |
|   Revision 1.14  2003/05/23 18:51:26  jonas
 | |
|     * fixed support for nested procedures and more parameters than those
 | |
|       which fit in registers (untested/probably not working: calling a
 | |
|       nested procedure from a deeper nested procedure)
 | |
| 
 | |
|   Revision 1.13  2003/05/18 15:15:59  florian
 | |
|     + added abi field to tsysteminfo
 | |
| 
 | |
|   Revision 1.12  2003/05/16 23:15:51  jonas
 | |
|     * workaround for nested procedures until Peter fixes it properly :)
 | |
| 
 | |
|   Revision 1.11  2003/05/16 20:00:39  jonas
 | |
|     * powerpc nested procedure fixes, should work completely now if all
 | |
|       local variables of the parent procedure are declared before the
 | |
|       nested procedures are declared
 | |
| 
 | |
|   Revision 1.10  2003/04/27 11:21:36  peter
 | |
|     * aktprocdef renamed to current_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.9  2003/04/27 10:41:47  florian
 | |
|     * fixed nested procedures to get them working as before
 | |
| 
 | |
|   Revision 1.8  2003/04/27 07:48:05  peter
 | |
|     * updated for removed lexlevel
 | |
| 
 | |
|   Revision 1.7  2003/04/24 11:24:00  florian
 | |
|     * fixed several issues with nested procedures
 | |
| 
 | |
|   Revision 1.6  2003/04/23 12:35:35  florian
 | |
|     * fixed several issues with powerpc
 | |
|     + applied a patch from Jonas for nested function calls (PowerPC only)
 | |
|     * ...
 | |
| 
 | |
|   Revision 1.5  2003/04/04 15:38:56  peter
 | |
|     * moved generic code from n386cal to ncgcal, i386 now also
 | |
|       uses the generic ncgcal
 | |
| 
 | |
|   Revision 1.4  2002/12/05 14:28:12  florian
 | |
|     * some variant <-> dyn. array stuff
 | |
| 
 | |
|   Revision 1.3  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.2  2002/08/17 09:23:49  florian
 | |
|     * first part of procinfo rewrite
 | |
| 
 | |
|   Revision 1.1  2002/08/13 21:40:59  florian
 | |
|     * more fixes for ppc calling conventions
 | |
| }
 | |
| 
 |