{ This file is part of the Free Pascal run time library. Copyright (c) 2003 by the Free Pascal development team. Processor dependent implementation for the system unit for Xtensa See the file COPYING.FPC, included in this distribution, for details about the copyright. 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. **********************************************************************} {$define FPC_SYSTEM_HAS_SYSRESETFPU} Procedure SysResetFPU;{$ifdef SYSTEMINLINE}inline;{$endif} begin end; {$define FPC_SYSTEM_HAS_SYSINITFPU} Procedure SysInitFPU;{$ifdef SYSTEMINLINE}inline;{$endif} begin end; procedure fpc_cpuinit; begin { don't let libraries influence the FPU cw set by the host program } if not IsLibrary then SysInitFPU; end; {$ifdef fpc_abi_windowed} const // Minimum call8 calls to force register spilling to stack for caller of forceSpilledRegs spillcount = 6; procedure forceSpilledRegs(n: uint32); assembler; public name 'forcespilledregs'; label done, fin; asm beqz a2, done addi a10, a2, -1 call8 forcespilledregs done: bnez a2, fin movi a15, 0 fin: end; procedure fixCodeAddress(var addr: pointer); begin // Check if valid code address if ptruint(addr) and $C0000000 >= $40000000 then begin // Replace windowed call prefix addr:=codepointer((ptruint(addr)and$00FFFFFF) or $40000000); // Rewind to call instruction address dec(addr,3); end else addr:=nil; end; {$endif fpc_abi_windowed} {$IFNDEF INTERNAL_BACKTRACE} {$define FPC_SYSTEM_HAS_GET_FRAME} function get_frame:pointer;assembler; label done; asm {$ifdef fpc_abi_windowed} // Force registers to spill onto stack movi a10, spillcount call8 forcespilledregs // now get frame pointer of caller addi a2, a1, -12 l32i a2, a2, 0 done: {$else} mov a2, a1 {$endif} end; {$ENDIF not INTERNAL_BACKTRACE} {$define FPC_SYSTEM_HAS_GET_CALLER_ADDR} function get_caller_addr(framebp:pointer;addr:pointer=nil):pointer; begin {$ifdef fpc_abi_windowed} forceSpilledRegs(spillcount); if (ptruint(framebp)>$3ff00000)and(ptruint(framebp)<$40000000) then begin get_caller_addr:=pointer((framebp-16)^); fixCodeAddress(get_caller_addr); end else get_caller_addr:=nil; {$else} get_caller_addr:=nil; {$endif} end; {$define FPC_SYSTEM_HAS_GET_CALLER_FRAME} function get_caller_frame(framebp:pointer;addr:pointer=nil):pointer; begin {$ifdef fpc_abi_windowed} if (ptruint(framebp)>$3ff00000)and(ptruint(framebp)<$40000000) then begin forceSpilledRegs(spillcount); get_caller_frame:=pointer((framebp-12)^); end else get_caller_frame:=nil; {$else} get_caller_frame:=nil; {$endif} end; {$ifdef fpc_abi_windowed} {$define FPC_SYSTEM_HAS_GET_CALLER_STACKINFO} procedure get_caller_stackinfo(var framebp : pointer; var addr : codepointer); begin if (ptruint(framebp)>$3ff00000)and(ptruint(framebp)<$40000000) then begin forceSpilledRegs(spillcount); addr:=codepointer((framebp-16)^); framebp := pointer((framebp-12)^); fixCodeAddress(addr); end else begin addr:=nil; framebp:=nil; end; end; {$endif fpc_abi_windowed} {$define FPC_SYSTEM_HAS_SPTR} Function Sptr : pointer;assembler; asm mov a2,a1 end; function InterLockedDecrement (var Target: longint) : longint; var temp_sreg : byte; begin Result:=Target-1; Target:=Result; end; function InterLockedIncrement (var Target: longint) : longint; var temp_sreg : byte; begin Result:=Target+1; Target:=Result; end; function InterLockedExchange (var Target: longint;Source : longint) : longint; var temp_sreg : byte; begin Result:=Target; Target:=Source; end; function InterlockedCompareExchange(var Target: longint; NewValue: longint; Comperand: longint): longint; var temp_sreg : byte; begin Result:=Target; if Result=Comperand then Target:=NewValue; end; function InterLockedExchangeAdd (var Target: longint;Source : longint) : longint; var temp_sreg : byte; begin Result:=Target; Target:=Result+Source; end; function InterLockedDecrement (var Target: smallint) : smallint; var temp_sreg : byte; begin Result:=Target-1; Target:=Result; end; function InterLockedIncrement (var Target: smallint) : smallint; var temp_sreg : byte; begin Result:=Target+1; Target:=Result; end; function InterLockedExchange (var Target: smallint;Source : smallint) : smallint; var temp_sreg : byte; begin Result:=Target; Target:=Source; end; function InterlockedCompareExchange(var Target: smallint; NewValue: smallint; Comperand: smallint): smallint; var temp_sreg : byte; begin Result:=Target; if Result=Comperand then Target:=NewValue; end; function InterLockedExchangeAdd (var Target: smallint;Source : smallint) : smallint; var temp_sreg : byte; begin Result:=Target; Target:=Result+Source; end;