123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796 |
- {
- $Id$
- This file is part of the Free Pascal run time library.
- Copyright (c) 1993-98 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.
- **********************************************************************}
- {$S-}
- unit syswin32;
- {$I os.inc}
- {$DEFINE WINHEAP}
- interface
- { include system-independent routine headers }
- {$I systemh.inc}
- {$ifndef WinHeap}
- { include heap support headers }
- {$I heaph.inc}
- {$endif}
- const
- { Default filehandles }
- UnusedHandle : longint = -1;
- StdInputHandle : longint = 0;
- StdOutputHandle : longint = 0;
- StdErrorHandle : longint = 0;
- type
- TStartupInfo=packed record
- cb : longint;
- lpReserved : Pointer;
- lpDesktop : Pointer;
- lpTitle : Pointer;
- dwX : longint;
- dwY : longint;
- dwXSize : longint;
- dwYSize : longint;
- dwXCountChars : longint;
- dwYCountChars : longint;
- dwFillAttribute : longint;
- dwFlags : longint;
- wShowWindow : Word;
- cbReserved2 : Word;
- lpReserved2 : Pointer;
- hStdInput : longint;
- hStdOutput : longint;
- hStdError : longint;
- end;
- var
- { C compatible arguments }
- argc : longint;
- argv : ppchar;
- { Win32 Info }
- startupinfo : tstartupinfo;
- hprevinst,
- hinstance,
- cmdshow : longint;
- {$ifdef WinHeap}
- var
- heaperror : pointer;
- function HeapSize:longint;
- {$endif}
- implementation
- { include system independent routines }
- {$I system.inc}
- { some declarations for Win32 API calls }
- {$I win32.inc}
- CONST
- { These constants are used for conversion of error codes }
- { from win32 i/o errors to tp i/o errors }
- { errors 1 to 18 are the same as in Turbo Pascal }
- { DO NOT MODIFY UNLESS YOU KNOW EXACTLY WHAT YOU ARE DOING! }
- { The media is write protected. }
- ERROR_WRITE_PROTECT = 19;
- { The system cannot find the device specified. }
- ERROR_BAD_UNIT = 20;
- { The device is not ready. }
- ERROR_NOT_READY = 21;
- { The device does not recognize the command. }
- ERROR_BAD_COMMAND = 22;
- { Data error (cyclic redundancy check) }
- ERROR_CRC = 23;
- { The program issued a command but the }
- { command length is incorrect. }
- ERROR_BAD_LENGTH = 24;
- { The drive cannot locate a specific }
- { area or track on the disk. }
- ERROR_SEEK = 25;
- { The specified disk or diskette cannot be accessed. }
- ERROR_NOT_DOS_DISK = 26;
- { The drive cannot find the sector requested. }
- ERROR_SECTOR_NOT_FOUND = 27;
- { The printer is out of paper. }
- ERROR_OUT_OF_PAPER = 28;
- { The system cannot write to the specified device. }
- ERROR_WRITE_FAULT = 29;
- { The system cannot read from the specified device. }
- ERROR_READ_FAULT = 30;
- { A device attached to the system is not functioning.}
- ERROR_GEN_FAILURE = 31;
- { The process cannot access the file because }
- { it is being used by another process. }
- ERROR_SHARING_VIOLATION = 32;
- var
- errno : longint;
- type
- plongint = ^longint;
- { misc. functions }
- function GetLastError : DWORD;
- external 'kernel32' name 'GetLastError';
- function MessageBox(w1:longint;l1,l2:pointer;w2:longint):longint;
- external 'user32' name 'MessageBoxA';
- { time and date functions }
- function GetTickCount : longint;
- external 'kernel32' name 'GetTickCount';
- { process functions }
- procedure ExitProcess(uExitCode : UINT);
- external 'kernel32' name 'ExitProcess';
- Procedure Errno2InOutRes;
- Begin
- { DO NOT MODIFY UNLESS YOU KNOW EXACTLY WHAT YOU ARE DOING }
- if (errno >= ERROR_WRITE_PROTECT) and (errno <= ERROR_GEN_FAILURE) THEN
- BEGIN
- { This is the offset to the Win32 to add to directly map }
- { to the DOS/TP compatible error codes when in this range }
- InOutRes := word(errno)+131;
- END
- else
- { This case is special }
- if errno=ERROR_SHARING_VIOLATION THEN
- BEGIN
- InOutRes :=5;
- END
- else
- { other error codes can directly be mapped }
- InOutRes := Word(errno);
- errno:=0;
- end;
- {$ifdef dummy}
- procedure int_stackcheck(stack_size:longint);[public,alias: 'STACKCHECK'];
- {
- called when trying to get local stack if the compiler directive $S
- is set this function must preserve esi !!!! because esi is set by
- the calling proc for methods it must preserve all registers !!
- With a 2048 byte safe area used to write to StdIo without crossing
- the stack boundary
- }
- begin
- asm
- pushl %eax
- pushl %ebx
- movl stack_size,%ebx
- addl $2048,%ebx
- movl %esp,%eax
- subl %ebx,%eax
- movl stacklimit,%ebx
- cmpl %eax,%ebx
- jae __short_on_stack
- popl %ebx
- popl %eax
- leave
- ret $4
- __short_on_stack:
- { can be usefull for error recovery !! }
- popl %ebx
- popl %eax
- end['EAX','EBX'];
- RunError(202);
- end;
- {$endif dummy}
- procedure halt(errnum : byte);
- begin
- do_exit;
- ExitProcess(errnum);
- end;
- function paramcount : longint;
- begin
- paramcount := argc - 1;
- end;
- function paramstr(l : longint) : string;
- begin
- if (l>=0) and (l+1<=argc) then
- paramstr:=strpas(argv[l])
- else
- paramstr:='';
- end;
- procedure randomize;
- begin
- randseed:=GetTickCount;
- end;
- {*****************************************************************************
- Heap Management
- *****************************************************************************}
- {$ifdef WinHeap}
- {$i winheap.inc}
- {$else}
- { memory functions }
- function GlobalAlloc(mode,size:longint):longint;
- external 'kernel32' name 'GlobalAlloc';
- function GlobalReAlloc(mode,size:longint):longint;
- external 'kernel32' name 'GlobalReAlloc';
- function GlobalHandle(p:pointer):longint;
- external 'kernel32' name 'GlobalHandle';
- function GlobalLock(handle:longint):pointer;
- external 'kernel32' name 'GlobalLock';
- function GlobalUnlock(h:longint):longint;
- external 'kernel32' name 'GlobalUnlock';
- function GlobalFree(h:longint):longint;
- external 'kernel32' name 'GlobalFree';
- function GlobalSize(h:longint):longint;
- external 'kernel32' name 'GlobalSize';
- procedure GlobalMemoryStatus(p:pointer);
- external 'kernel32' name 'GlobalMemoryStatus';
- function LocalAlloc(uFlags : UINT;uBytes :UINT) : HLOCAL;
- external 'kernel32' name 'LocalAlloc';
- function LocalFree(hMem:HLOCAL):HLOCAL;
- external 'kernel32' name 'LocalFree';
- function Sbrk(size : longint):longint;
- var
- h,l : longint;
- begin
- h:=GlobalAlloc(258,size);
- GlobalLock(h);
- l:=GlobalSize(h);
- writeln(l);
- sbrk:=l;
- end;
- { include standard heap management }
- {$I heap.inc}
- {$endif WinHeap}
- {*****************************************************************************
- Low Level File Routines
- *****************************************************************************}
- function WriteFile(fh:longint;buf:pointer;len:longint;var loaded:longint;
- overlap:pointer):longint;
- external 'kernel32' name 'WriteFile';
- function ReadFile(fh:longint;buf:pointer;len:longint;var loaded:longint;
- overlap:pointer):longint;
- external 'kernel32' name 'ReadFile';
- function CloseHandle(h : longint) : longint;
- external 'kernel32' name 'CloseHandle';
- function DeleteFile(p : pchar) : longint;
- external 'kernel32' name 'DeleteFileA';
- function MoveFile(old,_new : pchar) : longint;
- external 'kernel32' name 'MoveFileA';
- function SetFilePointer(l1,l2 : longint;l3 : pointer;l4 : longint) : longint;
- external 'kernel32' name 'SetFilePointer';
- function GetFileSize(h:longint;p:pointer) : longint;
- external 'kernel32' name 'GetFileSize';
- function CreateFile(name : pointer;access,sharing : longint;
- security : pointer;how,attr,template : longint) : longint;
- external 'kernel32' name 'CreateFileA';
- function SetEndOfFile(h : longint) : boolean;
- external 'kernel32' name 'SetEndOfFile';
- function GetFileType(Handle:DWORD):DWord;
- external 'kernel32' name 'GetFileType';
- procedure AllowSlash(p:pchar);
- var
- i : longint;
- begin
- { allow slash as backslash }
- for i:=0 to strlen(p) do
- if p[i]='/' then p[i]:='\';
- end;
- procedure do_close(h : longint);
- begin
- closehandle(h);
- end;
- procedure do_erase(p : pchar);
- begin
- AllowSlash(p);
- if DeleteFile(p)=0 then
- Begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- procedure do_rename(p1,p2 : pchar);
- begin
- AllowSlash(p1);
- AllowSlash(p2);
- if MoveFile(p1,p2)=0 then
- Begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- function do_write(h,addr,len : longint) : longint;
- var
- size:longint;
- begin
- if writefile(h,pointer(addr),len,size,nil)=0 then
- Begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- do_write:=size;
- end;
- function do_read(h,addr,len : longint) : longint;
- var
- result:longint;
- begin
- if readfile(h,pointer(addr),len,result,nil)=0 then
- Begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- do_read:=result;
- end;
- function do_filepos(handle : longint) : longint;
- var
- l:longint;
- begin
- l:=SetFilePointer(handle,0,nil,FILE_CURRENT);
- if l=-1 then
- begin
- l:=0;
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- do_filepos:=l;
- end;
- procedure do_seek(handle,pos : longint);
- begin
- if SetFilePointer(handle,pos,nil,FILE_BEGIN)=-1 then
- Begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- function do_seekend(handle:longint):longint;
- begin
- do_seekend:=SetFilePointer(handle,0,nil,FILE_END);
- if do_seekend=-1 then
- begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- function do_filesize(handle : longint) : longint;
- var
- aktfilepos : longint;
- begin
- aktfilepos:=do_filepos(handle);
- do_filesize:=do_seekend(handle);
- do_seek(handle,aktfilepos);
- end;
- procedure do_truncate (handle,pos:longint);
- begin
- do_seek(handle,pos);
- if not(SetEndOfFile(handle)) then
- begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- procedure do_open(var f;p : pchar;flags:longint);
- {
- 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 $10) the file will be append
- when (flags and $100) the file will be truncate/rewritten
- when (flags and $1000) there is no check for close (needed for textfiles)
- }
- var
- oflags,cd : longint;
- begin
- AllowSlash(p);
- { close first if opened }
- if ((flags and $1000)=0) then
- begin
- case filerec(f).mode of
- fminput,fmoutput,fminout : Do_Close(filerec(f).handle);
- fmclosed : ;
- else
- begin
- {not assigned}
- inoutres:=102;
- exit;
- end;
- end;
- end;
- { reset file handle }
- filerec(f).handle:=UnusedHandle;
- { convert filemode to filerec modes }
- case (flags and 3) of
- 0 : begin
- filerec(f).mode:=fminput;
- oflags:=GENERIC_READ;
- end;
- 1 : begin
- filerec(f).mode:=fmoutput;
- oflags:=GENERIC_WRITE;
- end;
- 2 : begin
- filerec(f).mode:=fminout;
- oflags:=GENERIC_WRITE or GENERIC_READ;
- end;
- end;
- { standard is opening and existing file }
- cd:=OPEN_EXISTING;
- { create it ? }
- if (flags and $100)<>0 then
- cd:=CREATE_ALWAYS
- { or append ? }
- else
- if (flags and $10)<>0 then
- cd:=OPEN_ALWAYS;
- { empty name is special }
- if p[0]=#0 then
- begin
- case filerec(f).mode of
- fminput : filerec(f).handle:=StdInputHandle;
- fmappend,
- fmoutput : begin
- filerec(f).handle:=StdOutputHandle;
- filerec(f).mode:=fmoutput; {fool fmappend}
- end;
- end;
- exit;
- end;
- filerec(f).handle:=CreateFile(p,oflags,0,nil,cd,FILE_ATTRIBUTE_NORMAL,0);
- { append mode }
- if (flags and $10)<>0 then
- begin
- do_seekend(filerec(f).handle);
- filerec(f).mode:=fmoutput; {fool fmappend}
- end;
- { get errors }
- if filerec(f).handle=0 then
- begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- function do_isdevice(handle:longint):boolean;
- begin
- do_isdevice:=(getfiletype(handle)=2);
- end;
- {*****************************************************************************
- UnTyped File Handling
- *****************************************************************************}
- {$i file.inc}
- {*****************************************************************************
- Typed File Handling
- *****************************************************************************}
- {$i typefile.inc}
- {*****************************************************************************
- Text File Handling
- *****************************************************************************}
- {$DEFINE EOF_CTRLZ}
- {$i text.inc}
- {*****************************************************************************
- Directory Handling
- *****************************************************************************}
- function CreateDirectory(name : pointer;sec : pointer) : longint;
- external 'kernel32' name 'CreateDirectoryA';
- function RemoveDirectory(name:pointer):longint;
- external 'kernel32' name 'RemoveDirectoryA';
- function SetCurrentDirectory(name : pointer) : longint;
- external 'kernel32' name 'SetCurrentDirectoryA';
- function GetCurrentDirectory(bufsize : longint;name : pchar) : longint;
- external 'kernel32' name 'GetCurrentDirectoryA';
- type
- TDirFnType=function(name:pointer):word;
- procedure dirfn(afunc : TDirFnType;const s:string);
- var
- buffer : array[0..255] of char;
- begin
- move(s[1],buffer,length(s));
- buffer[length(s)]:=#0;
- AllowSlash(pchar(@buffer));
- if aFunc(@buffer)=0 then
- begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- end;
- function CreateDirectoryTrunc(name:pointer):word;
- begin
- CreateDirectoryTrunc:=CreateDirectory(name,nil);
- end;
- procedure mkdir(const s:string);[IOCHECK];
- begin
- If InOutRes <> 0 then exit;
- dirfn(TDirFnType(@CreateDirectoryTrunc),s);
- end;
- procedure rmdir(const s:string);[IOCHECK];
- begin
- If InOutRes <> 0 then exit;
- dirfn(TDirFnType(@RemoveDirectory),s);
- end;
- procedure chdir(const s:string);[IOCHECK];
- begin
- If InOutRes <> 0 then exit;
- dirfn(TDirFnType(@SetCurrentDirectory),s);
- end;
- procedure getdir(drivenr:byte;var dir:string);
- const
- Drive:array[0..3]of char=(#0,':',#0,#0);
- var
- defaultdrive:boolean;
- DirBuf,SaveBuf:array[0..259] of Char;
- begin
- defaultdrive:=drivenr=0;
- if not defaultdrive then
- begin
- byte(Drive[0]):=Drivenr+64;
- GetCurrentDirectory(SizeOf(SaveBuf),SaveBuf);
- SetCurrentDirectory(@Drive);
- end;
- GetCurrentDirectory(SizeOf(DirBuf),DirBuf);
- if not defaultdrive then
- SetCurrentDirectory(@SaveBuf);
- dir:=strpas(DirBuf);
- end;
- {*****************************************************************************
- SystemUnit Initialization
- *****************************************************************************}
- { Startup }
- procedure GetStartupInfo(p : pointer);
- external 'kernel32' name 'GetStartupInfoA';
- function GetStdHandle(nStdHandle:DWORD):THANDLE;
- external 'kernel32' name 'GetStdHandle';
- { command line/enviroment functions }
- function GetCommandLine : pchar;
- external 'kernel32' name 'GetCommandLineA';
- { module functions }
- function GetModuleFileName(l1:longint;p:pointer;l2:longint):longint;
- external 'kernel32' name 'GetModuleFileNameA';
- function GetModuleHandle(p : pointer) : longint;
- external 'kernel32' name 'GetModuleHandleA';
- var
- ModuleName : array[0..255] of char;
- function GetCommandFile:pchar;
- begin
- GetModuleFileName(0,@ModuleName,255);
- GetCommandFile:=@ModuleName;
- end;
- procedure setup_arguments;
- var
- arglen,
- count : longint;
- argstart,
- cmdline : pchar;
- quote : set of char;
- argsbuf : array[0..127] of pchar;
- begin
- { create commandline, it starts with the executed filename which is argv[0] }
- cmdline:=GetCommandLine;
- count:=0;
- repeat
- { skip leading spaces }
- while cmdline^ in [' ',#9,#13] do
- inc(longint(cmdline));
- case cmdline^ of
- #0 : break;
- '"' : begin
- quote:=['"'];
- inc(longint(cmdline));
- end;
- '''' : begin
- quote:=[''''];
- inc(longint(cmdline));
- end;
- else
- quote:=[' ',#9,#13];
- end;
- { scan until the end of the argument }
- argstart:=cmdline;
- while (cmdline^<>#0) and not(cmdline^ in quote) do
- inc(longint(cmdline));
- { reserve some memory }
- arglen:=cmdline-argstart;
- getmem(argsbuf[count],arglen+1);
- move(argstart^,argsbuf[count]^,arglen);
- argsbuf[count][arglen]:=#0;
- { skip quote }
- if cmdline^ in quote then
- inc(longint(cmdline));
- inc(count);
- until false;
- { create argc }
- argc:=count;
- { create an nil entry }
- argsbuf[count]:=nil;
- inc(count);
- { create the argv }
- getmem(argv,count shl 2);
- move(argsbuf,argv^,count shl 2);
- end;
- {$ASMMODE DIRECT}
- procedure Entry;[public,alias: '_mainCRTStartup'];
- begin
- { call to the pascal main }
- asm
- call PASCALMAIN
- end;
- { that's all folks }
- ExitProcess(0);
- end;
- {$ifdef dummy}
- Function SetUpStack : longint;
- { This routine does the following : }
- { returns the value of the initial SP - __stklen }
- begin
- asm
- pushl %ebx
- pushl %eax
- movl __stklen,%ebx
- movl %esp,%eax
- subl %ebx,%eax
- movl %eax,__RESULT
- popl %eax
- popl %ebx
- end;
- end;
- {$endif}
- {$ASMMODE ATT}
- begin
- { get some helpful informations }
- GetStartupInfo(@startupinfo);
- { some misc Win32 stuff }
- hprevinst:=0;
- hinstance:=getmodulehandle(GetCommandFile);
- cmdshow:=startupinfo.wshowwindow;
- { to test stack depth }
- loweststack:=maxlongint;
- { real test stack depth }
- { stacklimit := setupstack; }
- { Setup heap }
- {$ifndef WinHeap}
- InitHeap;
- {$endif WinHeap}
- { Setup stdin, stdout and stderr }
- StdInputHandle:=longint(GetStdHandle(STD_INPUT_HANDLE));
- StdOutputHandle:=longint(GetStdHandle(STD_OUTPUT_HANDLE));
- StdErrorHandle:=longint(GetStdHandle(STD_ERROR_HANDLE));
- OpenStdIO(Input,fmInput,StdInputHandle);
- OpenStdIO(Output,fmOutput,StdOutputHandle);
- OpenStdIO(StdErr,fmOutput,StdErrorHandle);
- { Arguments }
- setup_arguments;
- { Reset IO Error }
- InOutRes:=0;
- { Reset internal error variable }
- errno := 0;
- end.
- {
- $Log$
- Revision 1.12 1998-07-07 12:37:28 carl
- * correct mapping of error codes for TP compatibility
- + implemented stack checking in ifdef dummy
- Revision 1.11 1998/07/02 12:33:18 carl
- * IOCheck/InOutRes check for mkdir,rmdir and chdir like in TP
- Revision 1.10 1998/07/01 15:30:02 peter
- * better readln/writeln
- Revision 1.9 1998/06/10 10:39:17 peter
- * working w32 rtl
- Revision 1.8 1998/06/08 23:07:47 peter
- * dos interface is now 100% compatible
- * fixed call PASCALMAIN which must be direct asm
- Revision 1.7 1998/05/06 12:36:51 michael
- + Removed log from before restored version.
- Revision 1.6 1998/04/27 18:29:09 florian
- + do_open implemented, the file-I/O should be now complete
- Revision 1.5 1998/04/27 13:58:21 florian
- + paramstr/paramcount implemented
- Revision 1.4 1998/04/26 22:37:22 florian
- * some small extensions
- Revision 1.3 1998/04/26 21:49:57 florian
- + more stuff added (??dir procedures etc.)
- }
|