123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386 |
- {
- This file is part of the Free Pascal run time library.
- Copyright (c) 2020,2021 by the Free Pascal development team.
- System unit for The WebAssembly System Interface (WASI).
- 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 system;
- interface
- {$define FPC_IS_SYSTEM}
- {$I systemh.inc}
- const
- LineEnding = #10;
- LFNSupport = true;
- DirectorySeparator = '/';
- DriveSeparator = '';
- ExtensionSeparator = '.';
- PathSeparator = ':';
- AllowDirectorySeparators : set of char = ['\','/'];
- AllowDriveSeparators : set of char = [];
- { FileNameCaseSensitive and FileNameCasePreserving are defined below! }
- maxExitCode = 65535;
- MaxPathLen = 4096;
- AllFilesMask = '*';
- const
- UnusedHandle = -1;
- StdInputHandle = 0;
- StdOutputHandle = 1;
- StdErrorHandle = 2;
- FileNameCaseSensitive : boolean = true;
- FileNameCasePreserving: boolean = true;
- CtrlZMarksEOF: boolean = false; (* #26 not considered as end of file *)
- sLineBreak = LineEnding;
- DefaultTextLineBreakStyle : TTextLineBreakStyle = tlbsLF;
- type
- PPreopenedDir = ^TPreopenedDir;
- TPreopenedDir = record
- dir_name: ansistring;
- fd: longint;
- end;
- PCurrentDir = ^TCurrentDir;
- TCurrentDir = record
- dir_name: ansistring;
- fd: longint;
- end;
- var
- argc: longint;
- argv: PPChar;
- envp: PPChar;
- preopened_dirs_count: longint;
- preopened_dirs: PPreopenedDir;
- drives_count: longint;
- current_dirs: PCurrentDir;
- current_drive: longint;
- function ConvertToFdRelativePath(path: ansistring; out fd: LongInt; out relfd_path: ansistring): Boolean;
- procedure DebugWrite(const P: PChar);
- procedure DebugWriteLn(const P: PChar);
- procedure DebugWriteChar(Ch: Char);
- procedure DebugWriteHexDigit(d: Byte);
- procedure DebugWriteHexByte(b: Byte);
- procedure DebugWriteHexWord(w: Word);
- procedure DebugWriteHexLongWord(lw: LongWord);
- implementation
- {$I wasitypes.inc}
- {$I wasiprocs.inc}
- {$I system.inc}
- var
- argv_size,
- argv_buf_size: __wasi_size_t;
- argv_buf: Pointer;
- environc,environ_buf_size,envp_size: __wasi_size_t;
- environ_buf: Pointer;
- function GetProcessID: SizeUInt;
- begin
- end;
- Procedure Randomize;
- Begin
- __wasi_random_get(@RandSeed,SizeOf(RandSeed));
- End;
- procedure System_exit;
- begin
- __wasi_proc_exit(ExitCode);
- End;
- function HasDriveLetter(const path: rawbytestring): Boolean;
- begin
- HasDriveLetter:=(path<>'') and (UpCase(path[1]) in ['A'..'Z']) and (path[2] = ':');
- end;
- function ConvertToFdRelativePath(path: ansistring; out fd: LongInt; out relfd_path: ansistring): Boolean;
- var
- drive_nr,I,pdir_drive,longest_match,pdir_length,chridx: longint;
- IsAbsolutePath: Boolean;
- pdir: ansistring;
- begin
- fd:=0;
- relfd_path:='';
- if HasDriveLetter(path) then
- begin
- drive_nr:=Ord(UpCase(path[1]))-(Ord('A')-1);
- delete(path,1,2);
- end
- else
- drive_nr:=current_drive;
- if path[1] in ['/','\'] then
- begin
- { path is absolute. Try to find it in the preopened dirs array }
- InOutRes:=3;
- ConvertToFdRelativePath:=false;
- longest_match:=0;
- for I:=0 to preopened_dirs_count-1 do
- begin
- pdir:=preopened_dirs[I].dir_name;
- if HasDriveLetter(pdir) then
- begin
- pdir_drive:=Ord(UpCase(pdir[1]))-(Ord('A')-1);
- delete(pdir,1,2);
- end
- else
- pdir_drive:=0;
- if pdir_drive<>drive_nr then
- continue;
- pdir_length:=Length(pdir);
- if pdir_length>Length(path) then
- continue;
- if Copy(path,1,pdir_length)<>Copy(pdir,1,pdir_length) then
- continue;
- chridx:=pdir_length+1;
- if (chridx>Length(path)) or not (path[chridx] in ['/','\']) then
- continue;
- if pdir_length>longest_match then
- begin
- longest_match:=pdir_length;
- while (chridx<=Length(path)) and (path[chridx] in ['/','\']) do
- Inc(chridx);
- fd:=preopened_dirs[I].fd;
- relfd_path:=Copy(path,chridx,Length(path)-chridx+1);
- InOutRes:=0;
- ConvertToFdRelativePath:=true;
- end;
- end;
- end
- else
- begin
- { path is relative to a current directory }
- if (drive_nr>=drives_count) or (current_dirs[drive_nr].dir_name='') then
- begin
- InOutRes:=15;
- ConvertToFdRelativePath:=false;
- exit;
- end;
- fd:=current_dirs[drive_nr].fd;
- relfd_path:=path;
- ConvertToFdRelativePath:=true;
- end;
- end;
- procedure Setup_PreopenedDirs;
- var
- fd: __wasi_fd_t;
- prestat: __wasi_prestat_t;
- res: __wasi_errno_t;
- prestat_dir_name: ansistring;
- drive_nr: longint;
- begin
- preopened_dirs_count:=0;
- preopened_dirs:=nil;
- drives_count:=0;
- current_dirs:=nil;
- current_drive:=0;
- fd:=3;
- repeat
- res:=__wasi_fd_prestat_get(fd, @prestat);
- if res=__WASI_ERRNO_SUCCESS then
- begin
- if (prestat.tag=__WASI_PREOPENTYPE_DIR) and (prestat.u.dir.pr_name_len>0) then
- begin
- SetLength(prestat_dir_name,prestat.u.dir.pr_name_len);
- if __wasi_fd_prestat_dir_name(fd,PByte(prestat_dir_name),prestat.u.dir.pr_name_len)=__WASI_ERRNO_SUCCESS then
- begin
- Inc(preopened_dirs_count);
- if preopened_dirs=nil then
- preopened_dirs:=AllocMem(preopened_dirs_count*SizeOf(TPreopenedDir))
- else
- ReAllocMem(preopened_dirs, preopened_dirs_count*SizeOf(TPreopenedDir));
- preopened_dirs[preopened_dirs_count-1].dir_name:=prestat_dir_name;
- preopened_dirs[preopened_dirs_count-1].fd:=fd;
- if HasDriveLetter(prestat_dir_name) then
- drive_nr:=Ord(UpCase(prestat_dir_name[1]))-(Ord('A')-1)
- else
- drive_nr:=0;
- if (drive_nr+1)>drives_count then
- begin
- drives_count:=drive_nr+1;
- if current_dirs=nil then
- current_dirs:=AllocMem(drives_count*SizeOf(TCurrentDir))
- else
- ReAllocMem(current_dirs,drives_count*SizeOf(TCurrentDir));
- end;
- if current_dirs[drive_nr].dir_name='' then
- begin
- current_dirs[drive_nr].dir_name:=prestat_dir_name;
- current_dirs[drive_nr].fd:=fd;
- end;
- end;
- end;
- end;
- Inc(fd);
- until res<>__WASI_ERRNO_SUCCESS;
- while (current_drive<drives_count) and (current_dirs[current_drive].dir_name='') do
- Inc(current_drive);
- end;
- procedure Setup_Environment;
- begin
- if envp<>nil then
- exit;
- if __wasi_environ_sizes_get(@environc, @environ_buf_size)<>__WASI_ERRNO_SUCCESS then
- begin
- envp:=nil;
- exit;
- end;
- envp_size:=(environc+1)*SizeOf(PChar);
- GetMem(envp, envp_size);
- GetMem(environ_buf, environ_buf_size);
- envp[environc]:=nil;
- if __wasi_environ_get(Pointer(envp), environ_buf)<>__WASI_ERRNO_SUCCESS then
- begin
- FreeMem(envp, envp_size);
- FreeMem(environ_buf, environ_buf_size);
- envp:=nil;
- end;
- end;
- procedure setup_arguments;
- begin
- if argv<>nil then
- exit;
- if __wasi_args_sizes_get(@argc, @argv_buf_size)<>__WASI_ERRNO_SUCCESS then
- begin
- argc:=0;
- argv:=nil;
- exit;
- end;
- argv_size:=(argc+1)*SizeOf(PChar);
- GetMem(argv, argv_size);
- GetMem(argv_buf, argv_buf_size);
- if __wasi_args_get(Pointer(argv), argv_buf)<>__WASI_ERRNO_SUCCESS then
- begin
- FreeMem(argv, argv_size);
- FreeMem(argv_buf, argv_buf_size);
- argc:=0;
- argv:=nil;
- end;
- 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 SysInitStdIO;
- begin
- OpenStdIO(Input,fmInput,StdInputHandle);
- OpenStdIO(Output,fmOutput,StdOutputHandle);
- OpenStdIO(ErrOutput,fmOutput,StdErrorHandle);
- OpenStdIO(StdOut,fmOutput,StdOutputHandle);
- OpenStdIO(StdErr,fmOutput,StdErrorHandle);
- end;
- function CheckInitialStkLen(stklen : SizeUInt) : SizeUInt;
- begin
- end;
- procedure DebugWrite(const P: PChar);
- var
- our_iov: __wasi_ciovec_t;
- our_nwritten: longint;
- begin
- our_iov.buf := PByte(P);
- our_iov.buf_len := StrLen(P);
- __wasi_fd_write(1, @our_iov, 1, @our_nwritten);
- end;
- procedure DebugWriteLn(const P: PChar);
- begin
- DebugWrite(P);
- DebugWriteChar(#10);
- end;
- procedure DebugWriteChar(Ch: Char);
- var
- CharArr: array [0..1] of Char;
- begin
- CharArr[0] := Ch;
- CharArr[1] := #0;
- DebugWrite(@CharArr);
- end;
- procedure DebugWriteHexDigit(d: Byte);
- const
- HexDigits: array [0..15] of Char = '0123456789ABCDEF';
- begin
- DebugWriteChar(HexDigits[d]);
- end;
- procedure DebugWriteHexByte(b: Byte);
- begin
- DebugWriteHexDigit(b shr 4);
- DebugWriteHexDigit(b and 15);
- end;
- procedure DebugWriteHexWord(w: Word);
- begin
- DebugWriteHexByte(w shr 8);
- DebugWriteHexByte(Byte(w));
- end;
- procedure DebugWriteHexLongWord(lw: LongWord);
- begin
- DebugWriteHexWord(lw shr 16);
- DebugWriteHexWord(Word(lw));
- end;
- begin
- { 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 }
- InitHeap;
- SysInitExceptions;
- initunicodestringmanager;
- { Setup stdin, stdout and stderr }
- SysInitStdIO;
- { Reset IO Error }
- InOutRes:=0;
- {$ifdef FPC_HAS_FEATURE_THREADING}
- InitSystemThreads;
- {$endif}
- Setup_Environment;
- Setup_PreopenedDirs;
- end.
|