Win32.Machine.Mod 32 KB

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