mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-08-07 16:06:16 +02:00
+ add initial RTL for MSX DOS. Simple programs are already working, but there are apparantly some code generation problems that lead to I/O as well as parameters not working correctly
git-svn-id: trunk@45600 -
This commit is contained in:
parent
e2d8f7b68f
commit
ac8552afc2
10
.gitattributes
vendored
10
.gitattributes
vendored
@ -11353,6 +11353,16 @@ rtl/msdos/sysosh.inc svneol=native#text/plain
|
|||||||
rtl/msdos/system.pp svneol=native#text/plain
|
rtl/msdos/system.pp svneol=native#text/plain
|
||||||
rtl/msdos/sysutils.pp svneol=native#text/plain
|
rtl/msdos/sysutils.pp svneol=native#text/plain
|
||||||
rtl/msdos/tthread.inc svneol=native#text/plain
|
rtl/msdos/tthread.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/Makefile.fpc svneol=native#text/plain
|
||||||
|
rtl/msxdos/registers.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/rtldefs.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/si_prc.pp svneol=native#text/pascal
|
||||||
|
rtl/msxdos/sysdir.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/sysfile.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/sysheap.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/sysos.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/sysosh.inc svneol=native#text/plain
|
||||||
|
rtl/msxdos/system.pp svneol=native#text/pascal
|
||||||
rtl/nativent/Makefile svneol=native#text/plain
|
rtl/nativent/Makefile svneol=native#text/plain
|
||||||
rtl/nativent/Makefile.fpc svneol=native#text/plain
|
rtl/nativent/Makefile.fpc svneol=native#text/plain
|
||||||
rtl/nativent/buildrtl.lpi svneol=native#text/plain
|
rtl/nativent/buildrtl.lpi svneol=native#text/plain
|
||||||
|
@ -52,6 +52,7 @@ dirs_win16=win16
|
|||||||
dirs_watcom=watcom
|
dirs_watcom=watcom
|
||||||
dirs_freertos=freertos
|
dirs_freertos=freertos
|
||||||
dirs_zxspectrum=zxspectrum
|
dirs_zxspectrum=zxspectrum
|
||||||
|
dirs_msxdos=msxdos
|
||||||
|
|
||||||
[install]
|
[install]
|
||||||
fpcpackage=y
|
fpcpackage=y
|
||||||
|
226
rtl/msxdos/Makefile.fpc
Normal file
226
rtl/msxdos/Makefile.fpc
Normal file
@ -0,0 +1,226 @@
|
|||||||
|
#
|
||||||
|
# Makefile.fpc for MSDOS RTL
|
||||||
|
#
|
||||||
|
[package]
|
||||||
|
main=rtl
|
||||||
|
[target]
|
||||||
|
#loaders=prt0s prt0t prt0m prt0c prt0l prt0h # exceptn fpu
|
||||||
|
units=system si_prc uuchar objpas iso7185
|
||||||
|
#uuchar objpas strings dos heaptrc lnfodwrf sysconst sysutils \
|
||||||
|
# math macpas iso7185 extpas rtlconsts typinfo cpu types \
|
||||||
|
# getopts sortbase fgl classes \
|
||||||
|
# msmouse ports \
|
||||||
|
# charset cpall ctypes \
|
||||||
|
# fpwidestring character unicodedata unicodenumtable
|
||||||
|
# cmem
|
||||||
|
# initc profile dxetype dxeload emu387 \
|
||||||
|
# cpu mmx \
|
||||||
|
# vesamode \
|
||||||
|
# rsts=math typinfo classes dateutil sysconst
|
||||||
|
implicitunits=exeinfo \
|
||||||
|
cp1250 cp1251 cp1252 cp1253 cp1254 cp1255 cp1256 cp1257 cp1258 \
|
||||||
|
cp437 cp646 cp737 cp775 cp850 cp852 cp855 cp856 cp857 cp860 cp861 cp862 \
|
||||||
|
cp863 cp864 cp865 cp866 cp869 cp874 cp3021 cp8859_1 cp8859_2 cp8859_3 cp8859_4 \
|
||||||
|
cp8859_5 cp8859_6 cp8859_7 cp8859_8 cp8859_9 cp8859_10 cp8859_11 \
|
||||||
|
cp8859_13 cp8859_14 cp8859_15 cp8859_16 cpkoi8_r cpkoi8_u
|
||||||
|
|
||||||
|
[require]
|
||||||
|
nortl=y
|
||||||
|
[install]
|
||||||
|
fpcpackage=y
|
||||||
|
[default]
|
||||||
|
fpcdir=../..
|
||||||
|
target=msxdos
|
||||||
|
cpu=z80
|
||||||
|
[compiler]
|
||||||
|
includedir=$(INC) $(PROCINC)
|
||||||
|
sourcedir=$(INC) $(PROCINC) $(COMMON)
|
||||||
|
[prerules]
|
||||||
|
RTL=..
|
||||||
|
INC=../inc
|
||||||
|
COMMON=$(RTL)/common
|
||||||
|
PROCINC=../$(CPU_TARGET)
|
||||||
|
UNITPREFIX=rtl
|
||||||
|
SYSTEMUNIT=system
|
||||||
|
|
||||||
|
# Paths
|
||||||
|
OBJPASDIR=$(RTL)/objpas
|
||||||
|
# Insert exception handler in system unit
|
||||||
|
ifdef EXCEPTIONS_IN_SYSTEM
|
||||||
|
override FPCOPT+=-dEXCEPTIONS_IN_SYSTEM
|
||||||
|
endif
|
||||||
|
# Insert exception handler in system unit
|
||||||
|
ifdef NO_EXCEPTIONS_IN_SYSTEM
|
||||||
|
override FPCOPT+=-dNO_EXCEPTIONS_IN_SYSTEM
|
||||||
|
endif
|
||||||
|
[rules]
|
||||||
|
# Get the system independent include file names.
|
||||||
|
# This will set the following variables :
|
||||||
|
# SYSINCNAMES
|
||||||
|
include $(INC)/makefile.inc
|
||||||
|
SYSINCDEPS=$(addprefix $(INC)/,$(SYSINCNAMES))
|
||||||
|
# Get the processor dependent include file names.
|
||||||
|
# This will set the following variables :
|
||||||
|
# CPUINCNAMES
|
||||||
|
include $(PROCINC)/makefile.cpu
|
||||||
|
SYSCPUDEPS=$(addprefix $(PROCINC)/,$(CPUINCNAMES))
|
||||||
|
# Put system unit dependencies together.
|
||||||
|
SYSDEPS=$(SYSINCDEPS) $(SYSCPUDEPS)
|
||||||
|
#
|
||||||
|
# Loaders
|
||||||
|
#
|
||||||
|
ifneq ($(findstring -dTEST_I8086_SMARTLINK_SECTIONS,$(FPCOPT)),)
|
||||||
|
override NASM_OPT+=-D__I8086_SMARTLINK_SECTIONS__
|
||||||
|
endif
|
||||||
|
|
||||||
|
#
|
||||||
|
# System Units (System, Objpas, Strings)
|
||||||
|
#
|
||||||
|
system$(PPUEXT) : system.pp $(SYSDEPS) $(INC)/tnyheaph.inc $(INC)/tinyheap.inc registers.inc
|
||||||
|
$(COMPILER) $(FPC_SYSTEM_OPT) -Us -Sg system.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
uuchar$(PPUEXT): $(SYSTEMUNIT)$(PPUEXT) $(INC)/uuchar.pp
|
||||||
|
$(COMPILER) $(INC)/uuchar.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
objpas$(PPUEXT): $(OBJPASDIR)/objpas.pp $(INC)/except.inc system$(PPUEXT)
|
||||||
|
$(COMPILER) -Fi$(OBJPASDIR) $(OBJPASDIR)/objpas.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
strings$(PPUEXT) : $(INC)/strings.pp $(INC)/stringsi.inc \
|
||||||
|
$(PROCINC)/strings.inc $(PROCINC)/stringss.inc \
|
||||||
|
$(INC)/genstr.inc $(INC)/genstrs.inc \
|
||||||
|
system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/strings.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
iso7185$(PPUEXT) : $(INC)/iso7185.pp system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/iso7185.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
extpas$(PPUEXT) : $(INC)/extpas.pp dos$(PPUEXT) $(SYSTEMUNIT)$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/extpas.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
#
|
||||||
|
# System Dependent Units
|
||||||
|
#
|
||||||
|
|
||||||
|
ports$(PPUEXT) : ports.pp system$(PPUEXT)
|
||||||
|
$(COMPILER) ports.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
#
|
||||||
|
# TP7 Compatible RTL Units
|
||||||
|
#
|
||||||
|
dos$(PPUEXT) : dos.pp registers.inc \
|
||||||
|
$(INC)/dosh.inc $(INC)/dos.inc $(INC)/fexpand.inc \
|
||||||
|
strings$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) dos.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
#
|
||||||
|
# Delphi Compatible Units
|
||||||
|
#
|
||||||
|
sysutils$(PPUEXT) : sysutils.pp $(wildcard $(OBJPASDIR)/sysutils/*.inc) \
|
||||||
|
objpas$(PPUEXT) dos$(PPUEXT) sysconst$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) -Fi$(OBJPASDIR)/sysutils sysutils.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
classes$(PPUEXT) : classes.pp $(wildcard $(OBJPASDIR)/classes/*.inc) \
|
||||||
|
sysutils$(PPUEXT) typinfo$(PPUEXT) rtlconsts$(PPUEXT) \
|
||||||
|
types$(PPUEXT) fgl$(PPUEXT) sortbase$(PPUEXT) \
|
||||||
|
objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) -Fi$(OBJPASDIR)/classes classes.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
fgl$(PPUEXT) : $(OBJPASDIR)/fgl.pp types$(PPUEXT) sysutils$(PPUEXT) rtlconsts$(PPUEXT) objpas$(PPUEXT) system$(PPUEXT) sortbase$(PPUEXT)
|
||||||
|
$(COMPILER) $(OBJPASDIR)/fgl.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
math$(PPUEXT): $(OBJPASDIR)/math.pp $(PROCINC)/mathu.inc objpas$(PPUEXT) sysutils$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(OBJPASDIR)/math.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
typinfo$(PPUEXT): $(OBJPASDIR)/typinfo.pp sysutils$(PPUEXT) objpas$(PPUEXT) rtlconsts$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) -Sg -Fi$(OBJPASDIR) $(OBJPASDIR)/typinfo.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
types$(PPUEXT) : $(OBJPASDIR)/types.pp objpas$(PPUEXT) math$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(OBJPASDIR)/types.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
rtlconsts$(PPUEXT) : $(OBJPASDIR)/rtlconsts.pp $(OBJPASDIR)/rtlconst.inc objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) -Fi$(OBJPASDIR) $(OBJPASDIR)/rtlconsts.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
sysconst$(PPUEXT) : $(OBJPASDIR)/sysconst.pp objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(OBJPASDIR)/sysconst.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
#
|
||||||
|
# Mac Pascal Model
|
||||||
|
#
|
||||||
|
macpas$(PPUEXT) : $(INC)/macpas.pp objpas$(PPUEXT) math$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/macpas.pp $(REDIR)
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
#
|
||||||
|
# Other system-independent RTL Units
|
||||||
|
#
|
||||||
|
cpu$(PPUEXT) : $(PROCINC)/cpu.pp sysutils$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(PROCINC)/cpu.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
getopts$(PPUEXT) : $(INC)/getopts.pp system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/getopts.pp $(REDIR)
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
heaptrc$(PPUEXT) : $(INC)/heaptrc.pp system$(PPUEXT)
|
||||||
|
$(COMPILER) -Sg $(INC)/heaptrc.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
lnfodwrf$(PPUEXT) : $(INC)/lnfodwrf.pp exeinfo$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/lnfodwrf.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
exeinfo$(PPUEXT) : $(INC)/exeinfo.pp strings$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/exeinfo.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
charset$(PPUEXT) : $(INC)/charset.pp objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/charset.pp
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
cpall$(PPUEXT): $(RTL)/charmaps/cpall.pas system$(PPUEXT) charset$(PPUEXT)
|
||||||
|
$(COMPILER) -Fu$(INC) -Fi$(RTL)/charmaps $(RTL)/charmaps/cpall.pas
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
fpwidestring$(PPUEXT): $(OBJPASDIR)/fpwidestring.pp unicodedata$(PPUEXT) charset$(PPUEXT) objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(OBJPASDIR)/fpwidestring.pp
|
||||||
|
|
||||||
|
character$(PPUEXT): $(OBJPASDIR)/character.pas sysutils$(PPUEXT) objpas$(PPUEXT) rtlconsts$(PPUEXT) unicodedata$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) $(OBJPASDIR)/character.pas
|
||||||
|
|
||||||
|
unicodenumtable$(PPUEXT) : $(OBJPASDIR)/unicodenumtable.pas objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) -Fi$(OBJPASDIR) $(OBJPASDIR)/unicodenumtable.pas
|
||||||
|
|
||||||
|
unicodedata$(PPUEXT) : $(OBJPASDIR)/unicodedata.pas unicodenumtable$(PPUEXT) objpas$(PPUEXT) system$(PPUEXT)
|
||||||
|
$(COMPILER) -Fi$(OBJPASDIR) $(OBJPASDIR)/unicodedata.pas
|
||||||
|
|
||||||
|
sortbase$(PPUEXT) : $(INC)/sortbase.pp objpas$(PPUEXT) $(SYSTEMUNIT)$(PPUEXT)
|
||||||
|
$(COMPILER) $<
|
||||||
|
|
||||||
|
#
|
||||||
|
# Other system-dependent RTL Units
|
||||||
|
#
|
||||||
|
msmouse$(PPUEXT) : msmouse.pp system$(PPUEXT)
|
||||||
|
$(COMPILER) msmouse.pp $(REDIR)
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
ctypes$(PPUEXT) : $(INC)/ctypes.pp system$(PPUEXT)
|
||||||
|
$(COMPILER) $(INC)/ctypes.pp $(REDIR)
|
||||||
|
$(EXECPPAS)
|
||||||
|
|
||||||
|
si_prc$(PPUEXT) : system$(PPUEXT)
|
||||||
|
$(COMPILER) si_prc.pp
|
10
rtl/msxdos/registers.inc
Normal file
10
rtl/msxdos/registers.inc
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
{ Registers record used by Intr and MsxDos. This include file is shared between
|
||||||
|
the system unit and the dos unit. }
|
||||||
|
|
||||||
|
type
|
||||||
|
Registers = packed record
|
||||||
|
case Integer of
|
||||||
|
0: (BC, DE, AF, HL, IX, IY: Word);
|
||||||
|
1: (C, B, E, D, Flags, A, L, H, IXl, IXh, IYl, IYh: Byte);
|
||||||
|
end;
|
||||||
|
|
24
rtl/msxdos/rtldefs.inc
Normal file
24
rtl/msxdos/rtldefs.inc
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 2012 by Free Pascal development team
|
||||||
|
|
||||||
|
This file contains platform-specific defines that are used in
|
||||||
|
multiple RTL units.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
{ the single byte OS APIs always use UTF-8 }
|
||||||
|
{ define FPCRTL_FILESYSTEM_UTF8}
|
||||||
|
|
||||||
|
{ The OS supports a single byte file system operations API that we use }
|
||||||
|
{$define FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||||||
|
|
||||||
|
{ The OS supports a two byte file system operations API that we use }
|
||||||
|
{ define FPCRTL_FILESYSTEM_TWO_BYTE_API}
|
36
rtl/msxdos/si_prc.pp
Normal file
36
rtl/msxdos/si_prc.pp
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 2020 by Free Pascal development team
|
||||||
|
|
||||||
|
This file contains startup code for the ZX Spectrum
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
unit si_prc;
|
||||||
|
|
||||||
|
{$SMARTLINK OFF}
|
||||||
|
|
||||||
|
interface
|
||||||
|
|
||||||
|
implementation
|
||||||
|
|
||||||
|
var
|
||||||
|
stktop: word; external name '__stktop';
|
||||||
|
|
||||||
|
procedure PascalMain; external name 'PASCALMAIN';
|
||||||
|
|
||||||
|
{ this *must* always remain the first procedure with code in this unit }
|
||||||
|
procedure _start; assembler; nostackframe; public name 'start';
|
||||||
|
asm
|
||||||
|
ld (stktop), sp
|
||||||
|
jp PASCALMAIN
|
||||||
|
end;
|
||||||
|
|
||||||
|
end.
|
153
rtl/msxdos/sysdir.inc
Normal file
153
rtl/msxdos/sysdir.inc
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 1999-2000 by Florian Klaempfl and Pavel Ozerski
|
||||||
|
member of the Free Pascal development team.
|
||||||
|
|
||||||
|
FPC Pascal system unit for the Win32 API.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
{*****************************************************************************
|
||||||
|
Directory Handling
|
||||||
|
*****************************************************************************}
|
||||||
|
|
||||||
|
{$ifdef todo}
|
||||||
|
procedure DosDir(func:byte;s: rawbytestring);
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
len : Integer;
|
||||||
|
begin
|
||||||
|
DoDirSeparators(s);
|
||||||
|
{ True DOS does not like backslashes at end
|
||||||
|
Win95 DOS accepts this !!
|
||||||
|
but "\" and "c:\" should still be kept and accepted hopefully PM }
|
||||||
|
len:=length(s);
|
||||||
|
if (len>0) and (s[len]='\') and
|
||||||
|
Not ((len=1) or ((len=3) and (s[2]=':'))) then
|
||||||
|
s[len]:=#0;
|
||||||
|
regs.DX:=Ofs(s[1]);
|
||||||
|
regs.DS:=Seg(s[1]);
|
||||||
|
if LFNSupport then
|
||||||
|
regs.AX:=$7100+func
|
||||||
|
else
|
||||||
|
regs.AX:=func shl 8;
|
||||||
|
MsDos(regs);
|
||||||
|
if (regs.Flags and fCarry) <> 0 then
|
||||||
|
GetInOutRes(regs.AX);
|
||||||
|
end;
|
||||||
|
{$endif}
|
||||||
|
|
||||||
|
Procedure do_MkDir(const s: rawbytestring);
|
||||||
|
begin
|
||||||
|
GetInOutRes(153);
|
||||||
|
//DosDir($39,s);
|
||||||
|
end;
|
||||||
|
|
||||||
|
Procedure do_RmDir(const s: rawbytestring);
|
||||||
|
begin
|
||||||
|
if s='.' then
|
||||||
|
begin
|
||||||
|
InOutRes:=16;
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
GetInOutRes(153);
|
||||||
|
//DosDir($3a,s);
|
||||||
|
end;
|
||||||
|
|
||||||
|
Procedure do_ChDir(const s: rawbytestring);
|
||||||
|
{$ifdef todo}
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
len : Integer;
|
||||||
|
{$endif}
|
||||||
|
begin
|
||||||
|
GetInOutRes(153);
|
||||||
|
{$ifdef todo}
|
||||||
|
len:=Length(s);
|
||||||
|
{ First handle Drive changes }
|
||||||
|
if (len>=2) and (s[2]=':') then
|
||||||
|
begin
|
||||||
|
regs.DX:=(ord(s[1]) and (not 32))-ord('A');
|
||||||
|
regs.AX:=$0e00;
|
||||||
|
MsDos(regs);
|
||||||
|
regs.AX:=$1900;
|
||||||
|
MsDos(regs);
|
||||||
|
if regs.AL<>regs.DL then
|
||||||
|
begin
|
||||||
|
Inoutres:=15;
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
{ DosDir($3b,'c:') give Path not found error on
|
||||||
|
pure DOS PM }
|
||||||
|
if len=2 then
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
{ do the normal dos chdir }
|
||||||
|
DosDir($3b,s);
|
||||||
|
{$endif}
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure do_GetDir (DriveNr: byte; var Dir: RawByteString);
|
||||||
|
{$ifdef todo}
|
||||||
|
var
|
||||||
|
temp : array[0..260] of char;
|
||||||
|
i : integer;
|
||||||
|
regs : Registers;
|
||||||
|
{$endif}
|
||||||
|
begin
|
||||||
|
GetInOutRes(153);
|
||||||
|
{$ifdef todo}
|
||||||
|
regs.DX:=drivenr;
|
||||||
|
regs.SI:=Ofs(temp);
|
||||||
|
regs.DS:=Seg(temp);
|
||||||
|
if LFNSupport then
|
||||||
|
regs.AX:=$7147
|
||||||
|
else
|
||||||
|
regs.AX:=$4700;
|
||||||
|
MsDos(regs);
|
||||||
|
if (regs.Flags and fCarry) <> 0 then
|
||||||
|
Begin
|
||||||
|
GetInOutRes (regs.AX);
|
||||||
|
Dir := char (DriveNr + 64) + ':\';
|
||||||
|
SetCodePage (Dir,DefaultFileSystemCodePage,false);
|
||||||
|
exit;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
temp[252] := #0; { to avoid shortstring buffer overflow }
|
||||||
|
{ conversion to Pascal string including slash conversion }
|
||||||
|
i:=0;
|
||||||
|
SetLength(dir,260);
|
||||||
|
while (temp[i]<>#0) do
|
||||||
|
begin
|
||||||
|
if temp[i] in AllowDirectorySeparators then
|
||||||
|
temp[i]:=DirectorySeparator;
|
||||||
|
dir[i+4]:=temp[i];
|
||||||
|
inc(i);
|
||||||
|
end;
|
||||||
|
dir[2]:=':';
|
||||||
|
dir[3]:='\';
|
||||||
|
SetLength(dir,i+3);
|
||||||
|
SetCodePage (dir,DefaultFileSystemCodePage,false);
|
||||||
|
{ upcase the string }
|
||||||
|
if not FileNameCasePreserving then
|
||||||
|
dir:=upcase(dir);
|
||||||
|
if drivenr<>0 then { Drive was supplied. We know it }
|
||||||
|
dir[1]:=char(65+drivenr-1)
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
{ We need to get the current drive from DOS function 19H }
|
||||||
|
{ because the drive was the default, which can be unknown }
|
||||||
|
regs.AX:=$1900;
|
||||||
|
MsDos(regs);
|
||||||
|
i:= (regs.AX and $ff) + ord('A');
|
||||||
|
dir[1]:=chr(i);
|
||||||
|
end;
|
||||||
|
{$endif}
|
||||||
|
end;
|
373
rtl/msxdos/sysfile.inc
Normal file
373
rtl/msxdos/sysfile.inc
Normal file
@ -0,0 +1,373 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 2001 by Free Pascal development team
|
||||||
|
|
||||||
|
Low leve file functions
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
{ Keep Track of open files }
|
||||||
|
const
|
||||||
|
max_files = 50;
|
||||||
|
var
|
||||||
|
openfiles : array [0..max_files-1] of boolean;
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
opennames : array [0..max_files-1] of pchar;
|
||||||
|
const
|
||||||
|
free_closed_names : boolean = true;
|
||||||
|
verbose_files : boolean = true;
|
||||||
|
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
|
||||||
|
|
||||||
|
{****************************************************************************
|
||||||
|
Low level File Routines
|
||||||
|
****************************************************************************}
|
||||||
|
|
||||||
|
procedure do_close(handle : thandle);
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
if Handle <= 4 then
|
||||||
|
exit;
|
||||||
|
regs.A := 0;
|
||||||
|
regs.B := Byte(handle);
|
||||||
|
if handle < max_files then
|
||||||
|
begin
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
if not openfiles[handle] then
|
||||||
|
Writeln(stderr,'Trying to close file h=',handle,' marked as closed');
|
||||||
|
if assigned(opennames[handle]) and free_closed_names then
|
||||||
|
begin
|
||||||
|
if verbose_files then
|
||||||
|
Writeln(stderr,'file ',opennames[handle],' closed');
|
||||||
|
|
||||||
|
sysfreememsize(opennames[handle],strlen(opennames[handle])+1);
|
||||||
|
opennames[handle]:=nil;
|
||||||
|
end;
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
openfiles[handle]:=false;
|
||||||
|
end;
|
||||||
|
regs.C := $45;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
begin
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
if verbose_files then
|
||||||
|
Writeln(stderr,'file close failed A = ',regs.A);
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
procedure do_erase(p : pchar; pchangeable: boolean);
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
oldp: pchar;
|
||||||
|
begin
|
||||||
|
oldp := p;
|
||||||
|
DoDirSeparators(p, pchangeable);
|
||||||
|
regs.A := 0;
|
||||||
|
regs.C := $4D;
|
||||||
|
regs.DE := PtrUInt(p);
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
if p <> oldp then
|
||||||
|
freemem(p);
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
procedure do_rename(p1,p2 : pchar; p1changeable, p2changeable: boolean);
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
oldp1, oldp2: pchar;
|
||||||
|
begin
|
||||||
|
oldp1 := p1;
|
||||||
|
oldp2 := p2;
|
||||||
|
DoDirSeparators(p1, p1changeable);
|
||||||
|
DoDirSeparators(p2, p2changeable);
|
||||||
|
regs.A := 0;
|
||||||
|
regs.C := $4E;
|
||||||
|
{ ToDo: check for same directory? }
|
||||||
|
regs.DE := PtrUInt(p1);
|
||||||
|
regs.HL := PtrUInt(p2);
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
if p1 <> oldp1 then
|
||||||
|
freemem(p1);
|
||||||
|
if p2 <> oldp2 then
|
||||||
|
freemem(p2);
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function do_write(h:thandle;addr:pointer;len : longint) : longint;
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $49;
|
||||||
|
regs.A := 0;
|
||||||
|
regs.B := h;
|
||||||
|
regs.DE := PtrUInt(addr);
|
||||||
|
regs.HL := len;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
begin
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
exit(0);
|
||||||
|
end;
|
||||||
|
do_write := regs.HL;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function do_read(h:thandle;addr:pointer;len : longint) : longint;
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $48;
|
||||||
|
regs.A := 0;
|
||||||
|
regs.B := h;
|
||||||
|
regs.DE := PtrUInt(addr);
|
||||||
|
regs.HL := len;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
begin
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
exit(0);
|
||||||
|
end;
|
||||||
|
do_read := regs.HL;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function do_filepos(handle : thandle) : longint;
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $4A;
|
||||||
|
regs.B := handle;
|
||||||
|
regs.A := 1;
|
||||||
|
regs.DE := 0;
|
||||||
|
regs.HL := 0;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
begin
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
do_filepos := 0;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
do_filepos := (longint(regs.DE) shl 16) + regs.HL;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
procedure do_seek(handle:thandle;pos : longint);
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $4A;
|
||||||
|
regs.B := handle;
|
||||||
|
regs.A := 0;
|
||||||
|
regs.DE := pos shr 16;
|
||||||
|
regs.HL := pos and $ffff;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
function do_seekend(handle:thandle):longint;
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $4A;
|
||||||
|
regs.B := handle;
|
||||||
|
regs.A := 2;
|
||||||
|
regs.DE := 0;
|
||||||
|
regs.HL := 0;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
begin
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
do_seekend := 0;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
do_seekend := (longint(regs.DE) shl 16) + regs.HL;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function do_filesize(handle : thandle) : longint;
|
||||||
|
var
|
||||||
|
aktfilepos : longint;
|
||||||
|
begin
|
||||||
|
aktfilepos:=do_filepos(handle);
|
||||||
|
do_filesize:=do_seekend(handle);
|
||||||
|
do_seek(handle,aktfilepos);
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
{ truncate at a given position }
|
||||||
|
procedure do_truncate (handle:thandle;pos:longint);
|
||||||
|
{var
|
||||||
|
regs : Registers;}
|
||||||
|
begin
|
||||||
|
GetInOutRes(153);
|
||||||
|
{do_seek(handle,pos);
|
||||||
|
regs.C:=??;
|
||||||
|
regs.B:=handle;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
GetInOutRes(regs.A);}
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure do_open(var f;p:pchar;flags:longint; pchangeable: boolean);
|
||||||
|
{
|
||||||
|
filerec and textrec have both handle and mode as the first items so
|
||||||
|
they could use the same routine for opening/creating.
|
||||||
|
when (flags and $100) the file will be append
|
||||||
|
when (flags and $1000) the file will be truncate/rewritten
|
||||||
|
when (flags and $10000) there is no check for close (needed for textfiles)
|
||||||
|
}
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
action : word;
|
||||||
|
oldp : pchar;
|
||||||
|
begin
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
if verbose_files then
|
||||||
|
Writeln(stderr,'do_open for file "',p,'" called');
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
{ close first if opened }
|
||||||
|
if ((flags and $10000)=0) then
|
||||||
|
begin
|
||||||
|
case filerec(f).mode of
|
||||||
|
fminput,fmoutput,fminout : Do_Close(filerec(f).handle);
|
||||||
|
fmclosed : ;
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
inoutres:=102; {not assigned}
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
{ reset file handle }
|
||||||
|
filerec(f).handle:=UnusedHandle;
|
||||||
|
action:=$1;
|
||||||
|
{ convert filemode to filerec modes }
|
||||||
|
regs.A:=0;
|
||||||
|
case (flags and 3) of
|
||||||
|
0 : begin
|
||||||
|
filerec(f).mode:=fminput;
|
||||||
|
{ b1 -> no write }
|
||||||
|
regs.A := 1;
|
||||||
|
end;
|
||||||
|
1 : begin
|
||||||
|
filerec(f).mode:=fmoutput;
|
||||||
|
{ b2 -> no read }
|
||||||
|
regs.A := 2;
|
||||||
|
end;
|
||||||
|
2 : filerec(f).mode:=fminout;
|
||||||
|
end;
|
||||||
|
if (flags and $1000)<>0 then
|
||||||
|
action:=$12; {create file function}
|
||||||
|
{ empty name is special }
|
||||||
|
if p[0]=#0 then
|
||||||
|
begin
|
||||||
|
case FileRec(f).mode of
|
||||||
|
fminput :
|
||||||
|
FileRec(f).Handle:=StdInputHandle;
|
||||||
|
fminout, { this is set by rewrite }
|
||||||
|
fmoutput :
|
||||||
|
FileRec(f).Handle:=StdOutputHandle;
|
||||||
|
fmappend :
|
||||||
|
begin
|
||||||
|
FileRec(f).Handle:=StdOutputHandle;
|
||||||
|
FileRec(f).mode:=fmoutput; {fool fmappend}
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
oldp:=p;
|
||||||
|
DoDirSeparators(p,pchangeable);
|
||||||
|
|
||||||
|
if (action and $00f0) <> 0 then
|
||||||
|
regs.C := $44 { Map to Create/Replace API }
|
||||||
|
else
|
||||||
|
regs.C := $43; { Map to Open_Existing API }
|
||||||
|
|
||||||
|
regs.B := 0;
|
||||||
|
|
||||||
|
MsxDos(regs);
|
||||||
|
|
||||||
|
if regs.A <> 0 then
|
||||||
|
begin
|
||||||
|
FileRec(f).mode:=fmclosed;
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
if oldp<>p then
|
||||||
|
freemem(p);
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
if verbose_files then
|
||||||
|
Writeln(stderr,'MSXDOS INT open for file "',p,'" failed err=',regs.A);
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
exit;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
filerec(f).handle:=regs.B;
|
||||||
|
end;
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
if verbose_files then
|
||||||
|
Writeln(stderr,'MSXDOS INT open for file "',p,'" returned ',regs.B);
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
if regs.B<max_files then
|
||||||
|
begin
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
if openfiles[regs.B] and
|
||||||
|
assigned(opennames[regs.B]) then
|
||||||
|
begin
|
||||||
|
Writeln(stderr,'file ',opennames[regs.B],'(',regs.B,') not closed but handle reused!');
|
||||||
|
sysfreememsize(opennames[regs.B],strlen(opennames[regs.B])+1);
|
||||||
|
end;
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
openfiles[regs.B]:=true;
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
opennames[regs.B] := sysgetmem(strlen(p)+1);
|
||||||
|
move(p^,opennames[regs.B]^,strlen(p)+1);
|
||||||
|
if verbose_files then
|
||||||
|
Writeln(stderr,'file ',opennames[regs.B],' opened');
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
end;
|
||||||
|
{ append mode }
|
||||||
|
if ((flags and $100) <> 0) and
|
||||||
|
(FileRec (F).Handle <> UnusedHandle) then
|
||||||
|
begin
|
||||||
|
do_seekend(filerec(f).handle);
|
||||||
|
filerec(f).mode:=fmoutput; {fool fmappend}
|
||||||
|
end;
|
||||||
|
|
||||||
|
if oldp<>p then
|
||||||
|
freemem(p);
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function do_isdevice(handle:THandle):boolean;
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $4B;
|
||||||
|
regs.B := handle;
|
||||||
|
regs.A := $00;
|
||||||
|
MsxDos(regs);
|
||||||
|
do_isdevice := (regs.D and $80) <> 0;
|
||||||
|
if regs.A <> 0 then
|
||||||
|
GetInOutRes(regs.A);
|
||||||
|
end;
|
||||||
|
|
117
rtl/msxdos/sysheap.inc
Normal file
117
rtl/msxdos/sysheap.inc
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 2001 by Free Pascal development team
|
||||||
|
|
||||||
|
This file implements all the base types and limits required
|
||||||
|
for a minimal POSIX compliant subset required to port the compiler
|
||||||
|
to a new OS.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
|
||||||
|
{*****************************************************************************
|
||||||
|
Heap Management
|
||||||
|
*****************************************************************************}
|
||||||
|
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
{ Internal structure used by MSDOS }
|
||||||
|
type
|
||||||
|
MCB = packed record
|
||||||
|
sig : char;
|
||||||
|
psp : word;
|
||||||
|
paragraphs : word;
|
||||||
|
res : array [0..2] of char;
|
||||||
|
exename : array [0..7] of char;
|
||||||
|
end;
|
||||||
|
PMCB = ^MCB;
|
||||||
|
{$endif def DEBUG_TINY_HEAP}
|
||||||
|
|
||||||
|
|
||||||
|
function SysOSAlloc (size: ptruint): pointer;
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
nb_para : longint;
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
p : pmcb;
|
||||||
|
i : byte;
|
||||||
|
{$endif def DEBUG_TINY_HEAP}
|
||||||
|
begin
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
writeln('SysOSAlloc called size=',size);
|
||||||
|
{$endif}
|
||||||
|
{$if defined(FPC_X86_DATA_FAR) or defined(FPC_X86_DATA_HUGE)}
|
||||||
|
regs.ax:=$4800;
|
||||||
|
nb_para:=size div 16;
|
||||||
|
if nb_para > $ffff then
|
||||||
|
begin
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
writeln('SysOSAlloc size too big = ',size);
|
||||||
|
{$endif}
|
||||||
|
result:=nil;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
regs.bx:=nb_para;
|
||||||
|
msdos(regs);
|
||||||
|
if (regs.Flags and fCarry) <> 0 then
|
||||||
|
begin
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
writeln('SysOSAlloc failed, err = ',regs.AX);
|
||||||
|
{$endif}
|
||||||
|
{ Do not set InOutRes if ReturnNilIfGrowHeapFails is set }
|
||||||
|
if not ReturnNilIfGrowHeapFails then
|
||||||
|
GetInOutRes(regs.AX);
|
||||||
|
Result := nil;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
result:=ptr(regs.ax,0);
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
writeln('SysOSAlloc returned= $',hexstr(regs.ax,4),':$0');
|
||||||
|
p:=ptr(regs.ax-1,0);
|
||||||
|
writeln('Possibly prev MCB: at ',hexstr(p));
|
||||||
|
writeln(' sig=',p^.sig);
|
||||||
|
writeln(' psp=$',hexstr(p^.psp,4));
|
||||||
|
writeln(' paragraphs=',p^.paragraphs);
|
||||||
|
if (p^.exename[0]<>#0) then
|
||||||
|
begin
|
||||||
|
write(' name=');
|
||||||
|
for i:=0 to 7 do
|
||||||
|
if ord(p^.exename[i])>31 then
|
||||||
|
write(p^.exename[i]);
|
||||||
|
writeln;
|
||||||
|
end;
|
||||||
|
p:=ptr(regs.ax+p^.paragraphs,0);
|
||||||
|
writeln('Possibly next MCB: at ',hexstr(p));
|
||||||
|
writeln(' sig=',p^.sig);
|
||||||
|
writeln(' psp=$',hexstr(p^.psp,4));
|
||||||
|
writeln(' paragraphs=',p^.paragraphs);
|
||||||
|
if (p^.exename[0]<>#0) then
|
||||||
|
begin
|
||||||
|
write(' name=');
|
||||||
|
for i:=0 to 7 do
|
||||||
|
if ord(p^.exename[i])>31 then
|
||||||
|
write(p^.exename[i]);
|
||||||
|
writeln;
|
||||||
|
end;
|
||||||
|
{$endif}
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
{$else not DATA_FAR}
|
||||||
|
{$ifdef DEBUG_TINY_HEAP}
|
||||||
|
writeln('SysOSAlloc cannot be used in small data models');
|
||||||
|
{$endif}
|
||||||
|
Result := nil;
|
||||||
|
{$endif not DATA_FAR}
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure SysOSFree(p: pointer; size: ptruint);
|
||||||
|
begin
|
||||||
|
end;
|
33
rtl/msxdos/sysos.inc
Normal file
33
rtl/msxdos/sysos.inc
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 2013 by Free Pascal development team
|
||||||
|
|
||||||
|
This file implements all the base types and limits required
|
||||||
|
for a minimal POSIX compliant subset required to port the compiler
|
||||||
|
to a new OS.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
procedure GetInOutRes(def: Word);
|
||||||
|
var
|
||||||
|
regs : Registers;
|
||||||
|
begin
|
||||||
|
regs.C:=$65;
|
||||||
|
MsxDos(regs);
|
||||||
|
InOutRes:=regs.B;
|
||||||
|
case InOutRes of
|
||||||
|
19 : InOutRes:=150;
|
||||||
|
21 : InOutRes:=152;
|
||||||
|
32 : InOutRes:=5;
|
||||||
|
end;
|
||||||
|
if InOutRes=0 then
|
||||||
|
InOutRes:=Def;
|
||||||
|
end;
|
||||||
|
|
28
rtl/msxdos/sysosh.inc
Normal file
28
rtl/msxdos/sysosh.inc
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
{
|
||||||
|
This file is part of the Free Pascal run time library.
|
||||||
|
Copyright (c) 2013 by Free Pascal development team
|
||||||
|
|
||||||
|
This file implements all the base types and limits required
|
||||||
|
for a minimal POSIX compliant subset required to port the compiler
|
||||||
|
to a new OS.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
**********************************************************************}
|
||||||
|
|
||||||
|
{Platform specific information}
|
||||||
|
type
|
||||||
|
THandle = Byte;
|
||||||
|
TThreadID = THandle;
|
||||||
|
TOSTimestamp = Longint;
|
||||||
|
|
||||||
|
PRTLCriticalSection = ^TRTLCriticalSection;
|
||||||
|
TRTLCriticalSection = record
|
||||||
|
Locked: boolean
|
||||||
|
end;
|
||||||
|
|
722
rtl/msxdos/system.pp
Normal file
722
rtl/msxdos/system.pp
Normal file
@ -0,0 +1,722 @@
|
|||||||
|
unit System;
|
||||||
|
|
||||||
|
|
||||||
|
interface
|
||||||
|
|
||||||
|
{$define FPC_IS_SYSTEM}
|
||||||
|
{ The heap for MSDOS is implemented
|
||||||
|
in tinyheap.inc include file,
|
||||||
|
but it uses default SysGetMem names }
|
||||||
|
|
||||||
|
{$define HAS_MEMORYMANAGER}
|
||||||
|
{ define TEST_FPU_INT10 to force keeping local int10,
|
||||||
|
for testing purpose only }
|
||||||
|
|
||||||
|
|
||||||
|
{$DEFINE FPC_INCLUDE_SOFTWARE_MUL}
|
||||||
|
{$DEFINE FPC_INCLUDE_SOFTWARE_MOD_DIV}
|
||||||
|
|
||||||
|
{$DEFINE FPC_USE_SMALL_DEFAULTSTACKSIZE}
|
||||||
|
{ To avoid warnings in thread.inc code,
|
||||||
|
but value must be really given after
|
||||||
|
systemh.inc is included otherwise the
|
||||||
|
$mode switch is not effective }
|
||||||
|
|
||||||
|
{ Use Ansi Char for files }
|
||||||
|
{$define FPC_ANSI_TEXTFILEREC}
|
||||||
|
{$define FPC_STDOUT_TRUE_ALIAS}
|
||||||
|
|
||||||
|
{$ifdef NO_WIDESTRINGS}
|
||||||
|
{ Do NOT use wide Char for files }
|
||||||
|
{$undef FPC_HAS_FEATURE_WIDESTRINGS}
|
||||||
|
{$endif NO_WIDESTRINGS}
|
||||||
|
|
||||||
|
{$I systemh.inc}
|
||||||
|
{$I tnyheaph.inc}
|
||||||
|
{.$I portsh.inc}
|
||||||
|
|
||||||
|
{$ifndef FPUNONE}
|
||||||
|
{$ifdef FPC_HAS_FEATURE_SOFTFPU}
|
||||||
|
|
||||||
|
{$define fpc_softfpu_interface}
|
||||||
|
{$i softfpu.pp}
|
||||||
|
{$undef fpc_softfpu_interface}
|
||||||
|
|
||||||
|
{$endif FPC_HAS_FEATURE_SOFTFPU}
|
||||||
|
{$endif FPUNONE}
|
||||||
|
|
||||||
|
const
|
||||||
|
LineEnding = #13#10;
|
||||||
|
{ LFNSupport is a variable here, defined below!!! }
|
||||||
|
DirectorySeparator = '\';
|
||||||
|
DriveSeparator = ':';
|
||||||
|
ExtensionSeparator = '.';
|
||||||
|
PathSeparator = ';';
|
||||||
|
AllowDirectorySeparators : set of char = ['\','/'];
|
||||||
|
AllowDriveSeparators : set of char = [':'];
|
||||||
|
{ FileNameCaseSensitive and FileNameCasePreserving are defined separately below!!! }
|
||||||
|
maxExitCode = 255;
|
||||||
|
MaxPathLen = 256;
|
||||||
|
|
||||||
|
const
|
||||||
|
{ Default filehandles }
|
||||||
|
UnusedHandle = $ffff;{ instead of -1, as it is a word value}
|
||||||
|
StdInputHandle = 0;
|
||||||
|
StdOutputHandle = 1;
|
||||||
|
{ MSX-DOS does not have a separate StdErr }
|
||||||
|
StdErrorHandle = 1;
|
||||||
|
|
||||||
|
FileNameCaseSensitive : boolean = false;
|
||||||
|
FileNameCasePreserving: boolean = false;
|
||||||
|
CtrlZMarksEOF: boolean = true; (* #26 is considered as end of file *)
|
||||||
|
|
||||||
|
sLineBreak = LineEnding;
|
||||||
|
DefaultTextLineBreakStyle : TTextLineBreakStyle = tlbsCRLF;
|
||||||
|
|
||||||
|
var
|
||||||
|
{ Mem[] support }
|
||||||
|
mem : array[0..$7fff-1] of byte absolute $0;
|
||||||
|
memw : array[0..($7fff div sizeof(word))-1] of word absolute $0;
|
||||||
|
meml : array[0..($7fff div sizeof(longint))-1] of longint absolute $0;
|
||||||
|
{ C-compatible arguments and environment }
|
||||||
|
argc:smallint; //!! public name 'operatingsystem_parameter_argc';
|
||||||
|
argv:PPchar; //!! public name 'operatingsystem_parameter_argv';
|
||||||
|
|
||||||
|
{ The DOS Program Segment Prefix segment (TP7 compatibility) }
|
||||||
|
PrefixSeg:Word;public name '__fpc_PrefixSeg';
|
||||||
|
|
||||||
|
SaveInt00: FarPointer;public name '__SaveInt00';
|
||||||
|
SaveInt10: FarPointer;public name '__SaveInt10';
|
||||||
|
SaveInt75: FarPointer;public name '__SaveInt75';
|
||||||
|
fpu_status: word;public name '__fpu_status';
|
||||||
|
|
||||||
|
const
|
||||||
|
AllFilesMask: string [3] = '*.*';
|
||||||
|
|
||||||
|
const
|
||||||
|
LFNSupport = false;
|
||||||
|
|
||||||
|
implementation
|
||||||
|
|
||||||
|
procedure DebugWrite(s: PChar); forward;
|
||||||
|
procedure DebugWrite(const S: string); forward;
|
||||||
|
procedure DebugWriteLn(const S: string); forward;
|
||||||
|
|
||||||
|
{$ifdef todo}
|
||||||
|
const
|
||||||
|
{ used for an offset fixup for accessing the proc parameters in asm routines
|
||||||
|
that use nostackframe. We can't use the parameter name directly, because
|
||||||
|
i8086 doesn't support sp relative addressing. }
|
||||||
|
{$ifdef FPC_X86_CODE_FAR}
|
||||||
|
extra_param_offset = 2;
|
||||||
|
{$else FPC_X86_CODE_FAR}
|
||||||
|
extra_param_offset = 0;
|
||||||
|
{$endif FPC_X86_CODE_FAR}
|
||||||
|
{$if defined(FPC_X86_DATA_FAR) or defined(FPC_X86_DATA_HUGE)}
|
||||||
|
extra_data_offset = 2;
|
||||||
|
{$else}
|
||||||
|
extra_data_offset = 0;
|
||||||
|
{$endif}
|
||||||
|
|
||||||
|
type
|
||||||
|
PFarByte = ^Byte;//far;
|
||||||
|
PFarChar = ^Char;//far;
|
||||||
|
PFarWord = ^Word;//far;
|
||||||
|
PPFarChar = ^PFarChar;
|
||||||
|
{$endif}
|
||||||
|
|
||||||
|
var
|
||||||
|
stklen: word; external name '__stklen';
|
||||||
|
|
||||||
|
__heapsize: Word;external name '__heapsize';
|
||||||
|
__fpc_initialheap: array[0..0] of byte;external name '__fpc_initialheap';
|
||||||
|
|
||||||
|
var
|
||||||
|
__stktop : pointer;public name '__stktop';
|
||||||
|
dos_version:Word;public name 'dos_version';
|
||||||
|
dos_env_count:smallint;public name '__dos_env_count';
|
||||||
|
dos_argv0 : PChar;public name '__fpc_dos_argv0';
|
||||||
|
|
||||||
|
{$I registers.inc}
|
||||||
|
|
||||||
|
procedure Intr(IntNo: Byte; var Regs: Registers); external name 'FPC_INTR';
|
||||||
|
|
||||||
|
procedure MsxDos(var Regs: Registers); assembler; nostackframe; public name 'FPC_MSXDOS';
|
||||||
|
asm
|
||||||
|
//in a, (0x2e)
|
||||||
|
{ store registers contents }
|
||||||
|
push AF
|
||||||
|
push BC
|
||||||
|
push DE
|
||||||
|
push HL
|
||||||
|
push IX
|
||||||
|
push IY
|
||||||
|
{ allocate an additional scratch space }
|
||||||
|
push IY
|
||||||
|
{ Regs now resides at SP + 16 }
|
||||||
|
|
||||||
|
{ IY is not used for parameters, so base everything on that;
|
||||||
|
for that we need to load the address of Regs into IY }
|
||||||
|
ld IX, 0x10
|
||||||
|
add IX, SP
|
||||||
|
|
||||||
|
ld L,(IX+0)
|
||||||
|
ld H,(IX+1)
|
||||||
|
|
||||||
|
push HL
|
||||||
|
pop IY
|
||||||
|
|
||||||
|
{ fill IX with the help of HL }
|
||||||
|
ld L,(IY+8)
|
||||||
|
ld H,(IY+9)
|
||||||
|
|
||||||
|
push HL
|
||||||
|
pop IX
|
||||||
|
|
||||||
|
ld B,(IY+1)
|
||||||
|
ld C,(IY+0)
|
||||||
|
ld D,(IY+3)
|
||||||
|
ld E,(IY+2)
|
||||||
|
// load A last
|
||||||
|
//ld A,(IY+4)
|
||||||
|
ld H,(IY+7)
|
||||||
|
ld L,(IY+6)
|
||||||
|
|
||||||
|
ld A,(IY+4)
|
||||||
|
|
||||||
|
{ store IY to scratch location }
|
||||||
|
ex (SP),IY
|
||||||
|
|
||||||
|
{ call to DOS }
|
||||||
|
call 0x0005
|
||||||
|
|
||||||
|
{ store IY to scratch and restore pointer address of Regs }
|
||||||
|
ex (SP),IY
|
||||||
|
|
||||||
|
ld (IY+1),B
|
||||||
|
ld (IY+0),C
|
||||||
|
ld (IY+3),D
|
||||||
|
ld (IY+2),E
|
||||||
|
ld (IY+4),A
|
||||||
|
// skip F
|
||||||
|
ld (IY+7),H
|
||||||
|
ld (IY+6),L
|
||||||
|
|
||||||
|
{ store IX with the help of HL }
|
||||||
|
push IX
|
||||||
|
pop HL
|
||||||
|
ld (IY+8),L
|
||||||
|
ld (IY+9),H
|
||||||
|
|
||||||
|
{ store the stored IY with the help of HL }
|
||||||
|
ex (SP),HL
|
||||||
|
|
||||||
|
ld (IY+10),L
|
||||||
|
ld (IY+11),H
|
||||||
|
|
||||||
|
{ cleanup stack }
|
||||||
|
pop IY
|
||||||
|
pop IY
|
||||||
|
pop IX
|
||||||
|
pop HL
|
||||||
|
pop DE
|
||||||
|
pop BC
|
||||||
|
pop AF
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure InstallInterruptHandlers; external name 'FPC_INSTALL_INTERRUPT_HANDLERS';
|
||||||
|
procedure RestoreInterruptHandlers; external name 'FPC_RESTORE_INTERRUPT_HANDLERS';
|
||||||
|
|
||||||
|
function CheckNullArea: Boolean; external name 'FPC_CHECK_NULLAREA';
|
||||||
|
|
||||||
|
{$I system.inc}
|
||||||
|
|
||||||
|
{$I tinyheap.inc}
|
||||||
|
|
||||||
|
{.$I ports.inc}
|
||||||
|
|
||||||
|
{$ifndef FPUNONE}
|
||||||
|
{$ifdef FPC_HAS_FEATURE_SOFTFPU}
|
||||||
|
|
||||||
|
{$define fpc_softfpu_implementation}
|
||||||
|
{$i softfpu.pp}
|
||||||
|
{$undef fpc_softfpu_implementation}
|
||||||
|
|
||||||
|
{ we get these functions and types from the softfpu code }
|
||||||
|
{$define FPC_SYSTEM_HAS_float64}
|
||||||
|
{$define FPC_SYSTEM_HAS_float32}
|
||||||
|
{$define FPC_SYSTEM_HAS_flag}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat64Frac0}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat64Frac1}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat64Exp}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat64Frac}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat64Sign}
|
||||||
|
{$define FPC_SYSTEM_HAS_ExtractFloat32Frac}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat32Exp}
|
||||||
|
{$define FPC_SYSTEM_HAS_extractFloat32Sign}
|
||||||
|
|
||||||
|
{$endif FPC_HAS_FEATURE_SOFTFPU}
|
||||||
|
{$endif FPUNONE}
|
||||||
|
|
||||||
|
procedure DebugWrite(S: PChar);
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
while S^ <> #0 do begin
|
||||||
|
regs.C := $02;
|
||||||
|
regs.E := Ord(S^);
|
||||||
|
MsxDos(regs);
|
||||||
|
Inc(S);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure DebugWrite(const S: string);
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
i: Byte;
|
||||||
|
begin
|
||||||
|
for i := 1 to Length(S) do begin
|
||||||
|
regs.C := $02;
|
||||||
|
regs.E := Ord(S[i]);
|
||||||
|
MsxDos(regs);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure DebugWriteLn(const S: string);
|
||||||
|
begin
|
||||||
|
DebugWrite(S);
|
||||||
|
DebugWrite(#13#10);
|
||||||
|
end;
|
||||||
|
|
||||||
|
{*****************************************************************************
|
||||||
|
ParamStr/Randomize
|
||||||
|
*****************************************************************************}
|
||||||
|
|
||||||
|
var
|
||||||
|
internal_envp : PPChar = nil;
|
||||||
|
|
||||||
|
procedure setup_environment;
|
||||||
|
{$ifdef todo}
|
||||||
|
var
|
||||||
|
env_count : smallint;
|
||||||
|
cp, dos_env: PFarChar;
|
||||||
|
{$endif}
|
||||||
|
begin
|
||||||
|
{$ifdef todo}
|
||||||
|
env_count:=0;
|
||||||
|
dos_env:=Ptr(MemW[PrefixSeg:$2C], 0);
|
||||||
|
cp:=dos_env;
|
||||||
|
while cp^<>#0 do
|
||||||
|
begin
|
||||||
|
inc(env_count);
|
||||||
|
while (cp^ <> #0) do
|
||||||
|
inc(cp); { skip to NUL }
|
||||||
|
inc(cp); { skip to next character }
|
||||||
|
end;
|
||||||
|
internal_envp := getmem((env_count+1) * sizeof(PFarChar));
|
||||||
|
cp:=dos_env;
|
||||||
|
env_count:=0;
|
||||||
|
while cp^<>#0 do
|
||||||
|
begin
|
||||||
|
internal_envp[env_count] := cp;
|
||||||
|
inc(env_count);
|
||||||
|
while (cp^ <> #0) do
|
||||||
|
inc(cp); { skip to NUL }
|
||||||
|
inc(cp); { skip to next character }
|
||||||
|
end;
|
||||||
|
internal_envp[env_count]:=nil;
|
||||||
|
dos_env_count := env_count;
|
||||||
|
if dos_version >= $300 then
|
||||||
|
begin
|
||||||
|
if cp=dos_env then
|
||||||
|
inc(cp);
|
||||||
|
inc(cp, 3);
|
||||||
|
dos_argv0 := cp;
|
||||||
|
end
|
||||||
|
else
|
||||||
|
dos_argv0 := nil;
|
||||||
|
{$endif}
|
||||||
|
end;
|
||||||
|
|
||||||
|
function envp:PPChar;public name '__fpc_envp';
|
||||||
|
begin
|
||||||
|
if not assigned(internal_envp) then
|
||||||
|
setup_environment;
|
||||||
|
envp:=internal_envp;
|
||||||
|
end;
|
||||||
|
|
||||||
|
function GetEnvVar(aName: PChar): String;
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
i: SizeInt;
|
||||||
|
begin
|
||||||
|
SetLength(Result, 255);
|
||||||
|
regs.C := $6B;
|
||||||
|
regs.HL := PtrUInt(aName);
|
||||||
|
regs.DE := PtrUInt(@Result[1]);
|
||||||
|
regs.B := 255;
|
||||||
|
regs.A := 0;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A = 0 then begin
|
||||||
|
i := 1;
|
||||||
|
aName := PChar(@Result[1]);
|
||||||
|
while i < 256 do begin
|
||||||
|
if aName^ = #0 then begin
|
||||||
|
SetLength(Result, i);
|
||||||
|
Break;
|
||||||
|
end;
|
||||||
|
Inc(i);
|
||||||
|
Inc(aName);
|
||||||
|
end;
|
||||||
|
end else
|
||||||
|
SetLength(Result, 0);
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure setup_arguments;
|
||||||
|
var
|
||||||
|
i: SmallInt;
|
||||||
|
pc: PChar;
|
||||||
|
quote: Char;
|
||||||
|
count: SmallInt;
|
||||||
|
arglen, argv0len: SmallInt;
|
||||||
|
argblock: PChar;
|
||||||
|
arg: PChar;
|
||||||
|
doscmd : string[129]; { Dos commandline copied from PSP, max is 128 chars +1 for terminating zero }
|
||||||
|
tmp: String;
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
tmp := GetEnvVar('PROGRAM');
|
||||||
|
argv0len := Length(tmp);
|
||||||
|
|
||||||
|
tmp := GetEnvVar('PARAMETERS');
|
||||||
|
{$IfDef SYSTEM_DEBUG_STARTUP}
|
||||||
|
Writeln(stderr,'Dos command line is #',tmp,'# size = ',length(tmp));
|
||||||
|
{$EndIf }
|
||||||
|
{ parse dos commandline }
|
||||||
|
pc:=@tmp[1];
|
||||||
|
count:=1;
|
||||||
|
{ calc total arguments length and count }
|
||||||
|
arglen:=argv0len+1;
|
||||||
|
while pc^<>#0 do
|
||||||
|
begin
|
||||||
|
{ skip leading spaces }
|
||||||
|
while pc^ in [#1..#32] do
|
||||||
|
inc(pc);
|
||||||
|
if pc^=#0 then
|
||||||
|
break;
|
||||||
|
{ calc argument length }
|
||||||
|
quote:=' ';
|
||||||
|
while (pc^<>#0) do
|
||||||
|
begin
|
||||||
|
case pc^ of
|
||||||
|
#1..#32 :
|
||||||
|
begin
|
||||||
|
if quote<>' ' then
|
||||||
|
inc(arglen)
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
end;
|
||||||
|
'"' :
|
||||||
|
begin
|
||||||
|
if quote<>'''' then
|
||||||
|
begin
|
||||||
|
if pchar(pc+1)^<>'"' then
|
||||||
|
begin
|
||||||
|
if quote='"' then
|
||||||
|
quote:=' '
|
||||||
|
else
|
||||||
|
quote:='"';
|
||||||
|
end
|
||||||
|
else
|
||||||
|
inc(pc);
|
||||||
|
end
|
||||||
|
else
|
||||||
|
inc(arglen);
|
||||||
|
end;
|
||||||
|
'''' :
|
||||||
|
begin
|
||||||
|
if quote<>'"' then
|
||||||
|
begin
|
||||||
|
if pchar(pc+1)^<>'''' then
|
||||||
|
begin
|
||||||
|
if quote='''' then
|
||||||
|
quote:=' '
|
||||||
|
else
|
||||||
|
quote:='''';
|
||||||
|
end
|
||||||
|
else
|
||||||
|
inc(pc);
|
||||||
|
end
|
||||||
|
else
|
||||||
|
inc(arglen);
|
||||||
|
end;
|
||||||
|
else
|
||||||
|
inc(arglen);
|
||||||
|
end;
|
||||||
|
inc(pc);
|
||||||
|
end;
|
||||||
|
inc(arglen); { for the null terminator }
|
||||||
|
inc(count);
|
||||||
|
end;
|
||||||
|
Writeln(stderr,'Arg count: ', count, ', size: ', arglen);
|
||||||
|
{ set argc and allocate argv }
|
||||||
|
argc:=count;
|
||||||
|
argv:=AllocMem((count+1)*SizeOf(PChar));
|
||||||
|
{ allocate a single memory block for all arguments }
|
||||||
|
argblock:=GetMem(arglen);
|
||||||
|
writeln('Allocated arg vector at ', hexstr(argv), ' and block at ', hexstr(argblock));
|
||||||
|
{ create argv[0] }
|
||||||
|
argv[0]:=argblock;
|
||||||
|
arg:=argblock+argv0len;
|
||||||
|
|
||||||
|
arg^:=#0;
|
||||||
|
Inc(arg);
|
||||||
|
|
||||||
|
pc:=@tmp[1];
|
||||||
|
count:=1;
|
||||||
|
while pc^<>#0 do
|
||||||
|
begin
|
||||||
|
{ skip leading spaces }
|
||||||
|
while pc^ in [#1..#32] do
|
||||||
|
inc(pc);
|
||||||
|
if pc^=#0 then
|
||||||
|
break;
|
||||||
|
{ copy argument }
|
||||||
|
//writeln('Setting arg ',count,' to ', hexstr(arg));
|
||||||
|
asm
|
||||||
|
in a,(0x2e)
|
||||||
|
end ['a'];
|
||||||
|
argv[count]:=arg;
|
||||||
|
quote:=' ';
|
||||||
|
while (pc^<>#0) do
|
||||||
|
begin
|
||||||
|
case pc^ of
|
||||||
|
#1..#32 :
|
||||||
|
begin
|
||||||
|
if quote<>' ' then
|
||||||
|
begin
|
||||||
|
arg^:=pc^;
|
||||||
|
inc(arg);
|
||||||
|
end
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
end;
|
||||||
|
'"' :
|
||||||
|
begin
|
||||||
|
if quote<>'''' then
|
||||||
|
begin
|
||||||
|
if pchar(pc+1)^<>'"' then
|
||||||
|
begin
|
||||||
|
if quote='"' then
|
||||||
|
quote:=' '
|
||||||
|
else
|
||||||
|
quote:='"';
|
||||||
|
end
|
||||||
|
else
|
||||||
|
inc(pc);
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
arg^:=pc^;
|
||||||
|
inc(arg);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
'''' :
|
||||||
|
begin
|
||||||
|
if quote<>'"' then
|
||||||
|
begin
|
||||||
|
if pchar(pc+1)^<>'''' then
|
||||||
|
begin
|
||||||
|
if quote='''' then
|
||||||
|
quote:=' '
|
||||||
|
else
|
||||||
|
quote:='''';
|
||||||
|
end
|
||||||
|
else
|
||||||
|
inc(pc);
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
arg^:=pc^;
|
||||||
|
inc(arg);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
arg^:=pc^;
|
||||||
|
inc(arg);
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
inc(pc);
|
||||||
|
end;
|
||||||
|
arg^:=#0;
|
||||||
|
Inc(arg);
|
||||||
|
{$IfDef SYSTEM_DEBUG_STARTUP}
|
||||||
|
Writeln(stderr,'dos arg ',count,' #',strlen(argv[count]),'#',argv[count],'#');
|
||||||
|
{$EndIf SYSTEM_DEBUG_STARTUP}
|
||||||
|
inc(count);
|
||||||
|
end;
|
||||||
|
|
||||||
|
arg:=argblock;
|
||||||
|
tmp:=GetEnvVar('PROGRAM');
|
||||||
|
pc:=@tmp[1];
|
||||||
|
while pc^ <> #0 do
|
||||||
|
begin
|
||||||
|
arg^ := pc^;
|
||||||
|
Inc(arg);
|
||||||
|
Inc(pc);
|
||||||
|
end;
|
||||||
|
|
||||||
|
for count:=0 to argc-1 do
|
||||||
|
writeln('arg ',count,' at ',hexstr(argv[count]));
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function paramcount : longint;
|
||||||
|
begin
|
||||||
|
if argv=nil then
|
||||||
|
setup_arguments;
|
||||||
|
paramcount := argc - 1;
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
function paramstr(l : longint) : string;
|
||||||
|
begin
|
||||||
|
if argv=nil then
|
||||||
|
setup_arguments;
|
||||||
|
if (l>=0) and (l+1<=argc) then
|
||||||
|
paramstr:=strpas(argv[l])
|
||||||
|
else
|
||||||
|
paramstr:='';
|
||||||
|
end;
|
||||||
|
|
||||||
|
|
||||||
|
procedure randomize;
|
||||||
|
{$ifdef todo}
|
||||||
|
var
|
||||||
|
hl : longint;
|
||||||
|
regs : Registers;
|
||||||
|
{$endif}
|
||||||
|
begin
|
||||||
|
{$ifdef todo}
|
||||||
|
regs.AH:=$2C;
|
||||||
|
MsDos(regs);
|
||||||
|
hl:=regs.DX;
|
||||||
|
randseed:=hl*$10000+ regs.CX;
|
||||||
|
{$endif}
|
||||||
|
end;
|
||||||
|
|
||||||
|
{*****************************************************************************
|
||||||
|
System Dependent Exit code
|
||||||
|
*****************************************************************************}
|
||||||
|
|
||||||
|
procedure system_exit;
|
||||||
|
var
|
||||||
|
h : byte;
|
||||||
|
begin
|
||||||
|
{$ifdef todo}
|
||||||
|
RestoreInterruptHandlers;
|
||||||
|
{$endif}
|
||||||
|
for h:=0 to max_files-1 do
|
||||||
|
if openfiles[h] then
|
||||||
|
begin
|
||||||
|
{$ifdef SYSTEMDEBUG}
|
||||||
|
writeln(stderr,'file ',h,' "',opennames[h],'" not closed at exit');
|
||||||
|
{$endif SYSTEMDEBUG}
|
||||||
|
if h>=5 then
|
||||||
|
do_close(h);
|
||||||
|
end;
|
||||||
|
{$ifndef FPC_MM_TINY}
|
||||||
|
{$ifdef todo}
|
||||||
|
if not CheckNullArea then
|
||||||
|
writeln(stderr, 'Nil pointer assignment');
|
||||||
|
{$endif}
|
||||||
|
{$endif FPC_MM_TINY}
|
||||||
|
asm
|
||||||
|
ld a, exitcode
|
||||||
|
ld b, a
|
||||||
|
ld c, 0x62
|
||||||
|
call 0x0005
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
{*****************************************************************************
|
||||||
|
SystemUnit Initialization
|
||||||
|
*****************************************************************************}
|
||||||
|
|
||||||
|
procedure InitDosHeap;
|
||||||
|
begin
|
||||||
|
RegisterTinyHeapBlock_Simple_Prealigned(@__fpc_initialheap,__heapsize);
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure SysInitStdIO;
|
||||||
|
begin
|
||||||
|
OpenStdIO(Input,fmInput,StdInputHandle);
|
||||||
|
OpenStdIO(Output,fmOutput,StdOutputHandle);
|
||||||
|
OpenStdIO(ErrOutput,fmOutput,StdErrorHandle);
|
||||||
|
{$ifndef FPC_STDOUT_TRUE_ALIAS}
|
||||||
|
OpenStdIO(StdOut,fmOutput,StdOutputHandle);
|
||||||
|
OpenStdIO(StdErr,fmOutput,StdErrorHandle);
|
||||||
|
{$endif FPC_STDOUT_TRUE_ALIAS}
|
||||||
|
end;
|
||||||
|
|
||||||
|
function GetProcessID: SizeUInt;
|
||||||
|
begin
|
||||||
|
GetProcessID := PrefixSeg;
|
||||||
|
end;
|
||||||
|
|
||||||
|
function CheckInitialStkLen(stklen : SizeUInt) : SizeUInt;
|
||||||
|
begin
|
||||||
|
result := stklen;
|
||||||
|
end;
|
||||||
|
|
||||||
|
procedure InitDosVersion;
|
||||||
|
var
|
||||||
|
regs: Registers;
|
||||||
|
begin
|
||||||
|
regs.C := $6F;
|
||||||
|
regs.A := 0;
|
||||||
|
MsxDos(regs);
|
||||||
|
if regs.A <> 0 then
|
||||||
|
dos_version := 0
|
||||||
|
else if regs.B < 2 then
|
||||||
|
dos_version := $100
|
||||||
|
else
|
||||||
|
dos_version := regs.DE;
|
||||||
|
end;
|
||||||
|
|
||||||
|
begin
|
||||||
|
StackLength := stklen;
|
||||||
|
StackBottom := __stktop - stklen;
|
||||||
|
InitDosVersion;
|
||||||
|
{ for now we don't support MSX-DOS 1 }
|
||||||
|
if dos_version < $100 then
|
||||||
|
Halt($85);
|
||||||
|
{$ifdef todo}
|
||||||
|
InstallInterruptHandlers;
|
||||||
|
{$endif}
|
||||||
|
{ To be set if this is a GUI or console application }
|
||||||
|
IsConsole := TRUE;
|
||||||
|
{$ifdef FPC_HAS_FEATURE_DYNLIBS}
|
||||||
|
{ If dynlibs feature is disabled,
|
||||||
|
IsLibrary is a constant, which can thus not be set to a value }
|
||||||
|
{ To be set if this is a library and not a program }
|
||||||
|
IsLibrary := FALSE;
|
||||||
|
{$endif def FPC_HAS_FEATURE_DYNLIBS}
|
||||||
|
{ Setup heap }
|
||||||
|
InitDosHeap;
|
||||||
|
SysInitExceptions;
|
||||||
|
{$ifdef FPC_HAS_FEATURE_UNICODESTRINGS}
|
||||||
|
initunicodestringmanager;
|
||||||
|
{$endif def FPC_HAS_FEATURE_UNICODESTRINGS}
|
||||||
|
{ Setup stdin, stdout and stderr }
|
||||||
|
SysInitStdIO;
|
||||||
|
{ Setup environment and arguments }
|
||||||
|
{ Done on request only Setup_Environment; }
|
||||||
|
{ Done on request only Setup_Arguments; }
|
||||||
|
{ Reset IO Error }
|
||||||
|
InOutRes:=0;
|
||||||
|
{$ifdef FPC_HAS_FEATURE_THREADING}
|
||||||
|
InitSystemThreads;
|
||||||
|
{$endif}
|
||||||
|
end.
|
Loading…
Reference in New Issue
Block a user