Win64.Machine.Mod 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122
  1. MODULE Machine;
  2. (** AUTHOR "pjm,fof"; PURPOSE "Bootstrapping, configuration and machine interface, adaption to windows fof"; *)
  3. (* red marked parts are WinAos specific *)
  4. IMPORT SYSTEM, Trace, Kernel32;
  5. CONST
  6. Version = "WinAos (32bit) ";
  7. DefaultConfigFile = "aos.ini";
  8. DefaultGenericConfigFile = "aosg.ini";
  9. UserConfigFile = "myaos.ini";
  10. MaxCPU* = 8; (* dummy definition to make GC for both Win32 and I386 work *)
  11. DefaultObjectFileExtension* = ".Obw";
  12. (** bits in features variable *)
  13. MTTR* = 12; MMX* = 23;
  14. debug* = FALSE; (** display more debug output during booting *)
  15. CONST
  16. AddressSize = SIZEOF(ADDRESS);
  17. StaticBlockSize = 32; (* static heap block size *)
  18. BlockHeaderSize = 2 * AddressSize;
  19. (* change this when Heaps.HeapBlock is modified *)
  20. RecordDescSize = 4 * AddressSize; (* needs to be adapted in case Heaps.RecordBlockDesc is changed *)
  21. (** standard lock levels (in order) *) (* also refer to Traps.Show *)
  22. TraceOutput* = 0; (* Trace output *)
  23. Memory* = 1; (* Virtual memory management, stack and page allocation *)
  24. Heaps* = 2; (* Storage allocation and Garbage collection *)
  25. Interrupts* = 3; (* Interrupt handling. *)
  26. Modules* = 4; (* Module list *)
  27. Objects* = 5; (* Ready queue *)
  28. Processors* = 6; (* Interprocessor interrupts *)
  29. KernelLog* = 7; (* Atomic output *)
  30. GC* = 8;
  31. MaxLocks = 9; (* { <= 32 } *)
  32. StrongChecks = FALSE;
  33. HeaderSize = 40H; (* cf. Linker0 *)
  34. EndBlockOfs = 38H; (* cf. Linker0 *)
  35. MemoryBlockOfs = BlockHeaderSize + RecordDescSize + BlockHeaderSize; (* memory block (including header) starts at offset HeaderSize *)
  36. MemBlockSize = 128*1024*1024; (* must be multiple of StaticBlockSize *)
  37. MinMemBlockSize = 4*1024*1024;
  38. NilVal = 0;
  39. Second* = 1000; (* frequency of ticks increments in Hz *)
  40. CONST
  41. (* error codes *)
  42. Ok* = 0;
  43. NilAdr* = -1; (* nil value for addresses (not same as pointer NIL value) *)
  44. IsCooperative* = FALSE;
  45. TYPE
  46. Vendor* = ARRAY 13 OF CHAR;
  47. IDMap* = ARRAY 16 OF SHORTINT;
  48. Range* = RECORD
  49. adr*, size*: LONGINT
  50. END;
  51. MemoryBlock* = POINTER TO MemoryBlockDesc;
  52. MemoryBlockDesc* = RECORD
  53. next- {UNTRACED}: MemoryBlock;
  54. startAdr-: ADDRESS; (* sort key in linked list of memory blocks *)
  55. size-: SIZE;
  56. beginBlockAdr-, endBlockAdr-: ADDRESS
  57. END;
  58. (* dummy definition to make GC work for both I386 and Win32 - copied from I386.Machine.Mod, but not really used *)
  59. Stack* = RECORD (** values are read-only *)
  60. low: ADDRESS; (* lowest virtual address that may be allocated for stack *)
  61. adr*: ADDRESS; (* lowest address on allocated stack *) (* exported for Objects only *)
  62. high*: ADDRESS; (* next virtual address after stack *) (* exported for Objects only *)
  63. END;
  64. VAR
  65. LastAddress: RECORD END;
  66. MMXSupport*: BOOLEAN;
  67. SSESupport*: BOOLEAN;
  68. SSE2Support*: BOOLEAN;
  69. SSE3Support-: BOOLEAN; (* PH 04/11*)
  70. SSSE3Support-: BOOLEAN;
  71. SSE41Support-: BOOLEAN;
  72. SSE42Support-: BOOLEAN;
  73. SSE5Support-: BOOLEAN;
  74. AVXSupport-: BOOLEAN;
  75. version*: ARRAY 64 OF CHAR; (** Aos version *)
  76. features*,features2*: SET; (** processor features *)
  77. fcr*: SET; (** default floating-point control register value (default rounding mode is towards -infinity, for ENTIER) *)
  78. mhz*: HUGEINT; (** clock rate of GetTimer() in MHz, or 0 if not known *)
  79. boottime-: HUGEINT; (** in timer units *)
  80. commandLine-: ARRAY 256 OF CHAR;
  81. hin, hout: Kernel32.HANDLE;
  82. VAR
  83. lock-: ARRAY MaxLocks OF CHAR; (* not implemented as SET because of shared access *)
  84. cs: ARRAY MaxLocks OF Kernel32.CriticalSection;
  85. trace: ARRAY 2 OF CHAR;
  86. defaultConfigFile, userConfigFile, traceName: ARRAY Kernel32.MaxPath OF CHAR;
  87. gcThreshold-: SIZE;
  88. bootHeapAdr: ADDRESS; (* initialized by linker, variable name must not be changed, see Win32.Aos.Link *)
  89. bootHeapSize: SIZE; (* initialized by linker, variable name must not be changed, see Win32.Aos.Link *)
  90. memBlockHead-{UNTRACED}, memBlockTail-{UNTRACED}: MemoryBlock; (* head and tail of sorted list of memory blocks *)
  91. (** Convert a string to an integer. Parameter i specifies where in the string scanning should begin (usually 0 in the first call). Scanning stops at the first non-valid character, and i returns the updated position. Parameter s is the string to be scanned. The value is returned as result, or 0 if not valid. Syntax: number = ["-"] digit {digit} ["H" | "h"] . digit = "0" | ... "9" | "A" .. "F" | "a" .. "f" . If the number contains any hexdecimal letter, or if it ends in "H" or "h", it is interpreted as hexadecimal. *)
  92. PROCEDURE StrToInt*( VAR i: LONGINT; CONST s: ARRAY OF CHAR ): LONGINT;
  93. VAR vd, vh, sgn, d: LONGINT; hex: BOOLEAN;
  94. BEGIN
  95. vd := 0; vh := 0; hex := FALSE;
  96. IF s[i] = "-" THEN sgn := -1; INC( i ) ELSE sgn := 1 END;
  97. LOOP
  98. IF (s[i] >= "0") & (s[i] <= "9") THEN d := ORD( s[i] ) - ORD( "0" )
  99. ELSIF (CAP( s[i] ) >= "A") & (CAP( s[i] ) <= "F") THEN d := ORD( CAP( s[i] ) ) - ORD( "A" ) + 10; hex := TRUE
  100. ELSE EXIT
  101. END;
  102. vd := 10 * vd + d; vh := 16 * vh + d; INC( i )
  103. END;
  104. IF CAP( s[i] ) = "H" THEN hex := TRUE; INC( i ) END; (* optional H *)
  105. IF hex THEN vd := vh END;
  106. RETURN sgn * vd
  107. END StrToInt;
  108. (** -- Atomic operations -- *)
  109. (** Atomic INC(x). *)
  110. PROCEDURE -AtomicInc*(VAR x: LONGINT);
  111. CODE {SYSTEM.AMD64}
  112. POP RAX
  113. LOCK
  114. INC DWORD [RAX]
  115. END AtomicInc;
  116. (** Atomic DEC(x). *)
  117. PROCEDURE -AtomicDec*(VAR x: LONGINT);
  118. CODE {SYSTEM.AMD64}
  119. POP RAX
  120. LOCK
  121. DEC DWORD [RAX]
  122. END AtomicDec;
  123. (** Atomic EXCL. *)
  124. PROCEDURE AtomicExcl* (VAR s: SET; bit: LONGINT);
  125. CODE {SYSTEM.AMD64}
  126. MOV EAX, [RBP + bit]
  127. MOV RBX, [RBP + s]
  128. LOCK
  129. BTR [RBX], EAX
  130. END AtomicExcl;
  131. (** Atomic INC(x, y). *)
  132. PROCEDURE -AtomicAdd*(VAR x: LONGINT; y: LONGINT);
  133. CODE {SYSTEM.AMD64}
  134. POP EBX
  135. POP RAX
  136. LOCK
  137. ADD DWORD [RAX], EBX
  138. END AtomicAdd;
  139. (** Atomic test-and-set. Set x = TRUE and return old value of x. *)
  140. PROCEDURE -AtomicTestSet*(VAR x: BOOLEAN): BOOLEAN;
  141. CODE {SYSTEM.AMD64}
  142. POP RBX
  143. MOV AL, 1
  144. XCHG [RBX], AL
  145. END AtomicTestSet;
  146. (* Atomic compare-and-swap. Set x = new if x = old and return old value of x *)
  147. PROCEDURE -AtomicCAS* (VAR x: LONGINT; old, new: LONGINT): LONGINT;
  148. CODE {SYSTEM.AMD64}
  149. POP EBX ; new
  150. POP EAX ; old
  151. POP RCX ; address of x
  152. LOCK CMPXCHG [RCX], EBX ; atomicly compare x with old and set it to new if equal
  153. END AtomicCAS;
  154. (** -- Miscellaneous -- *)
  155. (* Return current instruction pointer *)
  156. PROCEDURE CurrentPC* (): ADDRESS;
  157. CODE {SYSTEM.AMD64}
  158. MOV RAX, [RBP + 8]
  159. END CurrentPC;
  160. (* Return current frame pointer *)
  161. PROCEDURE -CurrentBP* (): ADDRESS;
  162. CODE {SYSTEM.AMD64}
  163. MOV RAX, RBP
  164. END CurrentBP;
  165. (* Set current frame pointer *)
  166. PROCEDURE -SetBP* (bp: ADDRESS);
  167. CODE {SYSTEM.AMD64}
  168. POP RBP
  169. END SetBP;
  170. (* Return current stack pointer *)
  171. PROCEDURE -CurrentSP* (): ADDRESS;
  172. CODE {SYSTEM.AMD64}
  173. MOV RAX, RSP
  174. END CurrentSP;
  175. (* Set current stack pointer *)
  176. PROCEDURE -SetSP* (sp: ADDRESS);
  177. CODE {SYSTEM.AMD64}
  178. POP RSP
  179. END SetSP;
  180. (** This procedure should be called in all spin loops as a hint to the processor (e.g. Pentium 4). *)
  181. PROCEDURE -SpinHint*;
  182. CODE {SYSTEM.AMD64}
  183. PAUSE
  184. END SpinHint;
  185. (** Fill "size" bytes at "destAdr" with "filler". "size" must be multiple of 4. *)
  186. PROCEDURE Fill32* (destAdr: ADDRESS; size: SIZE; filler: LONGINT);
  187. CODE {SYSTEM.AMD64}
  188. MOV RDI, [RBP + destAdr]
  189. MOV RCX, [RBP + size]
  190. MOV EAX, [RBP + filler]
  191. TEST RCX, 3
  192. JZ ok
  193. PUSH 8 ; ASSERT failure
  194. INT 3
  195. ok:
  196. SHR RCX, 2
  197. CLD
  198. REP STOSD
  199. END Fill32;
  200. (** -- Processor initialization -- *)
  201. PROCEDURE -SetFCR( s: SET );
  202. CODE {SYSTEM.i386, SYSTEM.FPU}
  203. FLDCW [ESP] ; parameter s
  204. POP EAX
  205. END SetFCR;
  206. PROCEDURE -FCR( ): SET;
  207. CODE {SYSTEM.i386, SYSTEM.FPU}
  208. PUSH 0
  209. FNSTCW [ESP]
  210. FWAIT
  211. POP EAX
  212. END FCR;
  213. PROCEDURE -InitFPU;
  214. CODE {SYSTEM.i386, SYSTEM.FPU}
  215. FNINIT
  216. END InitFPU;
  217. (** Setup FPU control word of current processor. *)
  218. PROCEDURE SetupFPU*;
  219. BEGIN
  220. InitFPU; SetFCR( fcr )
  221. END SetupFPU;
  222. (** CPU identification. *)
  223. PROCEDURE CPUID*( VAR vendor: Vendor; VAR version: LONGINT; VAR features1,features2: SET );
  224. CODE {SYSTEM.i386, SYSTEM.Pentium}
  225. MOV EAX, 0
  226. CPUID
  227. CMP EAX, 0
  228. JNE ok
  229. MOV ESI, [EBP+vendor]
  230. MOV [ESI], AL ; AL = 0
  231. MOV ESI, [EBP+version]
  232. MOV [ESI], EAX ; EAX = 0
  233. MOV ESI, [EBP+features1]
  234. MOV [ESI], EAX
  235. MOV ESI, [EBP+features2]
  236. MOV [ESI], EAX
  237. JMP end
  238. ok:
  239. MOV ESI, [EBP+vendor]
  240. MOV [ESI], EBX
  241. MOV [ESI+4], EDX
  242. MOV [ESI+8], ECX
  243. MOV BYTE [ESI+12], 0
  244. MOV EAX, 1
  245. CPUID
  246. MOV ESI, [EBP+version]
  247. MOV [ESI], EAX
  248. MOV ESI, [EBP+features1]
  249. MOV [ESI], EDX
  250. MOV ESI, [EBP+features2]
  251. MOV [ESI], ECX
  252. end:
  253. END CPUID;
  254. PROCEDURE GetConfig*( CONST name: ARRAY OF CHAR; VAR val: ARRAY OF CHAR );
  255. CONST ConfigKey = "Configuration";
  256. BEGIN
  257. COPY ("", val);
  258. IF Kernel32.GetPrivateProfileString (ConfigKey, name, "", val, LEN (val), userConfigFile) # 0 THEN
  259. ELSIF Kernel32.GetPrivateProfileString (ConfigKey, name, "", val, LEN (val), defaultConfigFile) # 0 THEN
  260. END;
  261. IF (name = "ObjectFileExtension") & (val = "") THEN
  262. IF Kernel32.Generic THEN
  263. val := ".GofWw";
  264. ELSE
  265. val := ".Obww"
  266. END;
  267. END;
  268. END GetConfig;
  269. PROCEDURE Shutdown*( restart: BOOLEAN );
  270. BEGIN
  271. RemoveTraceFile;
  272. Kernel32.Shutdown( 0 ); (* calls the finalizer of Heaps *)
  273. END Shutdown;
  274. (* Dan: from new Machine *)
  275. PROCEDURE -GetTimer*(): HUGEINT;
  276. CODE {SYSTEM.Pentium}
  277. RDTSC ; set EDX:EAX
  278. END GetTimer;
  279. (* Dan: mono CPU PCs *)
  280. PROCEDURE ID*(): LONGINT;
  281. BEGIN
  282. RETURN 0
  283. END ID;
  284. (**
  285. * Flush Data Cache for the specified virtual address range. If len is negative, flushes the whole cache.
  286. * This is used on some architecture to interact with DMA hardware (e.g. Ethernet and USB. It can be
  287. * left empty on Intel architecture.
  288. *)
  289. PROCEDURE FlushDCacheRange * (adr: ADDRESS; len: LONGINT);
  290. END FlushDCacheRange;
  291. (**
  292. * Invalidate Data Cache for the specified virtual address range. If len is negative, flushes the whole cache.
  293. * This is used on some architecture to interact with DMA hardware (e.g. Ethernet and USB. It can be
  294. * left empty on Intel architecture.
  295. *)
  296. PROCEDURE InvalidateDCacheRange * (adr: ADDRESS; len: LONGINT);
  297. END InvalidateDCacheRange;
  298. (**
  299. * Invalidate Instruction Cache for the specified virtual address range. If len is negative, flushes the whole cache.
  300. * This is used on some architecture to interact with DMA hardware (e.g. Ethernet and USB. It can be
  301. * left empty on Intel architecture.
  302. *)
  303. PROCEDURE InvalidateICacheRange * (adr: ADDRESS; len: LONGINT);
  304. END InvalidateICacheRange;
  305. (* setup MMX, SSE and SSE2..SSE5 and AVX extension *)
  306. PROCEDURE SetupSSE2Ext;
  307. CONST
  308. MMXFlag=23;(*IN features from EBX*)
  309. FXSRFlag = 24;
  310. SSEFlag = 25;
  311. SSE2Flag = 26;
  312. SSE3Flag = 0; (*IN features2 from ECX*) (*PH 04/11*)
  313. SSSE3Flag =9;
  314. SSE41Flag =19;
  315. SSE42Flag =20;
  316. SSE5Flag = 11;
  317. AVXFlag = 28;
  318. BEGIN
  319. MMXSupport := MMXFlag IN features;
  320. SSESupport := SSEFlag IN features;
  321. SSE2Support := SSESupport & (SSE2Flag IN features);
  322. SSE3Support := SSE2Support & (SSE3Flag IN features2);
  323. SSSE3Support := SSE3Support & (SSSE3Flag IN features2); (* PH 04/11*)
  324. SSE41Support := SSE3Support & (SSE41Flag IN features2);
  325. SSE42Support := SSE3Support & (SSE42Flag IN features2);
  326. SSE5Support := SSE3Support & (SSE5Flag IN features2);
  327. AVXSupport := SSE3Support & (AVXFlag IN features2);
  328. IF SSESupport & (FXSRFlag IN features) THEN
  329. (* InitSSE(); *) (*! not privileged mode in Windows not allowed *)
  330. END;
  331. END SetupSSE2Ext;
  332. PROCEDURE ReadCommandLine(VAR commandLine: ARRAY OF CHAR);
  333. VAR adr: ADDRESS; i: LONGINT; ch: CHAR;
  334. BEGIN
  335. adr := Kernel32.GetCommandLine();
  336. SYSTEM.GET(adr,ch);
  337. i := 0;
  338. WHILE (i<LEN(commandLine)-1) & (ch # 0X) DO
  339. commandLine[i] := ch;
  340. INC(adr); INC(i);
  341. SYSTEM.GET(adr,ch);
  342. END;
  343. END ReadCommandLine;
  344. PROCEDURE ParseLine(VAR c: ARRAY OF CHAR; VAR iniFile: ARRAY OF CHAR);
  345. VAR i: LONGINT;
  346. PROCEDURE SkipSpaces;
  347. BEGIN
  348. WHILE (c[i] <= " ") & (c[i] # 0X) DO INC(i) END;
  349. END SkipSpaces;
  350. PROCEDURE SkipName;
  351. BEGIN
  352. WHILE (c[i] > " ") DO INC(i) END;
  353. END SkipName;
  354. PROCEDURE CheckName(CONST name: ARRAY OF CHAR): BOOLEAN;
  355. VAR j: LONGINT;
  356. BEGIN
  357. j := 0;
  358. WHILE (c[i] = name[j]) & (c[i] # 0X) & (name[j] # 0X) DO
  359. INC(i); INC(j);
  360. END;
  361. RETURN (name[j] = 0X);
  362. END CheckName;
  363. PROCEDURE ReadName(VAR name: ARRAY OF CHAR);
  364. VAR j: LONGINT;
  365. BEGIN
  366. SkipSpaces;
  367. j := 0;
  368. WHILE (c[i] > " ") & (j < LEN(name)-1) DO
  369. name[j] := c[i];
  370. INC(i); INC(j);
  371. END;
  372. name[j] := 0X;
  373. END ReadName;
  374. BEGIN
  375. c[LEN(c)-1] := 0X;
  376. i := 0;
  377. SkipSpaces;
  378. SkipName;
  379. SkipSpaces;
  380. IF c[i] = "-" THEN (* option *)
  381. INC(i);
  382. IF CheckName("ini") THEN SkipSpaces; ReadName(iniFile) END;
  383. END;
  384. END ParseLine;
  385. PROCEDURE TraceChar(c: CHAR);
  386. VAR len: LONGINT; b: Kernel32.BOOL;
  387. BEGIN
  388. len := 1;
  389. b := Kernel32.WriteFile(hout,c,len,len,NIL);
  390. END TraceChar;
  391. PROCEDURE SetTraceFile(VAR filename: ARRAY OF CHAR);
  392. BEGIN
  393. Trace.String("trace -> file "); Trace.String(filename); Trace.Ln;
  394. hout := Kernel32.CreateFile(filename, SetToDW({Kernel32.GenericWrite}), SetToDW({Kernel32.FileShareRead}), NIL, Kernel32.CreateAlways, SetToDW({Kernel32.FileAttributeNormal}), Kernel32.NULL);
  395. Kernel32.GetFullPathName(filename, LEN(filename), filename, NIL);
  396. Trace.Char := TraceChar;
  397. END SetTraceFile;
  398. PROCEDURE SetTraceConsole;
  399. VAR res: LONGINT;
  400. BEGIN
  401. Trace.String("trace -> console"); Trace.Ln;
  402. res := Kernel32.AllocConsole ();
  403. hin := Kernel32.GetStdHandle (Kernel32.STDInput);
  404. hout := Kernel32.GetStdHandle (Kernel32.STDOutput);
  405. Trace.Char := TraceChar;
  406. END SetTraceConsole;
  407. PROCEDURE SetupTraceName(VAR traceName: ARRAY OF CHAR);
  408. VAR
  409. ext: ARRAY 256 OF CHAR;
  410. extPos,i,j: LONGINT;
  411. systemTime: Kernel32.SystemTime;
  412. ch: CHAR;
  413. PROCEDURE AppendDecimals(int: LONGINT; from, to: LONGINT);
  414. VAR ten: LONGINT;
  415. BEGIN
  416. WHILE to >= from DO
  417. traceName[i] := CHR(ORD("0")+ int DIV to MOD 10); INC(i);
  418. to := to DIV 10;
  419. END;
  420. END AppendDecimals;
  421. BEGIN
  422. Kernel32.GetLocalTime(systemTime);
  423. extPos := 0;
  424. REPEAT
  425. ch := traceName[i];
  426. IF ch = "." THEN j := 0; extPos := i END;
  427. ext[j] := ch;
  428. INC(j); INC(i);
  429. UNTIL ch = 0X;
  430. IF extPos > 0 THEN i := extPos END;
  431. ext[j] := 0X;
  432. AppendDecimals(systemTime.wYear,1,1000);
  433. AppendDecimals(systemTime.wMonth,1,10);
  434. AppendDecimals(systemTime.wDay,1,10);
  435. traceName[i] := "_"; INC(i);
  436. AppendDecimals(systemTime.wHour,1,10);
  437. AppendDecimals(systemTime.wMinute,1,10);
  438. AppendDecimals(systemTime.wSecond,1,10);
  439. traceName[i] := "_"; INC(i);
  440. AppendDecimals(systemTime.wMilliseconds,10,100);
  441. j := 0;
  442. REPEAT
  443. ch := ext[j];
  444. traceName[i] := ch;
  445. INC(i); INC(j);
  446. UNTIL ch = 0X;
  447. END SetupTraceName;
  448. PROCEDURE RemoveTraceFile;
  449. VAR res: LONGINT;
  450. BEGIN
  451. IF (traceName # "") & (traceName # "Console") THEN
  452. Trace.String("removing "); Trace.String(traceName); Trace.Ln;
  453. (*Trace.Char := LogChar;*)
  454. res := Kernel32.CloseHandle(hout);
  455. TRACE(res);
  456. IF res = 0 THEN
  457. res := Kernel32.GetLastError();
  458. Trace.String("could not close "); Trace.String("; res = "); Trace.Int(res,1); Trace.Ln;
  459. END;
  460. res := Kernel32.DeleteFile(traceName);
  461. TRACE(res);
  462. IF res = 0 THEN
  463. res := Kernel32.GetLastError();
  464. Trace.String("could not delete "); Trace.String(traceName); Trace.String("; res = "); Trace.Int(res,1); Trace.Ln;
  465. END;
  466. TRACE(res);
  467. END;
  468. TRACE(traceName);
  469. END RemoveTraceFile;
  470. PROCEDURE ToExecutablePath(CONST name: ARRAY OF CHAR; VAR fullName: ARRAY OF CHAR);
  471. VAR i,j: LONGINT;
  472. BEGIN
  473. Kernel32.GetModuleFileName(Kernel32.hInstance, fullName, LEN( fullName ) );
  474. j := -1; i := 0;
  475. WHILE fullName[i] # 0X DO
  476. IF fullName[i] = '\' THEN j := i END;
  477. INC( i )
  478. END;
  479. i := 0; INC(j);
  480. WHILE name[i] # 0X DO
  481. fullName[j] := name[i]; INC(i); INC(j);
  482. END;
  483. fullName[j] := 0X;
  484. END ToExecutablePath;
  485. PROCEDURE Append(VAR s: ARRAY OF CHAR; CONST t: ARRAY OF CHAR);
  486. VAR i,j: LONGINT;
  487. BEGIN
  488. i := 0;
  489. WHILE(s[i] # 0X) DO INC(i) END;
  490. j := 0;
  491. WHILE (t[j] # 0X) DO
  492. s[i] := t[j];
  493. INC(i); INC(j);
  494. END;
  495. s[i] := 0X;
  496. END Append;
  497. PROCEDURE Init*;
  498. VAR vendor: Vendor; ver: LONGINT; hfile: Kernel32.HANDLE;
  499. BEGIN
  500. Kernel32.Init;
  501. Trace.String("Machine 1"); Trace.Ln;
  502. (* trace[1] := 0X; Trace.Char := LogChar; Trace.Color := TraceColor; *)
  503. InitLocks();
  504. boottime:=GetTimer();
  505. COPY( Version, version );
  506. Append(version, SYSTEM.Date);
  507. CPUID(vendor, ver, features,features2); SetupSSE2Ext;
  508. fcr := (FCR() - {0,2,3,10,11}) + {0..5,8,9}; (* default FCR RC=00B *)
  509. ReadCommandLine(commandLine);
  510. IF Kernel32.Generic THEN
  511. ToExecutablePath(DefaultGenericConfigFile, defaultConfigFile);
  512. ELSE
  513. ToExecutablePath(DefaultConfigFile, defaultConfigFile);
  514. END;
  515. COPY(UserConfigFile, userConfigFile);
  516. hfile := Kernel32.CreateFile( userConfigFile, SetToDW({Kernel32.GenericRead}), SetToDW({Kernel32.FileShareRead}), NIL , Kernel32.OpenExisting, SetToDW({Kernel32.FileAttributeNormal}), 0 );
  517. IF hfile = Kernel32.InvalidHandleValue THEN
  518. ToExecutablePath(UserConfigFile, userConfigFile);
  519. ELSE
  520. Kernel32.CloseHandle(hfile)
  521. END;
  522. (* ever used ? *)
  523. ParseLine(commandLine, userConfigFile);
  524. userConfigFile[Kernel32.GetFullPathName (userConfigFile, Kernel32.MaxPath, userConfigFile, 0)] := 0X;
  525. Trace.String("config file = "); Trace.String(defaultConfigFile); Trace.Ln;
  526. Trace.String("user config file = "); Trace.String(userConfigFile); Trace.Ln;
  527. traceName[0] := 0X;
  528. GetConfig("Trace",traceName);
  529. Trace.String("traceName "); Trace.String(traceName); Trace.Ln;
  530. IF traceName = "File" THEN
  531. traceName := "SystemTrace.txt";
  532. SetupTraceName(traceName);
  533. Trace.String("trace file to "); Trace.String(traceName); Trace.Ln;
  534. SetTraceFile(traceName);
  535. ELSIF traceName = "Console" THEN SetTraceConsole
  536. (* else trace is on kernel log *)
  537. END;
  538. Trace.String("Machine init done"); Trace.Ln;
  539. END Init;
  540. PROCEDURE {INITIAL, NOPAF} Start;
  541. BEGIN
  542. Init; (* cannot allocate variables in here *)
  543. END Start;
  544. (* Initialize locks. *)
  545. PROCEDURE InitLocks;
  546. VAR i: LONGINT;
  547. BEGIN
  548. i := 0;
  549. WHILE i < MaxLocks DO Kernel32.InitializeCriticalSection( cs[i] ); lock[i] := "N"; INC( i ) END;
  550. END InitLocks;
  551. PROCEDURE CleanupLocks*;
  552. VAR i: LONGINT;
  553. BEGIN
  554. i := 0;
  555. WHILE i < MaxLocks DO Kernel32.DeleteCriticalSection( cs[i] ); INC( i ) END;
  556. END CleanupLocks;
  557. (** Acquire a spin-lock. *)
  558. PROCEDURE Acquire*( level: LONGINT ); (* non reentrant lock (non reentrance "ensured" by ASSERT statement ), CriticalSections are reentrant *)
  559. BEGIN
  560. Kernel32.EnterCriticalSection( cs[level] );
  561. IF StrongChecks THEN
  562. ASSERT ( lock[level] = "N", 1001 );
  563. ELSIF lock[level] # "N" THEN
  564. Trace.String("warning: reentered non-reentrant lock"); Trace.Ln;
  565. END;
  566. lock[level] := "Y";
  567. END Acquire;
  568. (** Release a spin-lock. *)
  569. PROCEDURE Release*( level: LONGINT ); (* release lock *)
  570. BEGIN
  571. IF StrongChecks THEN
  572. ASSERT ( lock[level] ="Y", 1002 );
  573. ELSIF lock[level] # "Y" THEN
  574. Trace.String("warning: reentered non-reentrant lock"); Trace.Ln;
  575. END;
  576. lock[level] := "N";
  577. Kernel32.LeaveCriticalSection( cs[level] )
  578. END Release;
  579. (* added by Alexey *)
  580. PROCEDURE GetMemStatus(VAR stat: Kernel32.MemoryStatusEx): BOOLEAN;
  581. BEGIN
  582. stat.dwLength := 64;
  583. IF Kernel32.GlobalMemoryStatusEx(stat) = 1 THEN
  584. RETURN TRUE;
  585. ELSE
  586. RETURN FALSE;
  587. END;
  588. END GetMemStatus;
  589. (** dummy procedure to make GC work for both I386 and Win32 *)
  590. PROCEDURE GetKernelStacks*(VAR stack: ARRAY OF Stack);
  591. VAR i: LONGINT;
  592. BEGIN
  593. FOR i := 0 TO MaxCPU-1 DO
  594. stack[i].adr := NilVal;
  595. stack[i].high := NilVal
  596. END
  597. END GetKernelStacks;
  598. (* Set machine-dependent parameter gcThreshold *)
  599. PROCEDURE SetGCParams*;
  600. BEGIN
  601. gcThreshold := 10*1024*1024; (* 10 MB *)
  602. END SetGCParams;
  603. (* expand heap by allocating a new memory block - called during GC *)
  604. PROCEDURE InitHeap(VAR memoryBlock: MemoryBlock; VAR beginBlockAdr, endBlockAdr: ADDRESS);
  605. CONST MemBlockHeaderSize = BlockHeaderSize + RecordDescSize + BlockHeaderSize;
  606. TypeDescOffset = -AddressSize; (* see Heaps.Mod *)
  607. HeapBlockOffset = - 2 * AddressSize; (* see Heaps.Mod *)
  608. DataAdrOffset = AddressSize; (* offset of dataAdr field in Heaps.HeapBlockDesc *)
  609. VAR memDescSize, memBlkSize, alignOffset: SIZE; adr, memHeaderAdr, memBlockAdr, memBlockHeadAdr: ADDRESS;
  610. memBlock {UNTRACED}: MemoryBlock; i: LONGINT; ch: CHAR; h: HUGEINT; size: LONGINT;
  611. initVal: LONGINT;
  612. BEGIN
  613. (*
  614. HeapBlockPtr -- bootHeapAdr
  615. 4 Type
  616. 8 Mark
  617. 12 DataAdr
  618. 16 Size
  619. 20 HeapBlockPtr
  620. 24 Type
  621. 28 next -- MemoryBlock
  622. 32 startAdr
  623. 36 size
  624. 40 beginBlockAdr
  625. 44 endBlockAdr
  626. 48 --beginBlockAdr
  627. ....
  628. --endBlockAdr
  629. *)
  630. size := 1;
  631. memDescSize := MemBlockHeaderSize + SIZEOF(MemoryBlockDesc);
  632. INC(memDescSize, (-memDescSize) MOD StaticBlockSize); (* round up to multiple of StaticBlockSize *)
  633. INC(size, (-size) MOD StaticBlockSize); (* round up to multiple of StaticBlockSize *)
  634. memBlkSize := memDescSize + size + StaticBlockSize; (* add StaticBlockSize to account for alignments different from multiples of StaticBlockSize *)
  635. IF memBlkSize < MemBlockSize THEN memBlkSize := MemBlockSize END; (* MemBlockSize implicitly multiple of StaticBlockSize *)
  636. initVal := 8*1024*1024;
  637. adr := Kernel32.VirtualAlloc(initVal, memBlkSize, SetToDW({Kernel32.MEMCommit, Kernel32.MEMReserve}), SetToDW({Kernel32.PageExecuteReadWrite}));
  638. IF adr = NilVal THEN (* allocation failed *)
  639. adr := Kernel32.VirtualAlloc(NilVal, memBlkSize, SetToDW({Kernel32.MEMCommit}), SetToDW({Kernel32.PageExecuteReadWrite}));
  640. END;
  641. Trace.String("first heap block intVal "); Trace.Int(initVal,1); Trace.Ln;
  642. Trace.String("first heap block memBlkSize "); Trace.Int(memBlkSize,1); Trace.Ln;
  643. Trace.String("first heap block adr "); Trace.Int(adr,1); Trace.Ln;
  644. ASSERT(adr # 0);
  645. alignOffset := (-adr) MOD StaticBlockSize;
  646. memHeaderAdr := adr + alignOffset; (* force alignment of memory block start *)
  647. memBlockAdr := memHeaderAdr + MemBlockHeaderSize;
  648. memBlock := SYSTEM.VAL(MemoryBlock, memBlockAdr);
  649. beginBlockAdr := memHeaderAdr + memDescSize;
  650. memBlock.next := NIL;
  651. memBlock.startAdr := adr;
  652. memBlock.size := memBlkSize;
  653. beginBlockAdr := memHeaderAdr + memDescSize;
  654. endBlockAdr := adr + memBlkSize - alignOffset;
  655. memBlock.beginBlockAdr := beginBlockAdr;
  656. memBlock.endBlockAdr := endBlockAdr;
  657. (* correct fields *)
  658. SYSTEM.PUT(memBlockAdr + HeapBlockOffset, memHeaderAdr + BlockHeaderSize); (* set reference to header part of memory block correctly *)
  659. SYSTEM.PUT(memBlockAdr + TypeDescOffset, 0); (* set type descriptor field of memory block to default value, memory blocks are not traced by GC *)
  660. SYSTEM.PUT(memHeaderAdr + BlockHeaderSize + DataAdrOffset, memBlockAdr); (* set dataAdr of RecordBlockDesc to correct value *)
  661. SYSTEM.PUT(memHeaderAdr + BlockHeaderSize + 2*AddressSize , memBlkSize);
  662. (* fill first heap block *)
  663. SYSTEM.PUT(beginBlockAdr,0);
  664. SYSTEM.PUT(beginBlockAdr+AddressSize,0);
  665. SYSTEM.PUT(beginBlockAdr+2*AddressSize,0);
  666. (* change this when Heaps.HeapBlock is modified *)
  667. SYSTEM.PUT(beginBlockAdr+3*AddressSize,beginBlockAdr+7*AddressSize);
  668. SYSTEM.PUT(beginBlockAdr+4*AddressSize,endBlockAdr-beginBlockAdr);
  669. SYSTEM.PUT(beginBlockAdr+5*AddressSize,beginBlockAdr+2*AddressSize);
  670. SYSTEM.PUT(beginBlockAdr+6*AddressSize,0);
  671. memoryBlock := memBlock;
  672. END InitHeap;
  673. (** Get first memory block and first free address, the first free address is identical to memBlockHead.endBlockAdr *)
  674. PROCEDURE GetStaticHeap*(VAR beginBlockAdr, endBlockAdr, freeBlockAdr: ADDRESS);
  675. VAR memBlockAdr: ADDRESS;
  676. BEGIN
  677. InitHeap(memBlockHead,beginBlockAdr, endBlockAdr);
  678. memBlockTail := memBlockHead;
  679. (*
  680. SYSTEM.GET(bootHeapAdr + EndBlockOfs, freeBlockAdr);
  681. ASSERT(freeBlockAdr MOD StaticBlockSize = 0);
  682. memBlockAdr := bootHeapAdr + HeaderSize + MemoryBlockOfs;
  683. memBlockHead := SYSTEM.VAL(MemoryBlock, memBlockAdr); (* this block will never be freed since there is a global reference (initBlock in Heaps.Mod) to it *)
  684. memBlockHead.startAdr := bootHeapAdr;
  685. memBlockHead.size := bootHeapSize;
  686. ASSERT(memBlockHead.beginBlockAdr MOD StaticBlockSize = 0);
  687. ASSERT((memBlockHead.endBlockAdr - memBlockHead.beginBlockAdr) MOD StaticBlockSize = 0);
  688. memBlockTail := memBlockHead;
  689. *)
  690. beginBlockAdr := memBlockHead.beginBlockAdr;
  691. endBlockAdr := memBlockHead.endBlockAdr;
  692. freeBlockAdr := beginBlockAdr;
  693. END GetStaticHeap;
  694. (* returns if an address is a currently allocated heap address *)
  695. PROCEDURE ValidHeapAddress*(p: ADDRESS): BOOLEAN;
  696. BEGIN
  697. RETURN (p >= memBlockHead.beginBlockAdr) & (p <= memBlockTail.endBlockAdr)
  698. OR (p>=401000H) & (p<=ADDRESSOF(LastAddress))
  699. END ValidHeapAddress;
  700. PROCEDURE GetFreeK* (VAR total, lowFree, highFree: SIZE);
  701. VAR
  702. stat: Kernel32.MemoryStatusEx;
  703. BEGIN
  704. total := MAX(LONGINT); lowFree := 0; highFree := total;
  705. (*<< added by Alexey *)
  706. IF GetMemStatus(stat) THEN
  707. total := SHORT(stat.ullTotalVirtual DIV 1024);
  708. lowFree := 0;
  709. highFree := SHORT(stat.ullAvailVirtual DIV 1024);
  710. END;
  711. (* added by Alexey >>*)
  712. END GetFreeK;
  713. (* ug *)
  714. PROCEDURE TraceMemBlocks*;
  715. VAR memBlock {UNTRACED}: MemoryBlock; i : LONGINT;
  716. BEGIN
  717. memBlock := memBlockHead;
  718. i := 0;
  719. WHILE memBlock # NIL DO
  720. Trace.String("block "); Trace.Int(i, 0); Trace.String(": startAdr = "); Trace.Hex(memBlock.startAdr, 0);
  721. Trace.String(" size = "); Trace.Hex(memBlock.size, 0);
  722. Trace.String(" beginBlockAdr = "); Trace.Hex(memBlock.beginBlockAdr, 0);
  723. Trace.String(" endBlockAdr = "); Trace.Hex(memBlock.endBlockAdr, 0); Trace.Ln;
  724. memBlock := memBlock.next;
  725. INC(i)
  726. END
  727. END TraceMemBlocks;
  728. (* insert given memory block in sorted list of memory blocks, sort key is startAdr field - called during GC *)
  729. PROCEDURE InsertMemoryBlock(memBlock: MemoryBlock);
  730. VAR cur {UNTRACED}, prev {UNTRACED}: MemoryBlock;
  731. BEGIN
  732. cur := memBlockHead;
  733. prev := NIL;
  734. WHILE (cur # NIL) & (cur.startAdr < memBlock.startAdr) DO
  735. prev := cur;
  736. cur := cur.next
  737. END;
  738. IF prev = NIL THEN (* insert at head of list *)
  739. memBlock.next := memBlockHead;
  740. memBlockHead := memBlock
  741. ELSE (* insert in middle or at end of list *)
  742. memBlock.next := cur;
  743. prev.next := memBlock;
  744. IF cur = NIL THEN
  745. memBlockTail := memBlock
  746. END
  747. END
  748. END InsertMemoryBlock;
  749. PROCEDURE SetToDW(set: SET): Kernel32.DWORD;
  750. BEGIN
  751. RETURN SYSTEM.VAL(Kernel32.DWORD,set);
  752. END SetToDW;
  753. (* expand heap by allocating a new memory block - called during GC *)
  754. PROCEDURE ExpandHeap*(dummy: LONGINT; size: SIZE; VAR memoryBlock: MemoryBlock; VAR beginBlockAdr, endBlockAdr: ADDRESS);
  755. CONST MemBlockHeaderSize = BlockHeaderSize + RecordDescSize + BlockHeaderSize;
  756. TypeDescOffset = -AddressSize; (* see Heaps.Mod *)
  757. HeapBlockOffset = - 2 * AddressSize; (* see Heaps.Mod *)
  758. DataAdrOffset = AddressSize; (* offset of dataAdr field in Heaps.HeapBlockDesc *)
  759. VAR memDescSize, memBlkSize, alignOffset: SIZE; adr, memHeaderAdr, memBlockAdr, memBlockHeadAdr: ADDRESS;
  760. memBlock {UNTRACED}: MemoryBlock; i: LONGINT; ch: CHAR; h: HUGEINT; initVal: SIZE;
  761. continue: BOOLEAN;
  762. BEGIN
  763. memDescSize := MemBlockHeaderSize + SIZEOF(MemoryBlockDesc);
  764. INC(memDescSize, (-memDescSize) MOD StaticBlockSize); (* round up to multiple of StaticBlockSize *)
  765. INC(size, (-size) MOD StaticBlockSize); (* round up to multiple of StaticBlockSize *)
  766. memBlkSize := memDescSize + size + StaticBlockSize; (* add StaticBlockSize to account for alignments different from multiples of StaticBlockSize *)
  767. INC(memBlkSize, (-memBlkSize) MOD MemBlockSize);
  768. initVal := memBlockTail.startAdr + memBlockTail.size;
  769. adr := Kernel32.VirtualAlloc(initVal, memBlkSize, SetToDW({Kernel32.MEMCommit, Kernel32.MEMReserve}), SetToDW({Kernel32.PageExecuteReadWrite}));
  770. IF adr = NilVal THEN (* allocation failed *)
  771. adr := Kernel32.VirtualAlloc(NilVal, memBlkSize, SetToDW({Kernel32.MEMCommit}), SetToDW({Kernel32.PageExecuteReadWrite}));
  772. END;
  773. continue := adr = initVal;
  774. Trace.String("expand heap block intVal "); Trace.Int(initVal,1); Trace.Ln;
  775. Trace.String("expand heap block memBlkSize "); Trace.Int(memBlkSize,1); Trace.Ln;
  776. Trace.String("expand heap block adr "); Trace.Int(adr,1); Trace.Ln;
  777. ASSERT(adr # 0);
  778. IF adr # 0 THEN
  779. alignOffset := (-adr) MOD StaticBlockSize;
  780. IF continue THEN
  781. memBlock := memBlockTail;
  782. memBlock.size := memBlock.size + memBlkSize;
  783. beginBlockAdr := memBlockTail.endBlockAdr;
  784. endBlockAdr := beginBlockAdr;
  785. INC(endBlockAdr, memBlkSize);
  786. ELSE
  787. memHeaderAdr := adr + alignOffset; (* force alignment of memory block start *)
  788. memBlockAdr := memHeaderAdr + MemBlockHeaderSize;
  789. memBlock := SYSTEM.VAL(MemoryBlock, memBlockAdr);
  790. memBlock.next := NIL;
  791. memBlock.startAdr := adr;
  792. memBlock.size := memBlkSize;
  793. beginBlockAdr := memHeaderAdr + memDescSize;
  794. endBlockAdr := adr + memBlkSize - alignOffset;
  795. memBlock.beginBlockAdr := beginBlockAdr;
  796. memBlock.endBlockAdr := beginBlockAdr;
  797. (* upon memory block insertion memBlock.beginBlockAdr = memBlock.endBlockAdr to denote that the memory block has no valid heap blocks yet
  798. - necessary for real-time GC. Memory block end address is set by caller by using SetMemBlockEndAddress after fitting free block in. *)
  799. (* copy header of memBlockHead to header of memBlock - byte by byte *)
  800. memBlockHeadAdr := SYSTEM.VAL(ADDRESS, memBlockHead);
  801. FOR i := 0 TO MemBlockHeaderSize - 1 DO
  802. SYSTEM.GET(memBlockHeadAdr - MemBlockHeaderSize + i, ch);
  803. SYSTEM.PUT(memBlockAdr - MemBlockHeaderSize + i, ch)
  804. END;
  805. (* correct fields *)
  806. SYSTEM.PUT(memBlockAdr + HeapBlockOffset, memHeaderAdr + BlockHeaderSize); (* set reference to header part of memory block correctly *)
  807. SYSTEM.PUT(memBlockAdr + TypeDescOffset, 0); (* set type descriptor field of memory block to default value, memory blocks are not traced by GC *)
  808. SYSTEM.PUT(memHeaderAdr + BlockHeaderSize + DataAdrOffset, memBlockAdr); (* set dataAdr of RecordBlockDesc to correct value *)
  809. InsertMemoryBlock(memBlock);
  810. END;
  811. memoryBlock := memBlock;
  812. ELSE
  813. beginBlockAdr := 0; endBlockAdr := 0;
  814. END;
  815. END ExpandHeap;
  816. (* Set memory block end address *)
  817. PROCEDURE SetMemoryBlockEndAddress*(memBlock: MemoryBlock; endBlockAdr: ADDRESS);
  818. BEGIN
  819. ASSERT(endBlockAdr >= memBlock.beginBlockAdr);
  820. memBlock.endBlockAdr := endBlockAdr
  821. END SetMemoryBlockEndAddress;
  822. (* Free unused memory block - called during GC *)
  823. PROCEDURE FreeMemBlock*(memBlock: MemoryBlock);
  824. VAR cur {UNTRACED}, prev {UNTRACED}: MemoryBlock;
  825. startAdr: ADDRESS;
  826. BEGIN
  827. cur := memBlockHead;
  828. prev := NIL;
  829. WHILE (cur # NIL) & (cur # memBlock) DO
  830. prev := cur;
  831. cur := cur.next
  832. END;
  833. IF cur = memBlock THEN
  834. IF prev = NIL THEN
  835. memBlockHead := cur.next;
  836. ELSE
  837. prev.next := cur.next;
  838. IF prev.next = NIL THEN
  839. memBlockTail := prev
  840. END
  841. END;
  842. memBlock.next := NIL;
  843. startAdr := memBlock.startAdr; (* this value must be cached for the second call of Kernel32.VirtualFree *)
  844. Kernel32.VirtualFree(SYSTEM.VAL(LONGINT, memBlock.startAdr), memBlock.size, SetToDW({Kernel32.MEMDecommit}));
  845. Kernel32.VirtualFree(SYSTEM.VAL(LONGINT, startAdr ), 0, SetToDW({Kernel32.MEMRelease}));
  846. ELSE
  847. HALT(535) (* error in memory block management *)
  848. END;
  849. END FreeMemBlock;
  850. PROCEDURE PhysicalAdr*(adr: ADDRESS; size: SIZE): ADDRESS;
  851. END PhysicalAdr;
  852. (* function returning the number of processors that are available to Aos *)
  853. PROCEDURE NumberOfProcessors*( ): LONGINT;
  854. VAR info: Kernel32.SystemInfo;
  855. BEGIN
  856. Kernel32.GetSystemInfo( info );
  857. RETURN info.dwNumberOfProcessors
  858. END NumberOfProcessors;
  859. (* function for changing byte order *)
  860. PROCEDURE ChangeByteOrder* (n: LONGINT): LONGINT;
  861. CODE { SYSTEM.Pentium }
  862. MOV EAX, [EBP+n] ; load n in eax
  863. BSWAP EAX ; swap byte order
  864. END ChangeByteOrder;
  865. PROCEDURE TraceColor (c: SHORTINT);
  866. END TraceColor;
  867. PROCEDURE LogChar (c: CHAR);
  868. VAR trace: ARRAY 2 OF CHAR;
  869. BEGIN trace[0] := c; trace[1] := 0X; Kernel32.OutputString (trace);
  870. END LogChar;
  871. PROCEDURE -GetRAX*(): HUGEINT;
  872. CODE{SYSTEM.AMD64}
  873. END GetRAX;
  874. PROCEDURE -GetRCX*(): HUGEINT;
  875. CODE{SYSTEM.AMD64}
  876. MOV RAX,RCX
  877. END GetRCX;
  878. PROCEDURE -GetRSI*(): HUGEINT;
  879. CODE{SYSTEM.AMD64}
  880. MOV RAX,RSI
  881. END GetRSI;
  882. PROCEDURE -GetRDI*(): HUGEINT;
  883. CODE{SYSTEM.AMD64}
  884. MOV RAX,RDI
  885. END GetRDI;
  886. PROCEDURE -SetRAX*(n: HUGEINT);
  887. CODE{SYSTEM.AMD64}
  888. NOP
  889. POP RAX
  890. END SetRAX;
  891. PROCEDURE -SetRBX*(n: HUGEINT);
  892. CODE{SYSTEM.AMD64}
  893. NOP
  894. POP RBX
  895. END SetRBX;
  896. PROCEDURE -SetRCX*(n: HUGEINT);
  897. CODE{SYSTEM.AMD64}
  898. POP RCX
  899. END SetRCX;
  900. PROCEDURE -SetRDX*(n: HUGEINT);
  901. CODE{SYSTEM.AMD64}
  902. POP RDX
  903. END SetRDX;
  904. PROCEDURE -SetRSI*(n: HUGEINT);
  905. CODE{SYSTEM.AMD64}
  906. POP RSI
  907. END SetRSI;
  908. PROCEDURE -SetRDI*(n: HUGEINT);
  909. CODE{SYSTEM.AMD64}
  910. POP RDI
  911. END SetRDI;
  912. PROCEDURE Portin8*(port: LONGINT; VAR val: CHAR);
  913. CODE{SYSTEM.i386}
  914. MOV EDX,[EBP+port]
  915. IN AL, DX
  916. MOV ECX, [EBP+val]
  917. MOV [ECX], AL
  918. END Portin8;
  919. PROCEDURE Portin16*(port: LONGINT; VAR val: INTEGER);
  920. CODE{SYSTEM.i386}
  921. MOV EDX,[EBP+port]
  922. IN AX, DX
  923. MOV ECX, [EBP+val]
  924. MOV [ECX], AX
  925. END Portin16;
  926. PROCEDURE Portin32*(port: LONGINT; VAR val: LONGINT);
  927. CODE{SYSTEM.i386}
  928. MOV EDX,[EBP+port]
  929. IN EAX, DX
  930. MOV ECX, [EBP+val]
  931. MOV [ECX], EAX
  932. END Portin32;
  933. PROCEDURE Portout8*(port: LONGINT; val: CHAR);
  934. CODE{SYSTEM.i386}
  935. MOV AL,[EBP+val]
  936. MOV EDX,[EBP+port]
  937. OUT DX,AL
  938. END Portout8;
  939. PROCEDURE Portout16*(port: LONGINT; val: INTEGER);
  940. CODE{SYSTEM.i386}
  941. MOV AX,[EBP+val]
  942. MOV EDX,[EBP+port]
  943. OUT DX,AX
  944. END Portout16;
  945. PROCEDURE Portout32*(port: LONGINT; val: LONGINT);
  946. CODE{SYSTEM.i386}
  947. MOV EAX,[EBP+val]
  948. MOV EDX,[EBP+port]
  949. OUT DX,EAX
  950. END Portout32;
  951. BEGIN
  952. IF ~Kernel32.Generic THEN
  953. Init
  954. END;
  955. Trace.String("Machine (64 bit)"); Trace.Ln;
  956. END Machine.