system.inc 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678
  1. {
  2. $Id$
  3. This file is part of the Free Pascal Run time library.
  4. Copyright (c) 1999-2000 by the Free Pascal development team
  5. See the file COPYING.FPC, included in this distribution,
  6. For details about the copyright.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. **********************************************************************}
  11. {****************************************************************************
  12. Local types
  13. ****************************************************************************}
  14. {
  15. TextRec and FileRec are put in a separate file to make it available to other
  16. units without putting it explicitly in systemh.
  17. This way we keep TP compatibility, and the TextRec definition is available
  18. for everyone who needs it.
  19. }
  20. {$i filerec.inc}
  21. {$i textrec.inc}
  22. Procedure HandleError (Errno : Longint); forward;
  23. Procedure HandleErrorFrame (Errno : longint;frame : longint); forward;
  24. type
  25. FileFunc = Procedure(var t : TextRec);
  26. const
  27. { Random / Randomize constants }
  28. OldRandSeed : Cardinal = 0;
  29. InitialSeed : Boolean = TRUE;
  30. Seed2 : Cardinal = 0;
  31. Seed3 : Cardinal = 0;
  32. { For Error Handling.}
  33. ErrorBase : Longint = 0;
  34. { Used by the ansistrings and maybe also other things in the future }
  35. var
  36. emptychar : char;public name 'FPC_EMPTYCHAR';
  37. {****************************************************************************
  38. Routines which have compiler magic
  39. ****************************************************************************}
  40. {$I innr.inc}
  41. Function lo(i : Integer) : byte; [INTERNPROC: In_lo_Word];
  42. Function lo(w : Word) : byte; [INTERNPROC: In_lo_Word];
  43. Function lo(l : Longint) : Word; [INTERNPROC: In_lo_long];
  44. Function lo(l : DWord) : Word; [INTERNPROC: In_lo_long];
  45. Function hi(i : Integer) : byte; [INTERNPROC: In_hi_Word];
  46. Function hi(w : Word) : byte; [INTERNPROC: In_hi_Word];
  47. Function hi(l : Longint) : Word; [INTERNPROC: In_hi_long];
  48. Function hi(l : DWord) : Word; [INTERNPROC: In_hi_long];
  49. Function lo(q : QWord) : DWord; [INTERNPROC: In_lo_qword];
  50. Function lo(i : Int64) : DWord; [INTERNPROC: In_lo_qword];
  51. Function hi(q : QWord) : DWord; [INTERNPROC: In_hi_qword];
  52. Function hi(i : Int64) : DWord; [INTERNPROC: In_hi_qword];
  53. Function chr(b : byte) : Char; [INTERNPROC: In_chr_byte];
  54. Function Length(s : string) : byte; [INTERNPROC: In_Length_string];
  55. Function Length(c : char) : byte; [INTERNPROC: In_Length_string];
  56. Procedure Reset(var f : TypedFile); [INTERNPROC: In_Reset_TypedFile];
  57. Procedure Rewrite(var f : TypedFile); [INTERNPROC: In_Rewrite_TypedFile];
  58. {****************************************************************************
  59. Include processor specific routines
  60. ****************************************************************************}
  61. {$IFDEF I386}
  62. {$IFDEF M68K}
  63. {$Error Can't determine processor type !}
  64. {$ENDIF}
  65. {$I i386.inc} { Case dependent, don't change }
  66. {$ELSE}
  67. {$IFDEF M68K}
  68. {$I m68k.inc} { Case dependent, don't change }
  69. {$ELSE}
  70. {$Error Can't determine processor type !}
  71. {$ENDIF}
  72. {$ENDIF}
  73. { Include generic pascal only routines which are not defined in the processor
  74. specific include file }
  75. {$I generic.inc}
  76. {****************************************************************************
  77. Set Handling
  78. ****************************************************************************}
  79. { Include set support which is processor specific}
  80. {$I set.inc}
  81. {****************************************************************************
  82. Math Routines
  83. ****************************************************************************}
  84. {$ifndef RTLLITE}
  85. function Hi(b : byte): byte;
  86. begin
  87. Hi := b shr 4
  88. end;
  89. function Lo(b : byte): byte;
  90. begin
  91. Lo := b and $0f
  92. end;
  93. Function swap (X : Word) : Word;[internconst:in_const_swap_word];
  94. Begin
  95. swap:=(X and $ff) shl 8 + (X shr 8)
  96. End;
  97. Function Swap (X : Integer) : Integer;[internconst:in_const_swap_word];
  98. Begin
  99. swap:=(X and $ff) shl 8 + (X shr 8)
  100. End;
  101. Function swap (X : Longint) : Longint;[internconst:in_const_swap_long];
  102. Begin
  103. Swap:=(X and $ffff) shl 16 + (X shr 16)
  104. End;
  105. Function Swap (X : Cardinal) : Cardinal;[internconst:in_const_swap_long];
  106. Begin
  107. Swap:=(X and $ffff) shl 16 + (X shr 16)
  108. End;
  109. Function Swap (X : QWord) : QWord;
  110. Begin
  111. Swap:=(X and $ffffffff) shl 32 + (X shr 32);
  112. End;
  113. Function swap (X : Int64) : Int64;
  114. Begin
  115. Swap:=(X and $ffffffff) shl 32 + (X shr 32);
  116. End;
  117. {$endif RTLLITE}
  118. { Include processor specific routines }
  119. {$I math.inc}
  120. {****************************************************************************
  121. Subroutines for String handling
  122. ****************************************************************************}
  123. { Needs to be before RTTI handling }
  124. {$i sstrings.inc}
  125. { requires sstrings.inc for initval }
  126. {$I int64.inc}
  127. {Requires int64.inc, since that contains the VAL functions for int64 and qword}
  128. {$i astrings.inc}
  129. {$ifdef haswidechar}
  130. {$i wstrings.inc}
  131. {$endif haswidechar}
  132. {*****************************************************************************
  133. Dynamic Array support
  134. *****************************************************************************}
  135. {$i dynarr.inc}
  136. {*****************************************************************************
  137. Object Pascal support
  138. *****************************************************************************}
  139. {$i objpas.inc}
  140. {****************************************************************************
  141. Run-Time Type Information (RTTI)
  142. ****************************************************************************}
  143. {$i rtti.inc}
  144. {****************************************************************************
  145. Random function routines
  146. This implements a very long cycle random number generator by combining
  147. three independant generators. The technique was described in the March
  148. 1987 issue of Byte.
  149. Taken and modified with permission from the PCQ Pascal rtl code.
  150. ****************************************************************************}
  151. {$R-}
  152. {$Q-}
  153. Procedure NewSeed;Forward;
  154. Function Random : Extended;
  155. begin
  156. if (InitialSeed) OR (RandSeed <> OldRandSeed) then
  157. Begin
  158. { This is a pretty complicated affair }
  159. { Initially we must call NewSeed when RandSeed is initalized }
  160. { We must also call NewSeed each time RandSeed is reinitialized }
  161. { DO NOT CHANGE THE ORDER OF DECLARATIONS IN THIS BLOCK }
  162. { UNLESS YOU WANT RANDON TO CRASH OF COURSE (CEC) }
  163. InitialSeed:=FALSE;
  164. OldRandSeed:=RandSeed;
  165. NewSeed;
  166. end;
  167. Inc(RandSeed);
  168. RandSeed := (RandSeed * 706) mod 500009;
  169. OldRandSeed:=RandSeed;
  170. INC(Seed2);
  171. Seed2 := (Seed2 * 774) MOD 600011;
  172. INC(Seed3);
  173. Seed3 := (Seed3 * 871) MOD 765241;
  174. Random :=
  175. frac(RandSeed/500009.0 +
  176. Seed2/600011.0 +
  177. Seed3/765241.0);
  178. end;
  179. Function internRandom(l : Cardinal) : Cardinal;
  180. begin
  181. if (InitialSeed) OR (RandSeed <> OldRandSeed) then
  182. Begin
  183. { This is a pretty complicated affair }
  184. { Initially we must call NewSeed when RandSeed is initalized }
  185. { We must also call NewSeed each time RandSeed is reinitialized }
  186. { DO NOT CHANGE THE ORDER OF DECLARATIONS IN THIS BLOCK }
  187. { UNLESS YOU WANT RANDOM TO CRASH OF COURSE (CEC) }
  188. InitialSeed:=FALSE;
  189. OldRandSeed:=RandSeed;
  190. NewSeed;
  191. end;
  192. Inc(RandSeed);
  193. RandSeed := (RandSeed * 998) mod 1000003;
  194. OldRandSeed:=RandSeed;
  195. if l<>0 then
  196. begin
  197. internRandom := RandSeed mod l;
  198. end
  199. else internRandom:=0;
  200. end;
  201. function random(l:cardinal): cardinal;
  202. begin
  203. random := trunc(random()*l);
  204. end;
  205. {$ifndef cardinalmulfixed}
  206. function random(l:longint): longint;
  207. begin
  208. random := trunc(random()*l);
  209. end;
  210. {$endif cardinalmulfixed}
  211. Procedure NewSeed;
  212. begin
  213. randseed := randseed mod 1000003;
  214. Seed2 := (internRandom(65000) * internRandom(65000)) mod 600011;
  215. Seed3 := (internRandom(65000) * internRandom(65000)) mod 765241;
  216. end;
  217. {****************************************************************************
  218. Memory Management
  219. ****************************************************************************}
  220. {$ifndef RTLLITE}
  221. Function Ptr(sel,off : Longint) : farpointer;[internconst:in_const_ptr];
  222. Begin
  223. ptr:=farpointer((sel shl 4)+off);
  224. End;
  225. Function CSeg : Word;
  226. Begin
  227. Cseg:=0;
  228. End;
  229. Function DSeg : Word;
  230. Begin
  231. Dseg:=0;
  232. End;
  233. Function SSeg : Word;
  234. Begin
  235. Sseg:=0;
  236. End;
  237. {$endif RTLLITE}
  238. {*****************************************************************************
  239. Directory support.
  240. *****************************************************************************}
  241. Procedure getdir(drivenr:byte;Var dir:ansistring);
  242. { this is needed to also allow ansistrings, the shortstring version is
  243. OS dependent }
  244. var
  245. s : shortstring;
  246. begin
  247. getdir(drivenr,s);
  248. dir:=s;
  249. end;
  250. {$ifopt R+}
  251. {$define RangeCheckWasOn}
  252. {$R-}
  253. {$endif opt R+}
  254. {$ifopt I+}
  255. {$define IOCheckWasOn}
  256. {$I-}
  257. {$endif opt I+}
  258. {$ifopt Q+}
  259. {$define OverflowCheckWasOn}
  260. {$Q-}
  261. {$endif opt Q+}
  262. {*****************************************************************************
  263. Miscellaneous
  264. *****************************************************************************}
  265. procedure int_rangeerror;[public,alias:'FPC_RANGEERROR'];
  266. begin
  267. HandleErrorFrame(201,get_frame);
  268. end;
  269. procedure int_overflow;[public,alias:'FPC_OVERFLOW'];
  270. begin
  271. HandleErrorFrame(215,get_frame);
  272. end;
  273. procedure int_iocheck(addr : longint);[saveregisters,public,alias:'FPC_IOCHECK'];
  274. var
  275. l : longint;
  276. begin
  277. if InOutRes<>0 then
  278. begin
  279. l:=InOutRes;
  280. InOutRes:=0;
  281. HandleErrorFrame(l,get_frame);
  282. end;
  283. end;
  284. Function IOResult:Word;
  285. Begin
  286. IOResult:=InOutRes;
  287. InOutRes:=0;
  288. End;
  289. procedure fillchar(var x;count : longint;value : boolean);
  290. begin
  291. fillchar(x,count,byte(value));
  292. end;
  293. procedure fillchar(var x;count : longint;value : char);
  294. begin
  295. fillchar(x,count,byte(value));
  296. end;
  297. {*****************************************************************************
  298. Initialization / Finalization
  299. *****************************************************************************}
  300. const
  301. maxunits=1024; { See also files.pas of the compiler source }
  302. type
  303. TInitFinalRec=record
  304. InitProc,
  305. FinalProc : TProcedure;
  306. end;
  307. TInitFinalTable=record
  308. TableCount,
  309. InitCount : longint;
  310. Procs : array[1..maxunits] of TInitFinalRec;
  311. end;
  312. var
  313. InitFinalTable : TInitFinalTable;external name 'INITFINAL';
  314. procedure InitializeUnits;[public,alias:'FPC_INITIALIZEUNITS'];
  315. var
  316. i : longint;
  317. begin
  318. with InitFinalTable do
  319. begin
  320. for i:=1to TableCount do
  321. begin
  322. if assigned(Procs[i].InitProc) then
  323. Procs[i].InitProc();
  324. InitCount:=i;
  325. end;
  326. end;
  327. end;
  328. procedure FinalizeUnits;[public,alias:'FPC_FINALIZEUNITS'];
  329. begin
  330. with InitFinalTable do
  331. begin
  332. while (InitCount>0) do
  333. begin
  334. // we've to decrement the cound before calling the final. code
  335. // else a halt in the final. code leads to a endless loop
  336. dec(InitCount);
  337. if assigned(Procs[InitCount+1].FinalProc) then
  338. Procs[InitCount+1].FinalProc();
  339. end;
  340. end;
  341. end;
  342. {*****************************************************************************
  343. Error / Exit / ExitProc
  344. *****************************************************************************}
  345. Procedure system_exit;forward;
  346. Procedure do_exit;[Public,Alias:'FPC_DO_EXIT'];
  347. var
  348. current_exit : Procedure;
  349. Begin
  350. while exitProc<>nil Do
  351. Begin
  352. InOutRes:=0;
  353. current_exit:=tProcedure(exitProc);
  354. exitProc:=nil;
  355. current_exit();
  356. End;
  357. { Finalize units }
  358. FinalizeUnits;
  359. { Show runtime error }
  360. If erroraddr<>nil Then
  361. Begin
  362. Writeln(stdout,'Runtime error ',Errorcode,' at 0x',hexstr(Longint(Erroraddr),8));
  363. { to get a nice symify }
  364. Writeln(stdout,BackTraceStrFunc(Longint(Erroraddr)));
  365. dump_stack(stdout,ErrorBase);
  366. Writeln(stdout,'');
  367. End;
  368. { call system dependent exit code }
  369. System_exit;
  370. End;
  371. Procedure Halt(ErrNum: Byte);
  372. Begin
  373. ExitCode:=Errnum;
  374. Do_Exit;
  375. end;
  376. function SysBackTraceStr (Addr: longint): ShortString;
  377. begin
  378. SysBackTraceStr:=' 0x'+HexStr(addr,8);
  379. end;
  380. Procedure HandleErrorAddrFrame (Errno : longint;addr,frame : longint);[public,alias:'FPC_BREAK_ERROR'];
  381. begin
  382. If pointer(ErrorProc)<>Nil then
  383. ErrorProc(Errno,pointer(addr),pointer(frame));
  384. errorcode:=Errno;
  385. exitcode:=Errno;
  386. erroraddr:=pointer(addr);
  387. errorbase:=frame;
  388. halt(errorcode);
  389. end;
  390. Procedure HandleErrorFrame (Errno : longint;frame : longint);
  391. {
  392. Procedure to handle internal errors, i.e. not user-invoked errors
  393. Internal function should ALWAYS call HandleError instead of RunError.
  394. Can be used for exception handlers to specify the frame
  395. }
  396. begin
  397. HandleErrorAddrFrame(Errno,get_caller_addr(frame),get_caller_frame(frame));
  398. end;
  399. Procedure HandleError (Errno : longint);[public,alias : 'FPC_HANDLEERROR'];
  400. {
  401. Procedure to handle internal errors, i.e. not user-invoked errors
  402. Internal function should ALWAYS call HandleError instead of RunError.
  403. }
  404. begin
  405. HandleErrorFrame(Errno,get_frame);
  406. end;
  407. procedure runerror(w : word);[alias: 'FPC_RUNERROR'];
  408. begin
  409. errorcode:=w;
  410. exitcode:=w;
  411. erroraddr:=pointer(get_caller_addr(get_frame));
  412. errorbase:=get_caller_frame(get_frame);
  413. halt(errorcode);
  414. end;
  415. Procedure RunError;
  416. Begin
  417. RunError (0);
  418. End;
  419. Procedure Halt;
  420. Begin
  421. Halt(0);
  422. End;
  423. function do_isdevice(handle:longint):boolean;forward;
  424. Procedure dump_stack(var f : text;bp : Longint);
  425. var
  426. i, prevbp : Longint;
  427. is_dev : boolean;
  428. Begin
  429. prevbp:=bp-1;
  430. i:=0;
  431. is_dev:=do_isdevice(textrec(f).Handle);
  432. while bp > prevbp Do
  433. Begin
  434. Writeln(f,BackTraceStrFunc(get_caller_addr(bp)));
  435. Inc(i);
  436. If ((i>max_frame_dump) and is_dev) or (i>256) Then
  437. exit;
  438. prevbp:=bp;
  439. bp:=get_caller_frame(bp);
  440. End;
  441. End;
  442. Type
  443. PExitProcInfo = ^TExitProcInfo;
  444. TExitProcInfo = Record
  445. Next : PExitProcInfo;
  446. SaveExit : Pointer;
  447. Proc : TProcedure;
  448. End;
  449. const
  450. ExitProcList: PExitProcInfo = nil;
  451. Procedure DoExitProc;
  452. var
  453. P : PExitProcInfo;
  454. Proc : TProcedure;
  455. Begin
  456. P:=ExitProcList;
  457. ExitProcList:=P^.Next;
  458. ExitProc:=P^.SaveExit;
  459. Proc:=P^.Proc;
  460. DisPose(P);
  461. Proc();
  462. End;
  463. Procedure AddExitProc(Proc: TProcedure);
  464. var
  465. P : PExitProcInfo;
  466. Begin
  467. New(P);
  468. P^.Next:=ExitProcList;
  469. P^.SaveExit:=ExitProc;
  470. P^.Proc:=Proc;
  471. ExitProcList:=P;
  472. ExitProc:=@DoExitProc;
  473. End;
  474. {*****************************************************************************
  475. Abstract/Assert support.
  476. *****************************************************************************}
  477. procedure AbstractError;[public,alias : 'FPC_ABSTRACTERROR'];
  478. begin
  479. If pointer(AbstractErrorProc)<>nil then
  480. AbstractErrorProc();
  481. HandleErrorFrame(211,get_frame);
  482. end;
  483. Procedure int_assert(Const Msg,FName:Shortstring;LineNo,ErrorAddr:Longint); [SaveRegisters,Public,Alias : 'FPC_ASSERT'];
  484. begin
  485. if pointer(AssertErrorProc)<>nil then
  486. AssertErrorProc(Msg,FName,LineNo,ErrorAddr)
  487. else
  488. HandleErrorFrame(227,get_frame);
  489. end;
  490. Procedure SysAssert(Const Msg,FName:Shortstring;LineNo,ErrorAddr:Longint);
  491. begin
  492. If msg='' then
  493. write(stderr,'Assertion failed')
  494. else
  495. write(stderr,msg);
  496. Writeln(stderr,' (',FName,', line ',LineNo,').');
  497. Writeln(stderr,'');
  498. end;
  499. {*****************************************************************************
  500. SetJmp/LongJmp support.
  501. *****************************************************************************}
  502. {$i setjump.inc}
  503. {$ifdef IOCheckWasOn}
  504. {$I+}
  505. {$endif}
  506. {$ifdef RangeCheckWasOn}
  507. {$R+}
  508. {$endif}
  509. {$ifdef OverflowCheckWasOn}
  510. {$Q+}
  511. {$endif}
  512. {
  513. $Log$
  514. Revision 1.10 2000-11-13 14:47:46 jonas
  515. * support for range checking when converting from 64bit to something
  516. smaller (32bit, 16bit, 8bit)
  517. * fixed range checking between longint/cardinal and for array indexing
  518. with cardinal (values > $7fffffff were considered negative)
  519. Revision 1.9 2000/11/11 16:12:01 peter
  520. * ptr returns farpointer
  521. Revision 1.8 2000/11/06 21:35:59 peter
  522. * removed some warnings
  523. Revision 1.7 2000/11/04 17:52:46 florian
  524. * fixed linker errors
  525. Revision 1.6 2000/10/13 12:04:03 peter
  526. * FPC_BREAK_ERROR added
  527. Revision 1.5 2000/08/13 17:55:14 michael
  528. + Added some delphi compatibility types
  529. Revision 1.4 2000/08/09 19:31:18 marco
  530. * fixes for val(int64 or qword) to ansistring
  531. Revision 1.3 2000/07/14 10:33:10 michael
  532. + Conditionals fixed
  533. Revision 1.2 2000/07/13 11:33:45 michael
  534. + removed logs
  535. }