| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331 |
- {
- $Id$
- 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.
- **********************************************************************}
- unit syswin32;
- interface
- {$ifdef SYSTEMDEBUG}
- {$define SYSTEMEXCEPTIONDEBUG}
- {$endif SYSTEMDEBUG}
- {$ifdef i386}
- {$define Set_i386_Exception_handler}
- {$endif i386}
- { include system-independent routine headers }
- {$I systemh.inc}
- { include heap support headers }
- {$I heaph.inc}
- const
- { Default filehandles }
- UnusedHandle : longint = -1;
- StdInputHandle : longint = 0;
- StdOutputHandle : longint = 0;
- StdErrorHandle : longint = 0;
- FileNameCaseSensitive : boolean = true;
- 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,
- MainInstance,
- cmdshow : longint;
- DLLreason,DLLparam:longint;
- Win32StackTop : Dword;
- { Thread count for DLL }
- const
- Thread_count : longint = 0;
- type
- TDLL_Process_Entry_Hook = function (dllparam : longint) : longbool;
- TDLL_Entry_Hook = procedure (dllparam : longint);
- const
- Dll_Process_Attach_Hook : TDLL_Process_Entry_Hook = nil;
- Dll_Process_Detach_Hook : TDLL_Entry_Hook = nil;
- Dll_Thread_Attach_Hook : TDLL_Entry_Hook = nil;
- Dll_Thread_Detach_Hook : TDLL_Entry_Hook = nil;
- 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;
- {$ASMMODE ATT}
- { misc. functions }
- function GetLastError : DWORD;
- external 'kernel32' name 'GetLastError';
- { 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 .L__short_on_stack
- popl %ebx
- popl %eax
- leave
- ret $4
- .L__short_on_stack:
- { can be usefull for error recovery !! }
- popl %ebx
- popl %eax
- end['EAX','EBX'];
- HandleError(202);
- end;
- {$endif dummy}
- function paramcount : longint;
- begin
- paramcount := argc - 1;
- end;
- { module functions }
- function GetModuleFileName(l1:longint;p:pointer;l2:longint):longint;
- external 'kernel32' name 'GetModuleFileNameA';
- function GetModuleHandle(p : pointer) : longint;
- external 'kernel32' name 'GetModuleHandleA';
- function GetCommandFile:pchar;forward;
- function paramstr(l : longint) : string;
- begin
- if (l>=0) and (l<argc) then
- paramstr:=strpas(argv[l])
- else
- paramstr:='';
- end;
- procedure randomize;
- begin
- randseed:=GetTickCount;
- end;
- {*****************************************************************************
- Heap Management
- *****************************************************************************}
- { memory functions }
- function GetProcessHeap : DWord;
- external 'kernel32' name 'GetProcessHeap';
- function HeapAlloc(hHeap : DWord; dwFlags : DWord; dwBytes : DWord) : Longint;
- external 'kernel32' name 'HeapAlloc';
- {$IFDEF SYSTEMDEBUG}
- function HeapSize(hHeap : DWord; dwFlags : DWord; ptr : Pointer) : DWord;
- external 'kernel32' name 'HeapSize';
- {$ENDIF}
- var
- heap : longint;external name 'HEAP';
- intern_heapsize : longint;external name 'HEAPSIZE';
- function getheapstart:pointer;assembler;
- asm
- leal HEAP,%eax
- end ['EAX'];
- function getheapsize:longint;assembler;
- asm
- movl intern_HEAPSIZE,%eax
- end ['EAX'];
- function Sbrk(size : longint):longint;
- var
- l : longint;
- begin
- l := HeapAlloc(GetProcessHeap(), 0, size);
- if (l = 0) then
- l := -1;
- {$ifdef DUMPGROW}
- Writeln('new heap part at $',hexstr(l,8), ' size = ',HeapSize(GetProcessHeap()));
- {$endif}
- sbrk:=l;
- end;
- { include standard heap management }
- {$I heap.inc}
- {*****************************************************************************
- 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) : longbool;
- 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;
- function do_isdevice(handle:longint):boolean;
- begin
- do_isdevice:=(getfiletype(handle)=2);
- end;
- procedure do_close(h : longint);
- begin
- if do_isdevice(h) then
- exit;
- 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 $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)
- }
- Const
- file_Share_Read = $00000001;
- file_Share_Write = $00000002;
- Var
- shflags,
- oflags,cd : longint;
- begin
- AllowSlash(p);
- { 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
- {not assigned}
- inoutres:=102;
- exit;
- end;
- end;
- end;
- { reset file handle }
- filerec(f).handle:=UnusedHandle;
- { convert filesharing }
- shflags:=0;
- if ((filemode and fmshareExclusive) = fmshareExclusive) then
- { no sharing }
- else
- if (filemode = fmShareCompat) or ((filemode and fmshareDenyWrite) = fmshareDenyWrite) then
- shflags := file_Share_Read
- else
- if ((filemode and fmshareDenyRead) = fmshareDenyRead) then
- shflags := file_Share_Write
- else
- if ((filemode and fmshareDenyNone) = fmshareDenyNone) then
- shflags := file_Share_Read + file_Share_Write;
- { 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 $1000)<>0 then
- cd:=CREATE_ALWAYS
- { or append ? }
- else
- if (flags and $100)<>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;
- 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;
- filerec(f).handle:=CreateFile(p,oflags,shflags,nil,cd,FILE_ATTRIBUTE_NORMAL,0);
- { append mode }
- if (flags and $100)<>0 then
- begin
- do_seekend(filerec(f).handle);
- filerec(f).mode:=fmoutput; {fool fmappend}
- end;
- { get errors }
- { handle -1 is returned sometimes !! (PM) }
- if (filerec(f).handle=0) or (filerec(f).handle=-1) then
- begin
- errno:=GetLastError;
- Errno2InoutRes;
- end;
- 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:shortstring);
- 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);
- if not FileNameCaseSensitive then
- dir:=upcase(dir);
- 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';
- 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,
- pc : pchar;
- quote : set of char;
- argsbuf : array[0..127] of pchar;
- begin
- { create commandline, it starts with the executed filename which is argv[0] }
- { Win32 passes the command NOT via the args, but via getmodulefilename}
- count:=0;
- pc:=getcommandfile;
- Arglen:=0;
- repeat
- Inc(Arglen);
- until (pc[Arglen]=#0);
- getmem(argsbuf[count],arglen+1);
- move(pc^,argsbuf[count]^,arglen);
- { Now skip the first one }
- pc:=GetCommandLine;
- repeat
- { skip leading spaces }
- while pc^ in [' ',#9,#13] do
- inc(pc);
- case pc^ of
- #0 : break;
- '"' : begin
- quote:=['"'];
- inc(pc);
- end;
- '''' : begin
- quote:=[''''];
- inc(pc);
- end;
- else
- quote:=[' ',#9,#13];
- end;
- { scan until the end of the argument }
- argstart:=pc;
- while (pc^<>#0) and not(pc^ in quote) do
- inc(pc);
- { Don't copy the first one, it is already there.}
- If Count<>0 then
- begin
- { reserve some memory }
- arglen:=pc-argstart;
- getmem(argsbuf[count],arglen+1);
- move(argstart^,argsbuf[count]^,arglen);
- argsbuf[count][arglen]:=#0;
- end;
- { skip quote }
- if pc^ in quote then
- inc(pc);
- 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);
- { Setup cmdline variable }
- cmdline:=GetCommandLine;
- end;
- {*****************************************************************************
- System Dependent Exit code
- *****************************************************************************}
- procedure install_exception_handlers;forward;
- procedure remove_exception_handlers;forward;
- procedure PascalMain;external name 'PASCALMAIN';
- procedure fpc_do_exit;external name 'FPC_DO_EXIT';
- Procedure ExitDLL(Exitcode : longint); forward;
- Procedure system_exit;
- begin
- { don't call ExitProcess inside
- the DLL exit code !!
- This crashes Win95 at least PM }
- if IsLibrary then
- ExitDLL(ExitCode);
- if not IsConsole then
- begin
- Close(stderr);
- Close(stdout);
- { what about Input and Output ?? PM }
- end;
- remove_exception_handlers;
- ExitProcess(ExitCode);
- 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}
- var
- { value of the stack segment
- to check if the call stack can be written on exceptions }
- _SS : longint;
- const
- fpucw : word = $1332;
- procedure Exe_entry;[public, alias : '_FPC_EXE_Entry'];
- begin
- IsLibrary:=false;
- { install the handlers for exe only ?
- or should we install them for DLL also ? (PM) }
- install_exception_handlers;
- { This strange construction is needed to solve the _SS problem
- with a smartlinked syswin32 (PFV) }
- asm
- pushl %ebp
- xorl %ebp,%ebp
- movl %esp,%eax
- movl %eax,Win32StackTop
- movw %ss,%bp
- movl %ebp,_SS
- fninit
- fldcw fpucw
- xorl %ebp,%ebp
- call PASCALMAIN
- popl %ebp
- end;
- { if we pass here there was no error ! }
- system_exit;
- end;
- Const
- { DllEntryPoint }
- DLL_PROCESS_ATTACH = 1;
- DLL_THREAD_ATTACH = 2;
- DLL_PROCESS_DETACH = 0;
- DLL_THREAD_DETACH = 3;
- Var
- DLLBuf : Jmp_buf;
- Const
- DLLExitOK : boolean = true;
- function Dll_entry : longbool;[public, alias : '_FPC_DLL_Entry'];
- var
- res : longbool;
- begin
- IsLibrary:=true;
- Dll_entry:=false;
- case DLLreason of
- DLL_PROCESS_ATTACH :
- begin
- If SetJmp(DLLBuf) = 0 then
- begin
- if assigned(Dll_Process_Attach_Hook) then
- begin
- res:=Dll_Process_Attach_Hook(DllParam);
- if not res then
- exit(false);
- end;
- PASCALMAIN;
- Dll_entry:=true;
- end
- else
- Dll_entry:=DLLExitOK;
- end;
- DLL_THREAD_ATTACH :
- begin
- inc(Thread_count);
- if assigned(Dll_Thread_Attach_Hook) then
- Dll_Thread_Attach_Hook(DllParam);
- Dll_entry:=true; { return value is ignored }
- end;
- DLL_THREAD_DETACH :
- begin
- dec(Thread_count);
- if assigned(Dll_Thread_Detach_Hook) then
- Dll_Thread_Detach_Hook(DllParam);
- Dll_entry:=true; { return value is ignored }
- end;
- DLL_PROCESS_DETACH :
- begin
- Dll_entry:=true; { return value is ignored }
- If SetJmp(DLLBuf) = 0 then
- begin
- FPC_DO_EXIT;
- end;
- if assigned(Dll_Process_Detach_Hook) then
- Dll_Process_Detach_Hook(DllParam);
- end;
- end;
- end;
- Procedure ExitDLL(Exitcode : longint);
- begin
- DLLExitOK:=ExitCode=0;
- LongJmp(DLLBuf,1);
- end;
- {$ifdef Set_i386_Exception_handler}
- const
- EXCEPTION_MAXIMUM_PARAMETERS = 15;
- EXCEPTION_ACCESS_VIOLATION = $c0000005;
- EXCEPTION_BREAKPOINT = $80000003;
- EXCEPTION_DATATYPE_MISALIGNMENT = $80000002;
- EXCEPTION_SINGLE_STEP = $80000004;
- EXCEPTION_ARRAY_BOUNDS_EXCEEDED = $c000008c;
- EXCEPTION_FLT_DENORMAL_OPERAND = $c000008d;
- EXCEPTION_FLT_DIVIDE_BY_ZERO = $c000008e;
- EXCEPTION_FLT_INEXACT_RESULT = $c000008f;
- EXCEPTION_FLT_INVALID_OPERATION = $c0000090;
- EXCEPTION_FLT_OVERFLOW = $c0000091;
- EXCEPTION_FLT_STACK_CHECK = $c0000092;
- EXCEPTION_FLT_UNDERFLOW = $c0000093;
- EXCEPTION_INT_DIVIDE_BY_ZERO = $c0000094;
- EXCEPTION_INT_OVERFLOW = $c0000095;
- EXCEPTION_INVALID_HANDLE = $c0000008;
- EXCEPTION_PRIV_INSTRUCTION = $c0000096;
- EXCEPTION_NONCONTINUABLE_EXCEPTION = $c0000025;
- EXCEPTION_NONCONTINUABLE = $1;
- EXCEPTION_STACK_OVERFLOW = $c00000fd;
- EXCEPTION_INVALID_DISPOSITION = $c0000026;
- EXCEPTION_ILLEGAL_INSTRUCTION = $C000001D;
- EXCEPTION_IN_PAGE_ERROR = $C0000006;
- EXCEPTION_EXECUTE_HANDLER = 1;
- EXCEPTION_CONTINUE_EXECUTION = -(1);
- EXCEPTION_CONTINUE_SEARCH = 0;
- type
- FLOATING_SAVE_AREA = record
- ControlWord : DWORD;
- StatusWord : DWORD;
- TagWord : DWORD;
- ErrorOffset : DWORD;
- ErrorSelector : DWORD;
- DataOffset : DWORD;
- DataSelector : DWORD;
- RegisterArea : array[0..79] of BYTE;
- Cr0NpxState : DWORD;
- end;
- _FLOATING_SAVE_AREA = FLOATING_SAVE_AREA;
- TFLOATINGSAVEAREA = FLOATING_SAVE_AREA;
- PFLOATINGSAVEAREA = ^FLOATING_SAVE_AREA;
- CONTEXT = record
- ContextFlags : DWORD;
- Dr0 : DWORD;
- Dr1 : DWORD;
- Dr2 : DWORD;
- Dr3 : DWORD;
- Dr6 : DWORD;
- Dr7 : DWORD;
- FloatSave : FLOATING_SAVE_AREA;
- SegGs : DWORD;
- SegFs : DWORD;
- SegEs : DWORD;
- SegDs : DWORD;
- Edi : DWORD;
- Esi : DWORD;
- Ebx : DWORD;
- Edx : DWORD;
- Ecx : DWORD;
- Eax : DWORD;
- Ebp : DWORD;
- Eip : DWORD;
- SegCs : DWORD;
- EFlags : DWORD;
- Esp : DWORD;
- SegSs : DWORD;
- end;
- LPCONTEXT = ^CONTEXT;
- _CONTEXT = CONTEXT;
- TCONTEXT = CONTEXT;
- PCONTEXT = ^CONTEXT;
- type pexception_record = ^exception_record;
- EXCEPTION_RECORD = record
- ExceptionCode : longint;
- ExceptionFlags : longint;
- ExceptionRecord : pexception_record;
- ExceptionAddress : pointer;
- NumberParameters : longint;
- ExceptionInformation : array[0..EXCEPTION_MAXIMUM_PARAMETERS-1] of pointer;
- end;
- PEXCEPTION_POINTERS = ^EXCEPTION_POINTERS;
- EXCEPTION_POINTERS = record
- ExceptionRecord : PEXCEPTION_RECORD ;
- ContextRecord : PCONTEXT ;
- end;
- { type of functions that should be used for exception handling }
- LPTOP_LEVEL_EXCEPTION_FILTER = function(excep :PEXCEPTION_POINTERS) : longint;
- function SetUnhandledExceptionFilter(lpTopLevelExceptionFilter : LPTOP_LEVEL_EXCEPTION_FILTER)
- : LPTOP_LEVEL_EXCEPTION_FILTER;
- external 'kernel32' name 'SetUnhandledExceptionFilter';
- const
- MAX_Level = 16;
- except_level : byte = 0;
- var
- except_eip : array[0..Max_level-1] of longint;
- except_error : array[0..Max_level-1] of byte;
- reset_fpu : array[0..max_level-1] of boolean;
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- procedure DebugHandleErrorAddrFrame(error, addr, frame : longint);
- begin
- if IsConsole then
- begin
- write(stderr,'call to HandleErrorAddrFrame(error=',error);
- write(stderr,',addr=',hexstr(addr,8));
- writeln(stderr,',frame=',hexstr(frame,8),')');
- end;
- HandleErrorAddrFrame(error,addr,frame);
- end;
- {$endif SYSTEMEXCEPTIONDEBUG}
- procedure JumpToHandleErrorFrame;
- var
- eip,ebp,error : longint;
- begin
- asm
- movl (%ebp),%eax
- movl %eax,ebp
- end;
- if except_level>0 then
- dec(except_level);
- eip:=except_eip[except_level];
- error:=except_error[except_level];
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- if IsConsole then
- begin
- writeln(stderr,'In JumpToHandleErrorFrame error=',error);
- end;
- {$endif SYSTEMEXCEPTIONDEBUG}
- if reset_fpu[except_level] then
- asm
- fninit
- fldcw fpucw
- end;
- { build a fake stack }
- asm
- movl ebp,%eax
- pushl %eax
- movl eip,%eax
- pushl %eax
- movl error,%eax
- pushl %eax
- movl eip,%eax
- pushl %eax
- movl ebp,%ebp // Change frame pointer
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- jmpl DebugHandleErrorAddrFrame
- {$else not SYSTEMEXCEPTIONDEBUG}
- jmpl HandleErrorAddrFrame
- {$endif SYSTEMEXCEPTIONDEBUG}
- end;
- end;
- function syswin32_i386_exception_handler(excep :PEXCEPTION_POINTERS) : longint;
- var frame,res : longint;
- function SysHandleErrorFrame(error,frame : longint;must_reset_fpu : boolean) : longint;
- begin
- if frame=0 then
- SysHandleErrorFrame:=Exception_Continue_Search
- else
- begin
- if except_level >= Max_level then
- exit;
- except_eip[except_level]:=excep^.ContextRecord^.Eip;
- except_error[except_level]:=error;
- reset_fpu[except_level]:=must_reset_fpu;
- inc(except_level);
- excep^.ContextRecord^.Eip:=longint(@JumpToHandleErrorFrame);
- excep^.ExceptionRecord^.ExceptionCode:=0;
- SysHandleErrorFrame:=Exception_Continue_Execution;
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- if IsConsole then
- begin
- writeln(stderr,'Exception Continue Exception set at ',
- hexstr(except_eip[except_level],8));
- writeln(stderr,'Eip changed to ',
- hexstr(longint(@JumpToHandleErrorFrame),8), ' error=',error);
- end;
- {$endif SYSTEMEXCEPTIONDEBUG}
- end;
- end;
- begin
- if excep^.ContextRecord^.SegSs=_SS then
- frame:=excep^.ContextRecord^.Ebp
- else
- frame:=0;
- { default : unhandled !}
- res:=Exception_Continue_Search;
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- if IsConsole then
- writeln(stderr,'Exception ',
- hexstr(excep^.ExceptionRecord^.ExceptionCode,8));
- {$endif SYSTEMEXCEPTIONDEBUG}
- case excep^.ExceptionRecord^.ExceptionCode of
- EXCEPTION_ACCESS_VIOLATION :
- res:=SysHandleErrorFrame(216,frame,false);
- { EXCEPTION_BREAKPOINT = $80000003;
- EXCEPTION_DATATYPE_MISALIGNMENT = $80000002;
- EXCEPTION_SINGLE_STEP = $80000004; }
- EXCEPTION_ARRAY_BOUNDS_EXCEEDED :
- res:=SysHandleErrorFrame(201,frame,false);
- EXCEPTION_FLT_DENORMAL_OPERAND :
- begin
- res:=SysHandleErrorFrame(216,frame,true);
- end;
- EXCEPTION_FLT_DIVIDE_BY_ZERO :
- begin
- res:=SysHandleErrorFrame(200,frame,true);
- {excep^.ContextRecord^.FloatSave.StatusWord:=excep^.ContextRecord^.FloatSave.StatusWord and $ffffff00;}
- end;
- {EXCEPTION_FLT_INEXACT_RESULT = $c000008f; }
- EXCEPTION_FLT_INVALID_OPERATION :
- begin
- res:=SysHandleErrorFrame(207,frame,true);
- end;
- EXCEPTION_FLT_OVERFLOW :
- begin
- res:=SysHandleErrorFrame(205,frame,true);
- end;
- EXCEPTION_FLT_STACK_CHECK :
- begin
- res:=SysHandleErrorFrame(207,frame,true);
- end;
- EXCEPTION_FLT_UNDERFLOW :
- begin
- res:=SysHandleErrorFrame(206,frame,true); { should be accepted as zero !! }
- end;
- EXCEPTION_INT_DIVIDE_BY_ZERO :
- res:=SysHandleErrorFrame(200,frame,false);
- EXCEPTION_INT_OVERFLOW :
- res:=SysHandleErrorFrame(215,frame,false);
- {EXCEPTION_INVALID_HANDLE = $c0000008;
- EXCEPTION_PRIV_INSTRUCTION = $c0000096;
- EXCEPTION_NONCONTINUABLE_EXCEPTION = $c0000025;
- EXCEPTION_NONCONTINUABLE = $1;}
- EXCEPTION_STACK_OVERFLOW :
- res:=SysHandleErrorFrame(202,frame,false);
- {EXCEPTION_INVALID_DISPOSITION = $c0000026;}
- EXCEPTION_ILLEGAL_INSTRUCTION,
- EXCEPTION_PRIV_INSTRUCTION,
- EXCEPTION_IN_PAGE_ERROR,
- EXCEPTION_SINGLE_STEP : res:=SysHandleErrorFrame(217,frame,false);
- end;
- syswin32_i386_exception_handler:=res;
- end;
- procedure install_exception_handlers;
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- var
- oldexceptaddr,newexceptaddr : longint;
- {$endif SYSTEMEXCEPTIONDEBUG}
- begin
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- asm
- movl $0,%eax
- movl %fs:(%eax),%eax
- movl %eax,oldexceptaddr
- end;
- {$endif SYSTEMEXCEPTIONDEBUG}
- SetUnhandledExceptionFilter(@syswin32_i386_exception_handler);
- {$ifdef SYSTEMEXCEPTIONDEBUG}
- asm
- movl $0,%eax
- movl %fs:(%eax),%eax
- movl %eax,newexceptaddr
- end;
- if IsConsole then
- writeln(stderr,'Old exception ',hexstr(oldexceptaddr,8),
- ' new exception ',hexstr(newexceptaddr,8));
- {$endif SYSTEMEXCEPTIONDEBUG}
- end;
- procedure remove_exception_handlers;
- begin
- SetUnhandledExceptionFilter(nil);
- end;
- {$else not i386 (Processor specific !!)}
- procedure install_exception_handlers;
- begin
- end;
- procedure remove_exception_handlers;
- begin
- end;
- {$endif Set_i386_Exception_handler}
- {****************************************************************************
- Error Message writing using messageboxes
- ****************************************************************************}
- function MessageBox(w1:longint;l1,l2:pointer;w2:longint):longint;
- external 'user32' name 'MessageBoxA';
- const
- ErrorBufferLength = 1024;
- var
- ErrorBuf : array[0..ErrorBufferLength] of char;
- ErrorLen : longint;
- Function ErrorWrite(Var F: TextRec): Integer;
- {
- An error message should always end with #13#10#13#10
- }
- var
- p : pchar;
- i : longint;
- Begin
- if F.BufPos>0 then
- begin
- if F.BufPos+ErrorLen>ErrorBufferLength then
- i:=ErrorBufferLength-ErrorLen
- else
- i:=F.BufPos;
- Move(F.BufPtr^,ErrorBuf[ErrorLen],i);
- inc(ErrorLen,i);
- ErrorBuf[ErrorLen]:=#0;
- end;
- if ErrorLen>3 then
- begin
- p:=@ErrorBuf[ErrorLen];
- for i:=1 to 4 do
- begin
- dec(p);
- if not(p^ in [#10,#13]) then
- break;
- end;
- end;
- if ErrorLen=ErrorBufferLength then
- i:=4;
- if (i=4) then
- begin
- MessageBox(0,@ErrorBuf,pchar('Error'),0);
- ErrorLen:=0;
- end;
- F.BufPos:=0;
- ErrorWrite:=0;
- End;
- Function ErrorClose(Var F: TextRec): Integer;
- begin
- if ErrorLen>0 then
- begin
- MessageBox(0,@ErrorBuf,pchar('Error'),0);
- ErrorLen:=0;
- end;
- ErrorLen:=0;
- ErrorClose:=0;
- end;
- Function ErrorOpen(Var F: TextRec): Integer;
- Begin
- TextRec(F).InOutFunc:=@ErrorWrite;
- TextRec(F).FlushFunc:=@ErrorWrite;
- TextRec(F).CloseFunc:=@ErrorClose;
- ErrorOpen:=0;
- End;
- procedure AssignError(Var T: Text);
- begin
- Assign(T,'');
- TextRec(T).OpenFunc:=@ErrorOpen;
- Rewrite(T);
- end;
- const
- Exe_entry_code : pointer = @Exe_entry;
- Dll_entry_code : pointer = @Dll_entry;
- begin
- { get some helpful informations }
- GetStartupInfo(@startupinfo);
- { some misc Win32 stuff }
- hprevinst:=0;
- if not IsLibrary then
- HInstance:=getmodulehandle(GetCommandFile);
- MainInstance:=HInstance;
- { No idea how to know this issue !! }
- IsMultithreaded:=false;
- cmdshow:=startupinfo.wshowwindow;
- { to test stack depth }
- loweststack:=maxlongint;
- { real test stack depth }
- { stacklimit := setupstack; }
- { Setup heap }
- InitHeap;
- InitExceptions;
- { Setup stdin, stdout and stderr, for GUI apps redirect stderr,stdout to be
- displayed in and messagebox }
- StdInputHandle:=longint(GetStdHandle(STD_INPUT_HANDLE));
- StdOutputHandle:=longint(GetStdHandle(STD_OUTPUT_HANDLE));
- StdErrorHandle:=longint(GetStdHandle(STD_ERROR_HANDLE));
- if not IsConsole then
- begin
- AssignError(stderr);
- AssignError(stdout);
- Assign(Output,'');
- Assign(Input,'');
- end
- else
- begin
- OpenStdIO(Input,fmInput,StdInputHandle);
- OpenStdIO(Output,fmOutput,StdOutputHandle);
- OpenStdIO(StdOut,fmOutput,StdOutputHandle);
- OpenStdIO(StdErr,fmOutput,StdErrorHandle);
- end;
- { Arguments }
- setup_arguments;
- { Reset IO Error }
- InOutRes:=0;
- { Reset internal error variable }
- errno:=0;
- end.
- {
- $Log$
- Revision 1.3 2000-09-04 19:36:59 peter
- * new heapalloc calls, patch from Thomas Schatzl
- Revision 1.2 2000/07/13 11:33:58 michael
- + removed logs
- }
|