Unix.Machine.Mod 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966
  1. MODULE Machine; (** AUTHOR "pjm, G.F."; PURPOSE "Bootstrapping, configuration and machine interface"; *)
  2. IMPORT S := SYSTEM, Trace, Unix, Glue;
  3. CONST
  4. DefaultConfig = "Color 0 StackSize 128";
  5. #IF I386 THEN
  6. Version = "A2 Gen. 32-bit, ";
  7. DefaultObjectFileExtension* = ".GofU";
  8. #ELSIF AMD64 THEN
  9. Version = "A2 Gen. 64-bit, ";
  10. DefaultObjectFileExtension* = ".GofUu";
  11. #END
  12. Second* = 1000; (* frequency of ticks increments in Hz *)
  13. (** bits in features variable *)
  14. MTTR* = 12; MMX* = 23;
  15. AddressSize = SIZEOF(ADDRESS);
  16. StaticBlockSize = 8 * AddressSize; (* static heap block size *)
  17. MemBlockSize* = 64*1024*1024;
  18. TraceOutput* = 0; (* Trace output *)
  19. Memory* = 1; (*! Virtual memory management, stack and page allocation, not used in UnixAos *)
  20. Heaps* = 2; (* Storage allocation and Garbage collection *)
  21. Interrupts* = 3; (*! Interrupt handling, not used in UnixAos *)
  22. Modules* = 4; (* Module list *)
  23. Objects* = 5; (*! Ready queue, not used in UnixAos *)
  24. Processors* = 6; (*! Interprocessor interrupts, not used in UnixAos *)
  25. KernelLog* = 7; (* Atomic output *)
  26. X11* = 8; (* XWindows I/O *)
  27. MaxLocks* = 9; (* { <= 32 } *)
  28. MaxCPU* = 4;
  29. IsCooperative* = FALSE;
  30. TYPE
  31. Vendor* = ARRAY 13 OF CHAR;
  32. MemoryBlock* = POINTER {UNSAFE, UNTRACED} TO MemoryBlockDesc;
  33. MemoryBlockDesc* = RECORD
  34. next- : MemoryBlock;
  35. startAdr-: ADDRESS; (* sort key in linked list of memory blocks *)
  36. size-: SIZE;
  37. beginBlockAdr-, endBlockAdr-: ADDRESS
  38. END;
  39. (** processor state *)
  40. State* = RECORD
  41. PC*, BP*, SP*: ADDRESS
  42. END;
  43. VAR
  44. mtx : ARRAY MaxLocks OF Unix.Mutex_t;
  45. version-: ARRAY 64 OF CHAR; (** Aos version *)
  46. features-, features2 : SET;
  47. MMXSupport- : BOOLEAN;
  48. SSESupport- : BOOLEAN;
  49. SSE2Support- : BOOLEAN;
  50. SSE3Support- : BOOLEAN;
  51. SSSE3Support- : BOOLEAN;
  52. SSE41Support- : BOOLEAN;
  53. SSE42Support- : BOOLEAN;
  54. SSE5Support- : BOOLEAN;
  55. AVXSupport- : BOOLEAN;
  56. ticks-: LONGINT; (** timer ticks. Use Kernel.GetTicks() to read, don't write *)
  57. prioLow-, prioHigh-: LONGINT; (* permitted thread priorities *)
  58. fcr-: SET; (** default floating-point control register value (default rounding mode is towards -infinity, for ENTIER) *)
  59. mhz-: HUGEINT; (** clock rate of GetTimer in MHz, or 0 if not known *)
  60. gcThreshold-: SIZE;
  61. memBlockHead-{UNTRACED}, memBlockTail-{UNTRACED}: MemoryBlock; (* head and tail of sorted list of memory blocks *)
  62. config: ARRAY 2048 OF CHAR; (* config strings *)
  63. logname: ARRAY 32 OF CHAR;
  64. logfile: LONGINT;
  65. traceHeap: BOOLEAN;
  66. timer0 : HUGEINT;
  67. (** Return current processor ID (0 to MaxNum-1). *)
  68. PROCEDURE ID* (): LONGINT;
  69. BEGIN
  70. RETURN 0
  71. END ID;
  72. (**
  73. * Flush Data Cache for the specified virtual address range. If len is negative, flushes the whole cache.
  74. * This is used on some architecture to interact with DMA hardware (e.g. Ethernet and USB. It can be
  75. * left empty on Intel architecture.
  76. *)
  77. PROCEDURE FlushDCacheRange * (adr: ADDRESS; len: LONGINT);
  78. END FlushDCacheRange;
  79. (**
  80. * Invalidate Data Cache for the specified virtual address range. If len is negative, flushes the whole cache.
  81. * This is used on some architecture to interact with DMA hardware (e.g. Ethernet and USB. It can be
  82. * left empty on Intel architecture.
  83. *)
  84. PROCEDURE InvalidateDCacheRange * (adr: ADDRESS; len: LONGINT);
  85. END InvalidateDCacheRange;
  86. (**
  87. * Invalidate Instruction Cache for the specified virtual address range. If len is negative, flushes the whole cache.
  88. * This is used on some architecture to interact with DMA hardware (e.g. Ethernet and USB. It can be
  89. * left empty on Intel architecture.
  90. *)
  91. PROCEDURE InvalidateICacheRange * (adr: ADDRESS; len: LONGINT);
  92. END InvalidateICacheRange;
  93. (* insert given memory block in sorted list of memory blocks, sort key is startAdr field - called during GC *)
  94. PROCEDURE InsertMemoryBlock(memBlock: MemoryBlock);
  95. VAR cur {UNTRACED}, prev {UNTRACED}: MemoryBlock;
  96. BEGIN
  97. cur := memBlockHead;
  98. prev := NIL;
  99. WHILE (cur # NIL) & (ADDRESS OF cur^ < ADDRESS OF memBlock^) DO
  100. prev := cur;
  101. cur := cur.next
  102. END;
  103. IF prev = NIL THEN (* insert at head of list *)
  104. memBlock.next := memBlockHead;
  105. memBlockHead := memBlock
  106. ELSE (* insert in middle or at end of list *)
  107. prev.next := memBlock;
  108. memBlock.next := cur;
  109. END;
  110. IF cur = NIL THEN
  111. memBlockTail := memBlock
  112. END
  113. END InsertMemoryBlock;
  114. (* Free unused memory block - called during GC *)
  115. PROCEDURE FreeMemBlock*(memBlock: MemoryBlock);
  116. VAR cur {UNTRACED}, prev {UNTRACED}: MemoryBlock;
  117. BEGIN
  118. cur := memBlockHead;
  119. prev := NIL;
  120. WHILE (cur # NIL) & (cur # memBlock) DO
  121. prev := cur;
  122. cur := cur.next
  123. END;
  124. IF cur = memBlock THEN
  125. IF traceHeap THEN
  126. Trace.String( "Release memory block " ); Trace.Hex( memBlock.startAdr, -8 ); Trace.Ln
  127. END;
  128. IF prev = NIL THEN
  129. memBlockHead := cur.next
  130. ELSE
  131. prev.next := cur.next;
  132. IF cur.next = NIL THEN
  133. memBlockTail := prev
  134. END
  135. END;
  136. Unix.free( memBlock.startAdr )
  137. ELSE
  138. HALT(535) (* error in memory block management *)
  139. END;
  140. END FreeMemBlock;
  141. (* expand heap by allocating a new memory block *)
  142. PROCEDURE ExpandHeap*( dummy: LONGINT; size: SIZE; VAR memoryBlock: MemoryBlock; VAR beginBlockAdr, endBlockAdr: ADDRESS );
  143. VAR mBlock: MemoryBlock; alloc: SIZE; adr: ADDRESS;
  144. BEGIN
  145. ASSERT(SIZEOF(MemoryBlockDesc) <= StaticBlockSize); (* make sure MemoryBlock contents fits into one StaticBlock *)
  146. alloc := size + StaticBlockSize;
  147. IF alloc < MemBlockSize THEN alloc := MemBlockSize END;
  148. ASSERT((Unix.PageSize > StaticBlockSize) & (Unix.PageSize MOD StaticBlockSize = 0)); (* alignment to Unix.PageSize implies alignment to StaticBlockSize *)
  149. INC( alloc, (-alloc) MOD Unix.PageSize );
  150. IF Unix.posix_memalign( adr, Unix.PageSize, alloc ) # 0 THEN
  151. Unix.Perror( "Machine.ExpandHeap: posix_memalign" );
  152. beginBlockAdr := 0;
  153. endBlockAdr := 0;
  154. memoryBlock := NIL;
  155. ELSE
  156. IF Unix.mprotect( adr, alloc, 7 (* READ WRITE EXEC *) ) # 0 THEN
  157. Unix.Perror( "Machine.ExpandHeap: mprotect" )
  158. END;
  159. mBlock := adr;
  160. mBlock.next := NIL;
  161. mBlock.startAdr := adr;
  162. mBlock.size := alloc;
  163. beginBlockAdr := adr + StaticBlockSize;
  164. endBlockAdr := beginBlockAdr + alloc - StaticBlockSize;
  165. mBlock.beginBlockAdr := beginBlockAdr;
  166. mBlock.endBlockAdr := beginBlockAdr; (* block is still empty -- Heaps module will set the upper bound *)
  167. InsertMemoryBlock( mBlock );
  168. IF traceHeap THEN TraceHeap( mBlock ) END;
  169. memoryBlock := mBlock;
  170. END
  171. END ExpandHeap;
  172. (* Set memory block end address *)
  173. PROCEDURE SetMemoryBlockEndAddress*(memBlock: MemoryBlock; endBlockAdr: ADDRESS);
  174. BEGIN
  175. ASSERT(endBlockAdr >= memBlock.beginBlockAdr);
  176. memBlock.endBlockAdr := endBlockAdr
  177. END SetMemoryBlockEndAddress;
  178. PROCEDURE TraceHeap( new: MemoryBlock );
  179. VAR cur{UNTRACED}: MemoryBlock;
  180. BEGIN
  181. Trace.Ln;
  182. Trace.String( "Heap expanded" ); Trace.Ln;
  183. Trace.String("Static Heap: "); Trace.Hex(Glue.baseAdr, -8); Trace.String(" - "); Trace.Hex(Glue.endAdr, -8);
  184. Trace.Ln;
  185. cur := memBlockHead;
  186. WHILE cur # NIL DO
  187. Trace.Hex( cur.startAdr, -8 ); Trace.String( " " ); Trace.Hex( cur.startAdr + cur.size, -8 );
  188. IF cur = new THEN Trace.String( " (new)" ) END;
  189. Trace.Ln;
  190. cur := cur.next
  191. END
  192. END TraceHeap;
  193. (** Get first memory block and first free address, the first free address is identical to memBlockHead.endBlockAdr *)
  194. PROCEDURE GetStaticHeap*(VAR beginBlockAdr, endBlockAdr, freeBlockAdr: ADDRESS);
  195. BEGIN
  196. beginBlockAdr := NIL; endBlockAdr := NIL; freeBlockAdr := NIL;
  197. END GetStaticHeap;
  198. (* returns if an address is a currently allocated heap address *)
  199. PROCEDURE ValidHeapAddress*( p: ADDRESS ): BOOLEAN;
  200. VAR mb: MemoryBlock;
  201. BEGIN
  202. IF (p>=Glue.baseAdr) & (p<=Glue.endAdr) THEN RETURN TRUE END;
  203. mb := memBlockHead;
  204. WHILE mb # NIL DO
  205. IF (p >= mb.beginBlockAdr) & (p <= mb.endBlockAdr) THEN RETURN TRUE END;
  206. mb := mb.next;
  207. END;
  208. RETURN FALSE;
  209. END ValidHeapAddress;
  210. (** Return information on free memory in Kbytes. *)
  211. PROCEDURE GetFreeK*(VAR total, lowFree, highFree: SIZE);
  212. BEGIN
  213. (*! meaningless in Unix port, for interface compatibility only *)
  214. total := 0;
  215. lowFree := 0;
  216. highFree := 0
  217. END GetFreeK;
  218. (** Fill "size" bytes at "destAdr" with "filler". "size" must be multiple of 4. *)
  219. PROCEDURE Fill32* (destAdr: ADDRESS; size: SIZE; filler: LONGINT);
  220. CODE
  221. #IF I386 THEN
  222. MOV EDI, [EBP+destAdr]
  223. MOV ECX, [EBP+size]
  224. MOV EAX, [EBP+filler]
  225. TEST ECX, 3
  226. JZ ok
  227. PUSH 8 ; ASSERT failure
  228. INT 3
  229. ok:
  230. SHR ECX, 2
  231. CLD
  232. REP STOSD
  233. #ELSIF AMD64 THEN
  234. MOV RDI, [RBP + destAdr]
  235. MOV RCX, [RBP + size]
  236. MOV EAX, [RBP + filler]
  237. TEST RCX, 3
  238. JZ ok
  239. PUSH 8 ; ASSERT failure
  240. INT 3
  241. ok:
  242. SHR RCX, 2
  243. CLD
  244. REP STOSD
  245. #ELSE
  246. unimpemented
  247. #END
  248. END Fill32;
  249. PROCEDURE Portin8*(port: LONGINT; VAR val: CHAR);
  250. END Portin8;
  251. PROCEDURE Portin16*(port: LONGINT; VAR val: INTEGER);
  252. END Portin16;
  253. PROCEDURE Portin32*(port: LONGINT; VAR val: LONGINT);
  254. END Portin32;
  255. PROCEDURE Portout8*(port: LONGINT; val: CHAR);
  256. END Portout8;
  257. PROCEDURE Portout16*(port: LONGINT; val: INTEGER);
  258. END Portout16;
  259. PROCEDURE Portout32*(port: LONGINT; val: LONGINT);
  260. END Portout32;
  261. (** -- Atomic operations -- *)
  262. (** Atomic INC(x). *)
  263. PROCEDURE -AtomicInc*(VAR x: LONGINT);
  264. CODE
  265. #IF I386 THEN
  266. POP EAX
  267. LOCK
  268. INC DWORD [EAX]
  269. #ELSIF AMD64 THEN
  270. POP RAX
  271. LOCK
  272. INC DWORD [RAX]
  273. #ELSE
  274. unimplemented
  275. #END
  276. END AtomicInc;
  277. (** Atomic DEC(x). *)
  278. PROCEDURE -AtomicDec*(VAR x: LONGINT);
  279. CODE
  280. #IF I386 THEN
  281. POP EAX
  282. LOCK
  283. DEC DWORD [EAX]
  284. #ELSIF AMD64 THEN
  285. POP RAX
  286. LOCK
  287. DEC DWORD [RAX]
  288. #ELSE
  289. unimplemented
  290. #END
  291. END AtomicDec;
  292. (** Atomic INC(x, y). *)
  293. PROCEDURE -AtomicAdd*(VAR x: LONGINT; y: LONGINT);
  294. CODE
  295. #IF I386 THEN
  296. POP EBX
  297. POP EAX
  298. LOCK
  299. ADD DWORD [EAX], EBX
  300. #ELSIF AMD64 THEN
  301. POP EBX
  302. POP RAX
  303. LOCK
  304. ADD DWORD [RAX], EBX
  305. #ELSE
  306. unimplemented
  307. #END
  308. END AtomicAdd;
  309. (** Atomic EXCL. *)
  310. PROCEDURE AtomicExcl* (VAR s: SET; bit: LONGINT);
  311. CODE
  312. #IF I386 THEN
  313. MOV EAX, [EBP+bit]
  314. MOV EBX, [EBP+s]
  315. LOCK
  316. BTR [EBX], EAX
  317. #ELSIF AMD64 THEN
  318. MOV EAX, [RBP + bit]
  319. MOV RBX, [RBP + s]
  320. LOCK
  321. BTR [RBX], EAX
  322. #ELSE
  323. unimplemented
  324. #END
  325. END AtomicExcl;
  326. (** Atomic test-and-set. Set x = TRUE and return old value of x. *)
  327. PROCEDURE -AtomicTestSet*(VAR x: BOOLEAN): BOOLEAN;
  328. CODE
  329. #IF I386 THEN
  330. POP EBX
  331. MOV AL, 1
  332. XCHG [EBX], AL
  333. #ELSIF AMD64 THEN
  334. POP RBX
  335. MOV AL, 1
  336. XCHG [RBX], AL
  337. #ELSE
  338. unimplemented
  339. #END
  340. END AtomicTestSet;
  341. (* Atomic compare-and-swap. Set x = new if x = old and return old value of x *)
  342. PROCEDURE -AtomicCAS* (VAR x: LONGINT; old, new: LONGINT): LONGINT;
  343. CODE
  344. #IF I386 THEN
  345. POP EBX ; new
  346. POP EAX ; old
  347. POP ECX ; address of x
  348. LOCK CMPXCHG [ECX], EBX; atomicly compare x with old and set it to new if equal
  349. #ELSIF AMD64 THEN
  350. POP EBX ; new
  351. POP EAX ; old
  352. POP RCX ; address of x
  353. LOCK CMPXCHG [RCX], EBX ; atomicly compare x with old and set it to new if equal
  354. #ELSE
  355. unimplemented
  356. #END
  357. END AtomicCAS;
  358. (* Return current instruction pointer *)
  359. PROCEDURE CurrentPC* (): ADDRESS;
  360. CODE
  361. #IF I386 THEN
  362. MOV EAX, [EBP+4]
  363. #ELSIF AMD64 THEN
  364. MOV RAX, [RBP + 8]
  365. #ELSE
  366. unimplemented
  367. #END
  368. END CurrentPC;
  369. (* Return current frame pointer *)
  370. PROCEDURE -CurrentBP* (): ADDRESS;
  371. CODE
  372. #IF I386 THEN
  373. MOV EAX, EBP
  374. #ELSIF AMD64 THEN
  375. MOV RAX, RBP
  376. #ELSE
  377. unimplemented
  378. #END
  379. END CurrentBP;
  380. (* Set current frame pointer *)
  381. PROCEDURE -SetBP* (bp: ADDRESS);
  382. CODE
  383. #IF I386 THEN
  384. POP EBP
  385. #ELSIF AMD64 THEN
  386. POP RBP
  387. #ELSE
  388. unimplemented
  389. #END
  390. END SetBP;
  391. (* Return current stack pointer *)
  392. PROCEDURE -CurrentSP* (): ADDRESS;
  393. CODE
  394. #IF I386 THEN
  395. MOV EAX, ESP
  396. #ELSIF AMD64 THEN
  397. MOV RAX, RSP
  398. #ELSE
  399. unimplemented
  400. #END
  401. END CurrentSP;
  402. (* Set current stack pointer *)
  403. PROCEDURE -SetSP* (sp: ADDRESS);
  404. CODE
  405. #IF I386 THEN
  406. POP ESP
  407. #ELSIF AMD64 THEN
  408. POP RSP
  409. #ELSE
  410. unimplemented
  411. #END
  412. END SetSP;
  413. PROCEDURE -GetTimer* (): HUGEINT;
  414. CODE
  415. #IF I386 THEN
  416. RDTSC ; set EDX:EAX
  417. #ELSIF AMD64 THEN
  418. XOR RAX, RAX
  419. RDTSC ; set EDX:EAX
  420. SHL RDX, 32
  421. OR RAX, RDX
  422. #END
  423. END GetTimer;
  424. (** -- Configuration and bootstrapping -- *)
  425. (** Return the value of the configuration string specified by parameter name in parameter val. Returns val = "" if the string was not found, or has an empty value. *)
  426. PROCEDURE GetConfig* (CONST name: ARRAY OF CHAR; VAR val: ARRAY OF CHAR);
  427. VAR i, src: LONGINT; ch: CHAR;
  428. BEGIN
  429. ASSERT (name[0] # "="); (* no longer supported, use GetInit instead *)
  430. src := -1; val := "";
  431. LOOP
  432. REPEAT
  433. INC( src ); ch := config[src];
  434. IF ch = 0X THEN EXIT END;
  435. UNTIL ch > ' ';
  436. i := 0;
  437. LOOP
  438. ch := config[src];
  439. IF (ch # name[i]) OR (name[i] = 0X) THEN EXIT END;
  440. INC (i); INC (src)
  441. END;
  442. IF (ch <= ' ') & (name[i] = 0X) THEN (* found *)
  443. i := 0;
  444. REPEAT
  445. INC (src); ch := config[src]; val[i] := ch; INC (i);
  446. IF i = LEN(val) THEN val[i - 1] := 0X; RETURN END (* val too short *)
  447. UNTIL ch <= ' ';
  448. IF ch = ' ' THEN val[i -1] := 0X END;
  449. RETURN
  450. ELSE
  451. WHILE ch > ' ' DO (* skip to end of name *)
  452. INC (src); ch := config[src]
  453. END;
  454. INC (src);
  455. REPEAT (* skip to end of value *)
  456. ch := config[src]; INC (src)
  457. UNTIL ch <= ' '
  458. END
  459. END;
  460. IF (name = "ObjectFileExtension") & (val = "") THEN
  461. val := DefaultObjectFileExtension
  462. END;
  463. END GetConfig;
  464. (** 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. *)
  465. PROCEDURE StrToInt* (VAR i: LONGINT; CONST s: ARRAY OF CHAR): LONGINT;
  466. VAR vd, vh, sgn, d: LONGINT; hex: BOOLEAN;
  467. BEGIN
  468. vd := 0; vh := 0; hex := FALSE;
  469. IF s[i] = "-" THEN sgn := -1; INC (i) ELSE sgn := 1 END;
  470. LOOP
  471. IF (s[i] >= "0") & (s[i] <= "9") THEN d := ORD (s[i])-ORD ("0")
  472. ELSIF (CAP (s[i]) >= "A") & (CAP (s[i]) <= "F") THEN d := ORD (CAP (s[i]))-ORD ("A") + 10; hex := TRUE
  473. ELSE EXIT
  474. END;
  475. vd := 10*vd + d; vh := 16*vh + d;
  476. INC (i)
  477. END;
  478. IF CAP (s[i]) = "H" THEN hex := TRUE; INC (i) END; (* optional H *)
  479. IF hex THEN vd := vh END;
  480. RETURN sgn * vd
  481. END StrToInt;
  482. (* function returning the number of processors that are available to Aos *)
  483. PROCEDURE NumberOfProcessors*( ): LONGINT;
  484. VAR res: LONGINT;
  485. BEGIN
  486. res := Unix.getnprocs();
  487. RETURN res;
  488. END NumberOfProcessors;
  489. (*! non portable code, for native Aos only *)
  490. PROCEDURE SetNumberOfProcessors*( num: LONGINT );
  491. BEGIN
  492. (* numberOfProcessors := num; *)
  493. END SetNumberOfProcessors;
  494. (* function for changing byte order *)
  495. PROCEDURE ChangeByteOrder* (n: LONGINT): LONGINT;
  496. CODE
  497. #IF I386 THEN
  498. MOV EAX, [EBP+n] ; load n in eax
  499. BSWAP EAX ; swap byte order
  500. #ELSIF AMD64 THEN
  501. MOV EAX, [RBP+n] ; load n in eax
  502. BSWAP EAX ; swap byte order
  503. #ELSE
  504. unimplemented
  505. #END
  506. END ChangeByteOrder;
  507. (* Send and print character *)
  508. PROCEDURE TraceChar *(c: CHAR);
  509. BEGIN
  510. Trace.Char( c )
  511. END TraceChar;
  512. (** CPU identification *)
  513. PROCEDURE CPUID*( VAR vendor: Vendor; VAR version: LONGINT; VAR features1,features2: SET );
  514. CODE
  515. #IF I386 THEN
  516. PUSH ECX
  517. MOV EAX, 0
  518. CPUID
  519. CMP EAX, 0
  520. JNE ok
  521. MOV ESI, [EBP+vendor]
  522. MOV [ESI], AL ; AL = 0
  523. MOV ESI, [EBP+version]
  524. MOV [ESI], EAX ; EAX = 0
  525. MOV ESI, [EBP+features1]
  526. MOV [ESI], EAX
  527. MOV ESI, [EBP+features2]
  528. MOV [ESI], EAX
  529. JMP end
  530. ok:
  531. MOV ESI, [EBP+vendor]
  532. MOV [ESI], EBX
  533. MOV [ESI+4], EDX
  534. MOV [ESI+8], ECX
  535. MOV BYTE [ESI+12], 0
  536. MOV EAX, 1
  537. CPUID
  538. MOV ESI, [EBP+version]
  539. MOV [ESI], EAX
  540. MOV ESI, [EBP+features1]
  541. MOV [ESI], EDX
  542. MOV ESI, [EBP+features2]
  543. MOV [ESI], ECX
  544. end:
  545. POP ECX
  546. #ELSIF AMD64 THEN
  547. PUSH RCX
  548. MOV EAX, 0
  549. CPUID
  550. CMP EAX, 0
  551. JNE ok
  552. MOV RSI, [RBP+vendor]
  553. MOV [RSI], AL ; AL = 0
  554. MOV RSI, [RBP+version]
  555. MOV [RSI], EAX ; EAX = 0
  556. MOV RSI, [RBP+features1]
  557. MOV [RSI], EAX
  558. MOV RSI, [RBP+features2]
  559. MOV [RSI], EAX
  560. JMP end
  561. ok:
  562. MOV RSI, [RBP+vendor]
  563. MOV [RSI], EBX
  564. MOV [RSI+4], EDX
  565. MOV [RSI+8], ECX
  566. MOV BYTE [RSI+12], 0
  567. MOV EAX, 1
  568. CPUID
  569. MOV RSI, [RBP+version]
  570. MOV [RSI], EAX
  571. MOV RSI, [RBP+features1]
  572. MOV [RSI], EDX
  573. MOV RSI, [RBP+features2]
  574. MOV [RSI], RCX
  575. end:
  576. POP RCX
  577. #ELSE
  578. unimplemented
  579. #END
  580. END CPUID;
  581. (* If the CPUID instruction is supported, the ID flag (bit 21) of the EFLAGS register is r/w *)
  582. PROCEDURE CpuIdSupported( ) : BOOLEAN;
  583. CODE
  584. #IF I386 THEN
  585. PUSHFD ; save EFLAGS
  586. POP EAX ; store EFLAGS in EAX
  587. MOV EBX, EAX ; save EBX for later testing
  588. XOR EAX, 00200000H ; toggle bit 21
  589. PUSH EAX ; push to stack
  590. POPFD ; save changed EAX to EFLAGS
  591. PUSHFD ; push EFLAGS to TOS
  592. POP EAX ; store EFLAGS in EAX
  593. CMP EAX, EBX ; see if bit 21 has changed
  594. SETNE AL; ; return TRUE if bit 21 has changed, FALSE otherwise
  595. #ELSIF AMD64 THEN
  596. PUSHFQ ; save RFLAGS
  597. POP RAX ; store RFLAGS in EAX
  598. MOV RBX, RAX ; save RBX for later testing
  599. XOR RAX, 00200000H ; toggle bit 21
  600. PUSH RAX ; push to stack
  601. POPFQ ; save changed EAX to EFLAGS
  602. PUSHFQ ; push EFLAGS to TOS
  603. POP RAX ; store EFLAGS in EAX
  604. CMP RAX, RBX ; see if bit 21 has changed
  605. SETNE AL; ; return TRUE if bit 21 has changed, FALSE otherwise
  606. #ELSE
  607. unimplemented
  608. #END
  609. END CpuIdSupported;
  610. (* setup MMX, SSE and SSE2..SSE5 and AVX extension *)
  611. PROCEDURE SetupSSE2Ext;
  612. CONST
  613. MMXFlag=23;(*IN features from EBX*)
  614. FXSRFlag = 24;
  615. SSEFlag = 25;
  616. SSE2Flag = 26;
  617. SSE3Flag = 0; (*IN features2 from ECX*) (*PH 04/11*)
  618. SSSE3Flag =9;
  619. SSE41Flag =19;
  620. SSE42Flag =20;
  621. SSE5Flag = 11;
  622. AVXFlag = 28;
  623. BEGIN
  624. MMXSupport := MMXFlag IN features;
  625. SSESupport := SSEFlag IN features;
  626. SSE2Support := SSESupport & (SSE2Flag IN features);
  627. SSE3Support := SSE2Support & (SSE3Flag IN features2);
  628. SSSE3Support := SSE3Support & (SSSE3Flag IN features2); (* PH 04/11*)
  629. SSE41Support := SSE3Support & (SSE41Flag IN features2);
  630. SSE42Support := SSE3Support & (SSE42Flag IN features2);
  631. SSE5Support := SSE3Support & (SSE5Flag IN features2);
  632. AVXSupport := SSE3Support & (AVXFlag IN features2);
  633. IF SSESupport & (FXSRFlag IN features) THEN
  634. (* InitSSE(); *) (*! not privileged mode in Windows and Unix, not allowed *)
  635. END;
  636. END SetupSSE2Ext;
  637. (** -- Processor initialization -- *)
  638. PROCEDURE -SetFCR( s: SET );
  639. CODE
  640. #IF I386 THEN
  641. FLDCW [ESP] ; parameter s
  642. POP EAX
  643. #ELSIF AMD64 THEN
  644. FLDCW WORD [RSP] ; parameter s
  645. POP RAX
  646. #ELSE
  647. unimplemented
  648. #END
  649. END SetFCR;
  650. PROCEDURE -FCR( ): SET;
  651. CODE
  652. #IF I386 THEN
  653. PUSH 0
  654. FNSTCW [ESP]
  655. FWAIT
  656. POP EAX
  657. #ELSIF AMD64 THEN
  658. PUSH 0
  659. FNSTCW WORD [RSP]
  660. FWAIT
  661. POP RAX
  662. #ELSE
  663. unimplemented
  664. #END
  665. END FCR;
  666. PROCEDURE -InitFPU;
  667. CODE
  668. #IF I386 THEN
  669. FNINIT
  670. #ELSIF AMD64 THEN
  671. FNINIT
  672. #ELSE
  673. unimplemented
  674. #END
  675. END InitFPU;
  676. (** Setup FPU control word of current processor. *)
  677. PROCEDURE SetupFPU*;
  678. BEGIN
  679. InitFPU; SetFCR( fcr )
  680. END SetupFPU;
  681. (* Initialize locks. *)
  682. PROCEDURE InitLocks;
  683. VAR i: LONGINT;
  684. BEGIN
  685. i := 0;
  686. WHILE i < MaxLocks DO mtx[i] := Unix.NewMtx( ); INC( i ) END;
  687. END InitLocks;
  688. PROCEDURE CleanupLocks*;
  689. VAR i: LONGINT;
  690. BEGIN
  691. i := 0;
  692. WHILE i < MaxLocks DO Unix.MtxDestroy( mtx[i] ); INC( i ) END;
  693. END CleanupLocks;
  694. (** Acquire a spin-lock. *)
  695. PROCEDURE Acquire*( level: LONGINT ); (* non reentrant lock *)
  696. BEGIN
  697. Unix.MtxLock( mtx[level] );
  698. END Acquire;
  699. (** Release a spin-lock. *)
  700. PROCEDURE Release*( level: LONGINT );
  701. BEGIN
  702. Unix.MtxUnlock( mtx[level] );
  703. END Release;
  704. PROCEDURE Shutdown*( reboot: BOOLEAN );
  705. VAR r: LONGINT; logstat: Unix.Status;
  706. BEGIN
  707. IF logfile > 0 THEN
  708. r := Unix.fstat( logfile, logstat );
  709. r := Unix.close( logfile );
  710. IF logstat.size = 0 THEN r := Unix.unlink( ADDRESSOF( logname) ) END;
  711. END;
  712. IF reboot THEN Unix.exit( 0 ) ELSE Unix.exit( 1 ) END;
  713. END Shutdown;
  714. (* Set machine-dependent parameter gcThreshold *)
  715. PROCEDURE SetGCParams*;
  716. BEGIN
  717. gcThreshold := 10*1024*1024; (* 10 MB *)
  718. END SetGCParams;
  719. PROCEDURE InitConfig;
  720. VAR a: ADDRESS; i: LONGINT; c: CHAR;
  721. BEGIN
  722. a := Unix.getenv( ADDRESSOF( "AOSCONFIG" ) );
  723. IF a = 0 THEN config := DefaultConfig
  724. ELSE
  725. REPEAT
  726. S.GET( a, c ); INC( a ); config[i] := c; INC( i )
  727. UNTIL c = 0X
  728. END
  729. END InitConfig;
  730. PROCEDURE UpdateTicks*;
  731. BEGIN
  732. ticks := SHORT( (GetTimer() - timer0) DIV (mhz * 1000) );
  733. END UpdateTicks;
  734. PROCEDURE InitThreads;
  735. VAR res: BOOLEAN;
  736. BEGIN
  737. res := Unix.ThrInitialize( prioLow, prioHigh );
  738. IF ~res THEN
  739. Trace.StringLn( "Machine.InitThreads: no threads support in boot environment. teminating" );
  740. Unix.exit( 1 )
  741. END;
  742. IF Glue.debug # {} THEN
  743. Trace.String( "Threads initialized, priorities low, high: " );
  744. Trace.Int( prioLow, 0 ); Trace.String( ", " ); Trace.Int( prioHigh, 0 );
  745. Trace.Ln
  746. END
  747. END InitThreads;
  748. PROCEDURE CPUSpeed;
  749. VAR t0, t1: HUGEINT;
  750. BEGIN
  751. t0 := GetTimer(); Unix.ThrSleep( 100 ); t1 := GetTimer();
  752. mhz := (t1 - t0) DIV 100000;
  753. IF Glue.debug # {} THEN
  754. Trace.String( "CPU speed: ~" ); Trace.Int( SHORT( mhz ), 0); Trace.String( " MHz" ); Trace.Ln
  755. END
  756. END CPUSpeed;
  757. PROCEDURE Log( c: CHAR );
  758. VAR ignore: SIZE;
  759. BEGIN
  760. ignore := Unix.write( 1, ADDRESSOF( c ), 1 );
  761. ignore := Unix.write( logfile, ADDRESSOF( c ), 1 );
  762. END Log;
  763. PROCEDURE LogFileOnly( c: CHAR );
  764. VAR ignore: SIZE;
  765. BEGIN
  766. ignore := Unix.write( logfile, ADDRESSOF( c ), 1 );
  767. END LogFileOnly;
  768. PROCEDURE InitLog;
  769. VAR pid, i: LONGINT;
  770. BEGIN
  771. IF logfile > 0 THEN RETURN END;
  772. logname := "AOS.xxxxx.Log";
  773. pid := Unix.getpid(); i := 8;
  774. REPEAT
  775. logname[i] := CHR( pid MOD 10 + ORD( '0' ) ); DEC( i );
  776. pid := pid DIV 10;
  777. UNTIL i = 3;
  778. logfile := Unix.open( ADDRESSOF( logname ), Unix.rdwr + Unix.creat + Unix.trunc, Unix.rwrwr );
  779. END InitLog;
  780. PROCEDURE SilentLog*;
  781. BEGIN
  782. InitLog;
  783. Trace.Char := LogFileOnly
  784. END SilentLog;
  785. PROCEDURE VerboseLog*;
  786. BEGIN
  787. InitLog;
  788. Trace.Char := Log
  789. END VerboseLog;
  790. PROCEDURE Append( VAR a: ARRAY OF CHAR; CONST this: ARRAY OF CHAR );
  791. VAR i, j: LONGINT;
  792. BEGIN
  793. i := 0; j := 0;
  794. WHILE a[i] # 0X DO INC( i ) END;
  795. WHILE (i < LEN( a ) - 1) & (this[j] # 0X) DO a[i] := this[j]; INC( i ); INC( j ) END;
  796. a[i] := 0X
  797. END Append;
  798. PROCEDURE Init;
  799. VAR vendor: Vendor; ver: LONGINT;
  800. BEGIN
  801. COPY( Unix.Version, version ); Append( version, Version ); Append(version, S.Date);
  802. timer0 := GetTimer( ); ticks := 0;
  803. InitThreads;
  804. InitLocks;
  805. traceHeap := 1 IN Glue.debug;
  806. InitConfig;
  807. CPUSpeed;
  808. IF CpuIdSupported() THEN
  809. CPUID( vendor, ver, features, features2 ); SetupSSE2Ext
  810. END;
  811. fcr := (FCR() - {0,2,3,10,11}) + {0..5,8,9}; (* default FCR RC=00B *)
  812. END Init;
  813. PROCEDURE {INITIAL} Init0*;
  814. BEGIN
  815. Init;
  816. END Init0;
  817. END Machine.
  818. (*
  819. 03.03.1998 pjm First version
  820. 30.06.1999 pjm ProcessorID moved to AosProcessor
  821. *)
  822. (**
  823. Notes
  824. This module defines an interface to the boot environment of the system. The facilities provided here are only intended for the lowest levels of the system, and should never be directly imported by user modules (exceptions are noted below). They are highly specific to the system hardware and firmware architecture.
  825. Typically a machine has some type of firmware that performs initial testing and setup of the system. The firmware initiates the operating system bootstrap loader, which loads the boot file. This module is the first module in the statically linked boot file that gets control.
  826. There are two more-or-less general procedures in this module: GetConfig and StrToInt. GetConfig is used to query low-level system settings, e.g., the location of the boot file system. StrToInt is a utility procedure that parses numeric strings.
  827. Config strings:
  828. ExtMemSize Specifies size of extended memory (above 1MB) in MB. This value is not checked for validity. Setting it false may cause the system to fail, possible after running for some time. The memory size is usually detected automatically, but if the detection does not work for some reason, or if you want to limit the amount of memory detected, this string can be set. For example, if the machine has 64MB of memory, this value can be set as ExtMemSize="63".
  829. *)