Windows.I386.Machine.Mod 25 KB

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