{ This file is part of the Free Pascal run time library. Copyright (c) 1999-2005 by Florian Klaempfl and Pavel Ozerski and Yury Sidorov member of the Free Pascal development team. FPC Pascal system unit for the WinCE. 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 {$ifdef SYSTEMDEBUG} {$define SYSTEMEXCEPTIONDEBUG} {$endif SYSTEMDEBUG} {$define WINCE_EXCEPTION_HANDLING} { include system-independent routine headers } {$I systemh.inc} const LineEnding = #13#10; LFNSupport = true; DirectorySeparator = '\'; DriveSeparator = ':'; PathSeparator = ';'; { FileNameCaseSensitive is defined separately below!!! } maxExitCode = 65535; MaxPathLen = 260; const { Default filehandles } UnusedHandle : THandle = -1; StdInputHandle : THandle = 0; StdOutputHandle : THandle = 0; StdErrorHandle : THandle = 0; FileNameCaseSensitive : boolean = true; CtrlZMarksEOF: boolean = true; (* #26 not considered as end of file *) sLineBreak = LineEnding; DefaultTextLineBreakStyle : TTextLineBreakStyle = tlbsCRLF; { Thread count for DLL } Thread_count : longint = 0; var { C compatible arguments } argc : longint; argv : ppchar; { WinCE Info } hprevinst, MainInstance, DLLreason,DLLparam:longint; Win32StackTop : Dword; // Used by heaptrc unit 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; { ANSI <-> Wide } function AnsiToWideBuf(AnsiBuf: PChar; AnsiBufLen: longint; WideBuf: PWideChar; WideBufLen: longint): longint; function WideToAnsiBuf(WideBuf: PWideChar; WideBufLen: longint; AnsiBuf: PChar; AnsiBufLen: longint): longint; function PCharToPWideChar(str: PChar; strlen: longint = -1; outlen: PLongInt = nil): PWideChar; function StringToPWideChar(const s: AnsiString; outlen: PLongInt = nil): PWideChar; { Wrappers for some WinAPI calls } function CreateEvent(lpEventAttributes:pointer;bManualReset:longbool;bInitialState:longbool;lpName:pchar): THandle; stdcall; function ResetEvent(h: THandle): LONGBOOL; stdcall; function SetEvent(h: THandle): LONGBOOL; stdcall; function GetCurrentProcessId:DWORD; stdcall; function Win32GetCurrentThreadId:DWORD; stdcall; function TlsAlloc : DWord; stdcall; function TlsFree(dwTlsIndex : DWord) : LongBool; stdcall; function GetFileAttributes(p : pchar) : dword; stdcall; function DeleteFile(p : pchar) : longint; stdcall; function MoveFile(old,_new : pchar) : longint; stdcall; function CreateFile(lpFileName:pchar; dwDesiredAccess:DWORD; dwShareMode:DWORD; lpSecurityAttributes:pointer; dwCreationDisposition:DWORD; dwFlagsAndAttributes:DWORD; hTemplateFile:DWORD):longint; stdcall; function CreateDirectory(name : pointer;sec : pointer) : longbool; stdcall; function RemoveDirectory(name:pointer):longbool; stdcall; {$ifdef CPUARM} { the external directive isn't really necessary here because it is overriden by external (FK) } function addd(d1,d2 : double) : double; compilerproc; cdecl;external 'coredll' name '__addd'; function subd(d1,d2 : double) : double; compilerproc; cdecl;external 'coredll' name '__subd'; function muld(d1,d2 : double) : double; compilerproc; cdecl;external 'coredll' name '__muld'; function divd(d1,d2 : double) : double; compilerproc; cdecl;external 'coredll' name '__divd'; function eqd(d1,d2 : double) : boolean; compilerproc; cdecl;external 'coredll' name '__eqd'; function ned(d1,d2 : double) : boolean; compilerproc; cdecl;external 'coredll' name '__ned'; function ltd(d1,d2 : double) : boolean; compilerproc; cdecl;external 'coredll' name '__ltd'; function gtd(d1,d2 : double) : boolean; compilerproc; cdecl;external 'coredll' name '__gtd'; function ged(d1,d2 : double) : boolean; compilerproc; cdecl;external 'coredll' name '__ged'; function led(d1,d2 : double) : boolean; compilerproc; cdecl;external 'coredll' name '__led'; { ***************** single ******************** } function eqs(d1,d2 : single) : boolean; compilerproc; cdecl;external 'coredll' name '__eqs'; function nes(d1,d2 : single) : boolean; compilerproc; cdecl;external 'coredll' name '__nes'; function lts(d1,d2 : single) : boolean; compilerproc; cdecl;external 'coredll' name '__lts'; function gts(d1,d2 : single) : boolean; compilerproc; cdecl;external 'coredll' name '__gts'; function ges(d1,d2 : single) : boolean; compilerproc; cdecl;external 'coredll' name '__ges'; function les(d1,d2 : single) : boolean; compilerproc; cdecl;external 'coredll' name '__les'; function dtos(d : double) : single; compilerproc; cdecl;external 'coredll' name '__dtos'; function stod(d : single) : double; compilerproc; cdecl;external 'coredll' name '__stod'; function negs(d : single) : single; compilerproc; cdecl;external 'coredll' name '__negs'; function negd(d : double) : double; compilerproc; cdecl;external 'coredll' name '__negd'; function utod(i : dword) : double; compilerproc; cdecl;external 'coredll' name '__utod'; function itod(i : longint) : double; compilerproc; cdecl;external 'coredll' name '__itod'; function ui64tod(i : qword) : double; compilerproc; cdecl;external 'coredll' name '__u64tod'; function i64tod(i : int64) : double; compilerproc; cdecl;external 'coredll' name '__i64tod'; function adds(s1,s2 : single) : single; compilerproc; function subs(s1,s2 : single) : single; compilerproc; function muls(s1,s2 : single) : single; compilerproc; function divs(s1,s2 : single) : single; compilerproc; {$endif CPUARM} implementation var SysInstance : Longint; {$define HAS_RESOURCES} {$i winres.inc} function MessageBox(w1:longint;l1,l2:PWideChar;w2:longint):longint; stdcall;external 'coredll' name 'MessageBoxW'; {*****************************************************************************} {$define FPC_SYSTEM_HAS_MOVE} procedure memmove(dest, src: pointer; count: longint); cdecl; external 'coredll' name 'memmove'; procedure Move(const source;var dest;count:SizeInt);[public, alias: 'FPC_MOVE']; begin memmove(@dest, @source, count); end; {$define FPC_SYSTEM_HAS_COMPAREBYTE} function memcmp(buf1, buf2: pointer; count: longint): longint; cdecl; external 'coredll' name 'memcmp'; function CompareByte(Const buf1,buf2;len:SizeInt):SizeInt; begin CompareByte := memcmp(@buf1, @buf2, len); end; {$ifdef CPUARM} {$define FPC_SYSTEM_HAS_INT} function fpc_int_real(d: ValReal): ValReal;compilerproc; begin fpc_int_real := i64tod(trunc(d)); end; {$define FPC_SYSTEM_HAS_TRUNC} function fpc_trunc_real(d : ValReal) : int64;compilerproc; external 'coredll' name '__dtoi64'; {$define FPC_SYSTEM_HAS_ABS} function fpc_abs_real(d : ValReal) : ValReal;compilerproc; external 'coredll' name 'fabs'; {$define FPC_SYSTEM_HAS_SQRT} function fpc_sqrt_real(d : ValReal) : ValReal;compilerproc; external 'coredll' name 'sqrt'; function adds(s1,s2 : single) : single; begin adds := addd(s1, s2); end; function subs(s1,s2 : single) : single; begin subs := subd(s1, s2); end; function muls(s1,s2 : single) : single; begin muls := muld(s1, s2); end; function divs(s1,s2 : single) : single; begin divs := divd(s1, s2); end; {$endif CPUARM} {*****************************************************************************} { include system independent routines } {$I system.inc} {***************************************************************************** ANSI <-> Wide *****************************************************************************} const { MultiByteToWideChar } MB_PRECOMPOSED = 1; MB_COMPOSITE = 2; MB_ERR_INVALID_CHARS = 8; MB_USEGLYPHCHARS = 4; CP_ACP = 0; CP_OEMCP = 1; function MultiByteToWideChar(CodePage:UINT; dwFlags:DWORD; lpMultiByteStr:PChar; cchMultiByte:longint; lpWideCharStr:PWideChar;cchWideChar:longint):longint; stdcall; external 'coredll' name 'MultiByteToWideChar'; function WideCharToMultiByte(CodePage:UINT; dwFlags:DWORD; lpWideCharStr:PWideChar; cchWideChar:longint; lpMultiByteStr:PChar;cchMultiByte:longint; lpDefaultChar:PChar; lpUsedDefaultChar:pointer):longint; stdcall; external 'coredll' name 'WideCharToMultiByte'; function AnsiToWideBuf(AnsiBuf: PChar; AnsiBufLen: longint; WideBuf: PWideChar; WideBufLen: longint): longint; begin Result := MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, AnsiBuf, AnsiBufLen, WideBuf, WideBufLen div SizeOf(WideChar)); if ((AnsiBufLen <> -1) or (Result = 0)) and (WideBuf <> nil) then begin if (Result + 1)*SizeOf(WideChar) > WideBufLen then begin Result := 0; if WideBufLen < SizeOf(WideChar) then exit; end; WideBuf[Result] := #0; if (Result <> 0) or (AnsiBufLen = 0) then Inc(Result); end; Result:=Result*SizeOf(WideChar); end; function WideToAnsiBuf(WideBuf: PWideChar; WideBufLen: longint; AnsiBuf: PChar; AnsiBufLen: longint): longint; begin Result := WideCharToMultiByte(CP_ACP, 0, WideBuf, WideBufLen, AnsiBuf, AnsiBufLen, nil, nil); if ((WideBufLen <> -1) or (Result = 0)) and (AnsiBuf <> nil) then begin if Result + 1 > AnsiBufLen then begin Result := 0; if AnsiBufLen < 1 then exit; end; AnsiBuf[Result] := #0; if (Result <> 0) or (WideBufLen = 0) then Inc(Result); end; end; function PCharToPWideChar(str: PChar; strlen: longint = -1; outlen: PLongInt = nil): PWideChar; var len: longint; begin while True do begin if strlen <> -1 then len:=(strlen + 1) else len:=AnsiToWideBuf(str, -1, nil, 0); if len > 0 then begin len:=len*SizeOf(WideChar); GetMem(Result, len); if (AnsiToWideBuf(str, -1, Result, len) = 0) and (strlen <> -1) then begin strlen:=-1; continue; end; end else begin GetMem(Result, SizeOf(WideChar)); Inc(len); Result^:=#0; end; break; end; if outlen <> nil then outlen^:=(len - 1)*SizeOf(WideChar); end; function StringToPWideChar(const s: AnsiString; outlen: PLongInt = nil): PWideChar; var len, wlen: longint; begin len:=Length(s); wlen:=(len + 1)*SizeOf(WideChar); GetMem(Result, wlen); wlen:=AnsiToWideBuf(PChar(s), len, Result, wlen); if wlen = 0 then begin wlen:=AnsiToWideBuf(PChar(s), len, nil, 0); if wlen > 0 then begin ReAllocMem(Result, wlen); wlen:=AnsiToWideBuf(PChar(s), len, Result, wlen); end else begin Result^:=#0; wlen:=SizeOf(WideChar); end; end; if outlen <> nil then outlen^:=(wlen - 1) div SizeOf(WideChar); end; {***************************************************************************** WinAPI wrappers implementation *****************************************************************************} function GetFileAttributesW(p : pwidechar) : dword; stdcall;external KernelDLL name 'GetFileAttributesW'; function DeleteFileW(p : pwidechar) : longint; stdcall;external KernelDLL name 'DeleteFileW'; function MoveFileW(old,_new : pwidechar) : longint; stdcall;external KernelDLL name 'MoveFileW'; function CreateFileW(lpFileName:pwidechar; dwDesiredAccess:DWORD; dwShareMode:DWORD; lpSecurityAttributes:pointer; dwCreationDisposition:DWORD; dwFlagsAndAttributes:DWORD; hTemplateFile:DWORD):longint; stdcall;external KernelDLL name 'CreateFileW'; function CreateDirectoryW(name : pwidechar;sec : pointer) : longbool; stdcall;external KernelDLL name 'CreateDirectoryW'; function RemoveDirectoryW(name:pwidechar):longbool; stdcall;external KernelDLL name 'RemoveDirectoryW'; function GetFileAttributes(p : pchar) : dword; stdcall; var buf: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(p, -1, buf, SizeOf(buf)); GetFileAttributes := GetFileAttributesW(buf); end; function DeleteFile(p : pchar) : longint; stdcall; var buf: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(p, -1, buf, SizeOf(buf)); DeleteFile := DeleteFileW(buf); end; function MoveFile(old,_new : pchar) : longint; stdcall; var buf_old, buf_new: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(old, -1, buf_old, SizeOf(buf_old)); AnsiToWideBuf(_new, -1, buf_new, SizeOf(buf_new)); MoveFile := MoveFileW(buf_old, buf_new); end; function CreateFile(lpFileName:pchar; dwDesiredAccess:DWORD; dwShareMode:DWORD; lpSecurityAttributes:pointer; dwCreationDisposition:DWORD; dwFlagsAndAttributes:DWORD; hTemplateFile:DWORD):longint; stdcall; var buf: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(lpFileName, -1, buf, SizeOf(buf)); CreateFile := CreateFileW(buf, dwDesiredAccess, dwShareMode, lpSecurityAttributes, dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile); end; function CreateDirectory(name : pointer;sec : pointer) : longbool; stdcall; var buf: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(name, -1, buf, SizeOf(buf)); CreateDirectory := CreateDirectoryW(buf, sec); end; function RemoveDirectory(name:pointer):longbool; stdcall; var buf: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(name, -1, buf, SizeOf(buf)); RemoveDirectory := RemoveDirectoryW(buf); end; const {$ifdef CPUARM} UserKData = $FFFFC800; {$else CPUARM} UserKData = $00005800; {$endif CPUARM} SYSHANDLE_OFFSET = $004; SYS_HANDLE_BASE = 64; SH_CURTHREAD = 1; SH_CURPROC = 2; type PHandle = ^THandle; const EVENT_PULSE = 1; EVENT_RESET = 2; EVENT_SET = 3; function CreateEventW(lpEventAttributes:pointer;bManualReset:longbool;bInitialState:longbool;lpName:PWideChar): THandle; stdcall; external KernelDLL name 'CreateEventW'; function CreateEvent(lpEventAttributes:pointer;bManualReset:longbool;bInitialState:longbool;lpName:pchar): THandle; stdcall; var buf: array[0..MaxPathLen] of WideChar; begin AnsiToWideBuf(lpName, -1, buf, SizeOf(buf)); CreateEvent := CreateEventW(lpEventAttributes, bManualReset, bInitialState, buf); end; function EventModify(h: THandle; func: DWORD): LONGBOOL; stdcall; external KernelDLL name 'EventModify'; function TlsCall(p1, p2: DWORD): DWORD; stdcall; external KernelDLL name 'TlsCall'; function ResetEvent(h: THandle): LONGBOOL; stdcall; begin ResetEvent := EventModify(h,EVENT_RESET); end; function SetEvent(h: THandle): LONGBOOL; stdcall; begin SetEvent := EventModify(h,EVENT_SET); end; function GetCurrentProcessId:DWORD; stdcall; var p: PHandle; begin p:=PHandle(UserKData+SYSHANDLE_OFFSET + SH_CURPROC*SizeOf(THandle)); GetCurrentProcessId := p^; end; function Win32GetCurrentThreadId:DWORD; stdcall; var p: PHandle; begin p:=PHandle(UserKData+SYSHANDLE_OFFSET + SH_CURTHREAD*SizeOf(THandle)); Win32GetCurrentThreadId := p^; end; const TLS_FUNCALLOC = 0; TLS_FUNCFREE = 1; function TlsAlloc : DWord; stdcall; begin TlsAlloc := TlsCall(TLS_FUNCALLOC, 0); end; function TlsFree(dwTlsIndex : DWord) : LongBool; stdcall; begin TlsFree := LongBool(TlsCall(TLS_FUNCFREE, dwTlsIndex)); end; {***************************************************************************** Parameter Handling *****************************************************************************} function GetCommandLine : pwidechar; stdcall;external KernelDLL name 'GetCommandLineW'; var ModuleName : array[0..255] of char; function GetCommandFile:pchar; var buf: array[0..MaxPathLen] of WideChar; begin if ModuleName[0] = #0 then begin GetModuleFileName(0, @buf, SizeOf(buf)); WideToAnsiBuf(buf, -1, @ModuleName, SizeOf(ModuleName)); end; GetCommandFile:=@ModuleName; end; procedure setup_arguments; var arglen, count : longint; argstart, pc,arg : pchar; quote : char; argvlen : longint; procedure allocarg(idx,len:longint); var oldargvlen : longint; begin if idx>=argvlen then begin oldargvlen:=argvlen; argvlen:=(idx+8) and (not 7); sysreallocmem(argv,argvlen*sizeof(pointer)); fillchar(argv[oldargvlen],(argvlen-oldargvlen)*sizeof(pointer),0); end; { use realloc to reuse already existing memory } { always allocate, even if length is zero, since } { the arg. is still present! } sysreallocmem(argv[idx],len+1); end; begin { create commandline, it starts with the executed filename which is argv[0] } { WinCE passes the command NOT via the args, but via getmodulefilename} argv:=nil; argvlen:=0; pc:=getcommandfile; Arglen:=0; while pc[Arglen] <> #0 do Inc(Arglen); allocarg(0,arglen); move(pc^,argv[0]^,arglen+1); { Setup cmdline variable } arg:=PChar(GetCommandLine); count:=WideToAnsiBuf(PWideChar(arg), -1, nil, 0); cmdline:=SysGetMem(arglen + count + 3); cmdline^:='"'; move(pc^, (cmdline + 1)^, arglen); (cmdline + arglen + 1)^:='"'; (cmdline + arglen + 2)^:=' '; WideToAnsiBuf(PWideChar(arg), -1, cmdline + arglen + 3, count); { process arguments } count:=0; pc:=cmdline; {$IfDef SYSTEM_DEBUG_STARTUP} Writeln(stderr,'WinCE GetCommandLine is #',pc,'#'); {$EndIf } 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:=' '; argstart:=pc; arglen:=0; 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; { copy argument } { Don't copy the first one, it is already there.} If Count<>0 then begin allocarg(count,arglen); quote:=' '; pc:=argstart; arg:=argv[count]; 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; end; {$IfDef SYSTEM_DEBUG_STARTUP} Writeln(stderr,'dos arg ',count,' #',arglen,'#',argv[count],'#'); {$EndIf SYSTEM_DEBUG_STARTUP} inc(count); end; { get argc and create an nil entry } argc:=count; allocarg(argc,0); { free unused memory } sysreallocmem(argv,(argc+1)*sizeof(pointer)); end; function paramcount : longint; begin paramcount := argc - 1; end; function paramstr(l : longint) : string; begin if (l>=0) and (l 0) then dec(exceptLevel); eip:=exceptEip[exceptLevel]; error:=exceptError[exceptLevel]; {$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then writeln(stderr,'In JumpToHandleErrorFrame error=',error); {$endif SYSTEMEXCEPTIONDEBUG} if resetFPU[exceptLevel] then asm fninit fldcw fpucw end; { build a fake stack } asm {$ifdef REGCALL} movl ebp,%ecx movl eip,%edx movl error,%eax pushl eip movl ebp,%ebp // Change frame pointer {$else} 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 {$endif} {$ifdef SYSTEMEXCEPTIONDEBUG} jmpl DebugHandleErrorAddrFrame {$else not SYSTEMEXCEPTIONDEBUG} jmpl HandleErrorAddrFrame {$endif SYSTEMEXCEPTIONDEBUG} end; end; function i386_exception_handler(ExceptionRecord: PExceptionRecord; EstablisherFrame: pointer; ContextRecord: PContext; DispatcherContext: pointer): longint; cdecl; var res: longint; must_reset_fpu: boolean; begin res := ExceptionContinueSearch; if ContextRecord^.SegSs=_SS then begin must_reset_fpu := true; {$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then Writeln(stderr,'Exception ', hexstr(excep^.ExceptionRecord^.ExceptionCode, 8)); {$endif SYSTEMEXCEPTIONDEBUG} case cardinal(ExceptionRecord^.ExceptionCode) of STATUS_INTEGER_DIVIDE_BY_ZERO, STATUS_FLOAT_DIVIDE_BY_ZERO : res := 200; STATUS_ARRAY_BOUNDS_EXCEEDED : begin res := 201; must_reset_fpu := false; end; STATUS_STACK_OVERFLOW : begin res := 202; must_reset_fpu := false; end; STATUS_FLOAT_OVERFLOW : res := 205; STATUS_FLOAT_DENORMAL_OPERAND, STATUS_FLOAT_UNDERFLOW : res := 206; {excep^.ContextRecord^.FloatSave.StatusWord := excep^.ContextRecord^.FloatSave.StatusWord and $ffffff00;} STATUS_FLOAT_INEXACT_RESULT, STATUS_FLOAT_INVALID_OPERATION, STATUS_FLOAT_STACK_CHECK : res := 207; STATUS_INTEGER_OVERFLOW : begin res := 215; must_reset_fpu := false; end; STATUS_ILLEGAL_INSTRUCTION: res := 216; STATUS_ACCESS_VIOLATION: { Athlon prefetch bug? } if is_prefetch(pointer(ContextRecord^.Eip)) then begin { if yes, then retry } ExceptionRecord^.ExceptionCode := 0; res:=ExceptionContinueExecution; end else res := 216; STATUS_CONTROL_C_EXIT: res := 217; STATUS_PRIVILEGED_INSTRUCTION: begin res := 218; must_reset_fpu := false; end; else begin if ((ExceptionRecord^.ExceptionCode and SEVERITY_ERROR) = SEVERITY_ERROR) then res := 217 else res := 255; end; end; if (res >= 200) and (exceptLevel < MaxExceptionLevel) then begin exceptEip[exceptLevel] := ContextRecord^.Eip; exceptError[exceptLevel] := res; resetFPU[exceptLevel] := must_reset_fpu; inc(exceptLevel); ContextRecord^.Eip := Longint(@JumpToHandleErrorFrame); ExceptionRecord^.ExceptionCode := 0; res := ExceptionContinueExecution; {$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then begin writeln(stderr,'Exception Continue Exception set at ', hexstr(exceptEip[exceptLevel],8)); writeln(stderr,'Eip changed to ', hexstr(longint(@JumpToHandleErrorFrame),8), ' error=', error); end; {$endif SYSTEMEXCEPTIONDEBUG} end; end; i386_exception_handler := res; end; {$endif CPUI386} {$ifdef CPUARM} {**************************** ARM Exception handling *****************************************} var exceptPC : array[0..MaxExceptionLevel-1] of Longint; exceptError : array[0..MaxExceptionLevel-1] of Byte; procedure JumpToHandleErrorFrame; var _pc, _fp, _error : Longint; begin // get original fp asm ldr r0,[r11,#-12] str r0,_fp end; if (exceptLevel > 0) then dec(exceptLevel); _pc:=exceptPC[exceptLevel]; _error:=exceptError[exceptLevel]; asm ldr r0,_error ldr r1,_pc ldr r2,_fp mov r11,r2 // Change frame pointer b HandleErrorAddrFrame end; end; function ARM_ExceptionHandler(ExceptionRecord: PExceptionRecord; EstablisherFrame: pointer; ContextRecord: PContext; DispatcherContext: pointer): longint; [public, alias : '_ARM_ExceptionHandler']; var res: longint; begin res := ExceptionContinueSearch; case cardinal(ExceptionRecord^.ExceptionCode) of STATUS_INTEGER_DIVIDE_BY_ZERO, STATUS_FLOAT_DIVIDE_BY_ZERO : res := 200; STATUS_ARRAY_BOUNDS_EXCEEDED : res := 201; STATUS_STACK_OVERFLOW : res := 202; STATUS_FLOAT_OVERFLOW : res := 205; STATUS_FLOAT_DENORMAL_OPERAND, STATUS_FLOAT_UNDERFLOW : res := 206; STATUS_FLOAT_INEXACT_RESULT, STATUS_FLOAT_INVALID_OPERATION, STATUS_FLOAT_STACK_CHECK : res := 207; STATUS_INTEGER_OVERFLOW : res := 215; STATUS_ILLEGAL_INSTRUCTION: res := 216; STATUS_ACCESS_VIOLATION: res := 216; STATUS_DATATYPE_MISALIGNMENT: res := 214; STATUS_CONTROL_C_EXIT: res := 217; STATUS_PRIVILEGED_INSTRUCTION: res := 218; else begin if ((cardinal(ExceptionRecord^.ExceptionCode) and SEVERITY_ERROR) = SEVERITY_ERROR) then res := 217 else res := 255; end; end; if (res <> ExceptionContinueSearch) and (exceptLevel < MaxExceptionLevel) then begin exceptPC[exceptLevel] := ContextRecord^.PC; exceptError[exceptLevel] := res; inc(exceptLevel); ContextRecord^.PC := Longint(@JumpToHandleErrorFrame); ExceptionRecord^.ExceptionCode := 0; res := ExceptionContinueExecution; {$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then begin writeln(stderr,'Exception Continue Exception set at ', hexstr(exceptEip[exceptLevel],8)); writeln(stderr,'Eip changed to ', hexstr(longint(@JumpToHandleErrorFrame),8), ' error=', error); end; {$endif SYSTEMEXCEPTIONDEBUG} end; ARM_ExceptionHandler := res; end; {$endif CPUARM} {$endif WINCE_EXCEPTION_HANDLING} procedure Exe_entry;[public, alias : '_FPC_EXE_Entry']; begin IsLibrary:=false; {$ifdef CPUARM} asm mov fp,#0 ldr r12,.LPWin32StackTop str sp,[r12] bl PASCALMAIN; b .Lend .LPWin32StackTop: .long Win32StackTop .Lend: end; {$endif CPUARM} {$ifdef CPUI386} asm {$ifdef WINCE_EXCEPTION_HANDLING} pushl i386_exception_handler pushl %fs:(0) mov %esp,%fs:(0) {$endif WINCE_EXCEPTION_HANDLING} pushl %ebp xorl %ebp,%ebp movl %esp,%eax movl %eax,Win32StackTop movw %ss,%bp movl %ebp,_SS call SysResetFPU xorl %ebp,%ebp call PASCALMAIN popl %ebp {$ifdef WINCE_EXCEPTION_HANDLING} popl %fs:(0) addl $4, %esp {$endif WINCE_EXCEPTION_HANDLING} end; {$endif CPUI386} { if we pass here there was no error ! } system_exit; end; {**************************************************************************** OS dependend widestrings ****************************************************************************} function CharUpperBuff(lpsz:LPWSTR; cchLength:DWORD):DWORD; stdcall; external KernelDLL name 'CharUpperBuffW'; function CharLowerBuff(lpsz:LPWSTR; cchLength:DWORD):DWORD; stdcall; external KernelDLL name 'CharLowerBuffW'; procedure WinCEWide2AnsiMove(source:pwidechar;var dest:ansistring;len:SizeInt); var i: integer; begin if len = 0 then dest:='' else begin for i:=1 to 2 do begin setlength(dest, len); len:=WideCharToMultiByte(CP_ACP, 0, source, len, @dest[1], len, nil, nil); if len > 0 then break; len:=WideCharToMultiByte(CP_ACP, 0, source, len, nil, 0, nil, nil); end; setlength(dest, len); end; end; procedure WinCEAnsi2WideMove(source:pchar;var dest:widestring;len:SizeInt); var i: integer; begin if len = 0 then dest:='' else begin for i:=1 to 2 do begin setlength(dest, len); len:=MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, source, len, @dest[1], len); if len > 0 then break; len:=MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, source, len, nil, 0); end; setlength(dest, len); end; end; function WinCEWideUpper(const s : WideString) : WideString; begin result:=s; UniqueString(result); if length(result)>0 then CharUpperBuff(LPWSTR(result),length(result)); end; function WinCEWideLower(const s : WideString) : WideString; begin result:=s; UniqueString(result); if length(result)>0 then CharLowerBuff(LPWSTR(result),length(result)); end; { there is a similiar procedure in sysutils which inits the fields which are only relevant for the sysutils units } procedure InitWinCEWidestrings; begin widestringmanager.Wide2AnsiMoveProc:=@WinCEWide2AnsiMove; widestringmanager.Ansi2WideMoveProc:=@WinCEAnsi2WideMove; widestringmanager.UpperWideStringProc:=@WinCEWideUpper; widestringmanager.LowerWideStringProc:=@WinCEWideLower; end; {**************************************************************************** Error Message writing using messageboxes ****************************************************************************} const ErrorBufferLength = 1024; var ErrorBuf : array[0..ErrorBufferLength] of char; ErrorBufW : array[0..ErrorBufferLength] of widechar; 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 AnsiToWideBuf(@ErrorBuf, -1, @ErrorBufW, SizeOf(ErrorBufW)); MessageBox(0,@ErrorBufW,'Error',0); ErrorLen:=0; end; F.BufPos:=0; ErrorWrite:=0; End; Function ErrorClose(Var F: TextRec): Integer; begin if ErrorLen>0 then begin AnsiToWideBuf(@ErrorBuf, -1, @ErrorBufW, SizeOf(ErrorBufW)); MessageBox(0,@ErrorBufW,'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; function _getstdfilex(fd: integer): pointer; cdecl; external 'coredll'; function _fileno(fd: pointer): THandle; cdecl; external 'coredll'; function _controlfp(new: DWORD; mask: DWORD): DWORD; cdecl; external 'coredll'; procedure SysInitStdIO; begin { Setup stdin, stdout and stderr, for GUI apps redirect stderr,stdout to be displayed in and messagebox } if not IsConsole then begin AssignError(stderr); AssignError(stdout); Assign(Output,''); Assign(Input,''); Assign(ErrOutput,''); end else begin StdInputHandle:=_fileno(_getstdfilex(0)); StdOutputHandle:=_fileno(_getstdfilex(1)); StdErrorHandle:=_fileno(_getstdfilex(2)); OpenStdIO(Input,fmInput,StdInputHandle); OpenStdIO(Output,fmOutput,StdOutputHandle); OpenStdIO(ErrOutput,fmOutput,StdErrorHandle); OpenStdIO(StdOut,fmOutput,StdOutputHandle); OpenStdIO(StdErr,fmOutput,StdErrorHandle); end; end; (* ProcessID cached to avoid repeated calls to GetCurrentProcess. *) var ProcessID: SizeUInt; function GetProcessID: SizeUInt; begin GetProcessID := ProcessID; end; function CheckInitialStkLen(stklen : SizeUInt) : SizeUInt; begin result := stklen; end; const Exe_entry_code : pointer = @Exe_entry; Dll_entry_code : pointer = @Dll_entry; begin StackLength := CheckInitialStkLen(InitialStkLen); StackBottom := Sptr - StackLength; { Enable FPU exceptions } _controlfp(1, $0008001F); { some misc stuff } hprevinst:=0; if not IsLibrary then SysInstance:=GetModuleHandle(nil); MainInstance:=SysInstance; { Setup heap } InitHeap; SysInitExceptions; SysInitStdIO; { Arguments } setup_arguments; { Reset IO Error } InOutRes:=0; ProcessID := GetCurrentProcessID; { threading } InitSystemThreads; { Reset internal error variable } errno:=0; initvariantmanager; initwidestringmanager; InitWinCEWidestrings end.