mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-19 13:59:29 +02:00
351 lines
5.3 KiB
PHP
351 lines
5.3 KiB
PHP
{
|
|
|
|
This file is part of the Free Pascal run time library.
|
|
Copyright (c) 2017 by the Free Pascal development team.
|
|
|
|
Processor dependent implementation for the system unit for
|
|
Z80
|
|
|
|
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.
|
|
|
|
**********************************************************************}
|
|
|
|
|
|
var
|
|
z80_save_hl: Word; public name 'FPC_Z80_SAVE_HL';
|
|
|
|
{$define FPC_SYSTEM_HAS_FPC_CPUINIT}
|
|
procedure fpc_cpuinit;{$ifdef SYSTEMINLINE}inline;{$endif}
|
|
begin
|
|
end;
|
|
|
|
|
|
{$define FPC_SYSTEM_HAS_MOVE}
|
|
procedure Move(const source;var dest;count:SizeInt);assembler;[public, alias: 'FPC_MOVE'];
|
|
label
|
|
skip, forward_move;
|
|
asm
|
|
ld c, (count)
|
|
ld b, (count+1)
|
|
bit 7, b
|
|
jp NZ, skip
|
|
ld a, b
|
|
or a, c
|
|
jp Z, skip
|
|
|
|
ld l, (source)
|
|
ld h, (source+1)
|
|
ld e, (dest)
|
|
ld d, (dest+1)
|
|
|
|
ld a, d
|
|
cp a, h
|
|
jp C, forward_move
|
|
ld a, e
|
|
cp a, l
|
|
jp C, forward_move
|
|
|
|
{ backward move }
|
|
add hl, bc
|
|
dec hl
|
|
ex de, hl
|
|
add hl, bc
|
|
dec hl
|
|
ex de, hl
|
|
lddr
|
|
jp skip
|
|
|
|
forward_move:
|
|
ldir
|
|
|
|
skip:
|
|
end;
|
|
|
|
|
|
{$define FPC_SYSTEM_HAS_FILLCHAR}
|
|
Procedure FillChar(var x;count:SizeInt;value:byte);assembler;
|
|
label
|
|
skip, loop;
|
|
asm
|
|
ld c, (count)
|
|
ld b, (count+1)
|
|
bit 7, b
|
|
jp NZ, skip
|
|
|
|
ld a, b
|
|
or a, c
|
|
jp Z, skip
|
|
|
|
ld e, (value)
|
|
ld a, 0
|
|
ld l, (x)
|
|
ld h, (x+1)
|
|
loop:
|
|
ld (hl), e
|
|
inc hl
|
|
dec bc
|
|
cp a, c
|
|
jp NZ, loop
|
|
cp a, b
|
|
jp NZ, loop
|
|
skip:
|
|
end;
|
|
|
|
|
|
{$IFNDEF INTERNAL_BACKTRACE}
|
|
{$define FPC_SYSTEM_HAS_GET_FRAME}
|
|
function get_frame:pointer;assembler;nostackframe;
|
|
asm
|
|
push ix
|
|
pop hl
|
|
end;
|
|
{$ENDIF not INTERNAL_BACKTRACE}
|
|
|
|
|
|
{$define FPC_SYSTEM_HAS_GET_CALLER_ADDR}
|
|
function get_caller_addr(framebp:pointer;addr:pointer=nil):pointer;assembler;
|
|
label
|
|
framebp_null;
|
|
asm
|
|
ld l, (framebp)
|
|
ld h, (framebp+1)
|
|
|
|
ld a, l
|
|
or a, h
|
|
jp Z, framebp_null
|
|
|
|
inc hl
|
|
inc hl
|
|
|
|
ld e, (hl)
|
|
inc hl
|
|
ld d, (hl)
|
|
ex de, hl
|
|
framebp_null:
|
|
end;
|
|
|
|
|
|
{$define FPC_SYSTEM_HAS_GET_CALLER_FRAME}
|
|
function get_caller_frame(framebp:pointer;addr:pointer=nil):pointer;assembler;
|
|
label
|
|
framebp_null;
|
|
asm
|
|
ld l, (framebp)
|
|
ld h, (framebp+1)
|
|
|
|
ld a, l
|
|
or a, h
|
|
jp Z, framebp_null
|
|
|
|
ld e, (hl)
|
|
inc hl
|
|
ld d, (hl)
|
|
ex de, hl
|
|
framebp_null:
|
|
end;
|
|
|
|
|
|
{$define FPC_SYSTEM_HAS_SPTR}
|
|
Function Sptr : pointer;assembler;nostackframe;
|
|
asm
|
|
ld hl, 0
|
|
add hl, sp
|
|
end;
|
|
|
|
|
|
function InterLockedDecrement (var Target: longint) : longint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
dec(Target);
|
|
Result:=Target;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedIncrement (var Target: longint) : longint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
inc(Target);
|
|
Result:=Target;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedExchange (var Target: longint;Source : longint) : longint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
Result:=Target;
|
|
Target:=Source;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterlockedCompareExchange(var Target: longint; NewValue: longint; Comperand: longint): longint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
Result:=Target;
|
|
if Target=Comperand then
|
|
Target:=NewValue;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedExchangeAdd (var Target: longint;Source : longint) : longint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
Result:=Target;
|
|
inc(Target,Source);
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedDecrement (var Target: smallint) : smallint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
dec(Target);
|
|
Result:=Target;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedIncrement (var Target: smallint) : smallint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
inc(Target);
|
|
Result:=Target;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedExchange (var Target: smallint;Source : smallint) : smallint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
Result:=Target;
|
|
Target:=Source;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterlockedCompareExchange(var Target: smallint; NewValue: smallint; Comperand: smallint): smallint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
Result:=Target;
|
|
if Target=Comperand then
|
|
Target:=NewValue;
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|
|
|
|
|
|
function InterLockedExchangeAdd (var Target: smallint;Source : smallint) : smallint;
|
|
var
|
|
temp_sreg : byte;
|
|
begin
|
|
{ block interrupts }
|
|
asm
|
|
di
|
|
end;
|
|
|
|
Result:=Target;
|
|
inc(Target,Source);
|
|
|
|
{ release interrupts }
|
|
asm
|
|
ei
|
|
end;
|
|
end;
|