Unix.AMD64.Machine.Mod 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857
  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. Version = "Aos (rev.6242)";
  6. DefaultObjectFileExtension* = ".Obj";
  7. Second* = 1000; (* frequency of ticks increments in Hz *)
  8. (** bits in features variable *)
  9. MTTR* = 12; MMX* = 23;
  10. AddrSize = SIZEOF( ADDRESS );
  11. SizeSize = SIZEOF( SIZE );
  12. BlockSize = 32;
  13. MemBlockSize* = 64*1024*1024;
  14. TraceOutput* = 0; (* Trace output *)
  15. Memory* = 1; (*! Virtual memory management, stack and page allocation, not used in UnixAos *)
  16. Heaps* = 2; (* Storage allocation and Garbage collection *)
  17. Interrupts* = 3; (*! Interrupt handling, not used in UnixAos *)
  18. Modules* = 4; (* Module list *)
  19. Objects* = 5; (*! Ready queue, not used in UnixAos *)
  20. Processors* = 6; (*! Interprocessor interrupts, not used in UnixAos *)
  21. KernelLog* = 7; (* Atomic output *)
  22. X11* = 8; (* XWindows I/O *)
  23. Trap* = 9;
  24. GC* = 10;
  25. MaxLocks* = 11; (* { <= 32 } *)
  26. MaxCPU* = 4;
  27. TYPE
  28. Vendor* = ARRAY 13 OF CHAR;
  29. MemoryBlock* = POINTER TO MemoryBlockDesc;
  30. MemoryBlockDesc* = RECORD
  31. next- {UNTRACED}: MemoryBlock;
  32. startAdr-: ADDRESS; (* sort key in linked list of memory blocks *)
  33. size-: SIZE;
  34. beginBlockAdr-, endBlockAdr-: ADDRESS
  35. END;
  36. (** processor state, ordering of record fields is predefined! *)
  37. (*!(not used in UnixAos, for interface compatibility only)*)
  38. State* = RECORD (* offsets used in FieldInterrupt, FieldIRQ and Objects.RestoreState *)
  39. R15*, R14*, R13*, R12*, R11*, R10*, R9*, R8*: HUGEINT;
  40. RDI*, RSI*, ERR*, RSP0*, RBX*, RDX*, RCX*, RAX*: HUGEINT; (** RSP0 = ADR(s.INT) *)
  41. INT*, BP*, PC*, CS*: HUGEINT; (* RBP and ERR are exchanged by glue code, for procedure link *)
  42. FLAGS*: SET;
  43. SP*, SS*: HUGEINT;
  44. END;
  45. VAR
  46. lock- : ARRAY MaxLocks OF CHAR; (* not implemented as SET because of shared access *)
  47. mtx : ARRAY MaxLocks OF Unix.Mutex_t;
  48. version-: ARRAY 64 OF CHAR; (** Aos version *)
  49. features-, features2 : SET;
  50. MMXSupport- : BOOLEAN;
  51. SSESupport- : BOOLEAN;
  52. SSE2Support- : BOOLEAN;
  53. SSE3Support- : BOOLEAN;
  54. SSSE3Support- : BOOLEAN;
  55. SSE41Support- : BOOLEAN;
  56. SSE42Support- : BOOLEAN;
  57. SSE5Support- : BOOLEAN;
  58. AVXSupport- : BOOLEAN;
  59. ticks*: LONGINT; (** timer ticks. Use Kernel.GetTicks() to read, don't write *)
  60. prioLow-, prioHigh-: LONGINT; (* permitted thread priorities *)
  61. fcr-: SET; (** default floating-point control register value (default rounding mode is towards -infinity, for ENTIER) *)
  62. mhz-: HUGEINT; (** clock rate of GetTimer in MHz, or 0 if not known *)
  63. standaloneAppl-: BOOLEAN;
  64. firstMemBlock: MemoryBlockDesc; (* pseudo heap *)
  65. memBlockHead-{UNTRACED}, memBlockTail-{UNTRACED}: MemoryBlock; (* head and tail of sorted list of memory blocks *)
  66. config: ARRAY 2048 OF CHAR; (* config strings *)
  67. thrInitialize : PROCEDURE {REALTIME, C} ( VAR low, high: LONGINT ): BOOLEAN;
  68. mtxInit : PROCEDURE {REALTIME, C} ( dummy: LONGINT ): Unix.Mutex_t;
  69. mtxDestroy : PROCEDURE {REALTIME, C} ( mtx: Unix.Mutex_t );
  70. mtxLock : PROCEDURE {REALTIME, C} ( mtx: Unix.Mutex_t );
  71. mtxUnlock : PROCEDURE {REALTIME, C} ( mtx: Unix.Mutex_t );
  72. conInit : PROCEDURE {REALTIME, C} ( dummy: LONGINT ): Unix.Condition_t;
  73. conDestroy : PROCEDURE {REALTIME, C} ( cond: Unix.Condition_t );
  74. conWait : PROCEDURE {REALTIME, C} ( cond: Unix.Condition_t; mtx: Unix.Mutex_t );
  75. conSignal : PROCEDURE {REALTIME, C} ( cond: Unix.Condition_t );
  76. thrSleep : PROCEDURE {REALTIME, C} ( ms: LONGINT );
  77. thrThis : PROCEDURE {REALTIME, C} ( dummy: LONGINT ): Unix.Thread_t;
  78. saveSP* : PROCEDURE; (* save SP for usage by GC *)
  79. logfile: LONGINT;
  80. traceHeap: BOOLEAN;
  81. timer0 : HUGEINT;
  82. (** -- Processor identification -- *)
  83. (** Return current processor ID (0 to MaxNum-1). *)
  84. PROCEDURE {REALTIME} ID* (): LONGINT;
  85. BEGIN
  86. RETURN 0
  87. END ID;
  88. (* insert given memory block in sorted list of memory blocks, sort key is startAdr field - called during GC *)
  89. PROCEDURE InsertMemoryBlock(memBlock: MemoryBlock);
  90. VAR cur {UNTRACED}, prev {UNTRACED}: MemoryBlock;
  91. BEGIN
  92. cur := memBlockHead;
  93. prev := NIL;
  94. WHILE (cur # NIL) & (cur.startAdr < memBlock.startAdr) DO
  95. prev := cur;
  96. cur := cur.next
  97. END;
  98. IF prev = NIL THEN (* insert at head of list *)
  99. memBlock.next := memBlockHead;
  100. memBlockHead := memBlock
  101. ELSE (* insert in middle or at end of list *)
  102. prev.next := memBlock;
  103. memBlock.next := cur;
  104. IF cur = NIL THEN
  105. memBlockTail := memBlock
  106. END
  107. END
  108. END InsertMemoryBlock;
  109. (* Free unused memory block - called during GC *)
  110. PROCEDURE FreeMemBlock*(memBlock: MemoryBlock);
  111. VAR cur {UNTRACED}, prev {UNTRACED}: MemoryBlock;
  112. BEGIN
  113. cur := memBlockHead;
  114. prev := NIL;
  115. WHILE (cur # NIL) & (cur # memBlock) DO
  116. prev := cur;
  117. cur := cur.next
  118. END;
  119. IF cur = memBlock THEN
  120. IF traceHeap THEN
  121. Trace.String( "Release memory block " ); Trace.Hex( memBlock.startAdr, -8 ); Trace.Ln
  122. END;
  123. IF prev = NIL THEN
  124. memBlockHead := cur.next
  125. ELSE
  126. prev.next := cur.next;
  127. IF cur.next = NIL THEN
  128. memBlockTail := prev
  129. END
  130. END;
  131. Unix.free( memBlock.startAdr )
  132. ELSE
  133. HALT(535) (* error in memory block management *)
  134. END;
  135. END FreeMemBlock;
  136. (* expand heap by allocating a new memory block *)
  137. PROCEDURE ExpandHeap*( dummy: LONGINT; size: SIZE; VAR beginBlockAdr, endBlockAdr: ADDRESS );
  138. VAR mBlock: MemoryBlock; alloc, s: SIZE; a, adr: ADDRESS;
  139. BEGIN
  140. IF size < (MemBlockSize - (2*BlockSize)) THEN alloc := MemBlockSize
  141. ELSE alloc := size + (2*BlockSize);
  142. END;
  143. INC( alloc, (-alloc) MOD Unix.PageSize );
  144. IF Unix.posix_memalign( adr, Unix.PageSize, alloc ) # 0 THEN
  145. Unix.Perror( "Machine.ExpandHeap: posix_memalign" );
  146. beginBlockAdr := 0;
  147. endBlockAdr := 0
  148. ELSE
  149. IF Unix.mprotect( adr, alloc, 7 (* READ WRITE EXEC *) ) # 0 THEN
  150. Unix.Perror( "Machine.ExpandHeap: mprotect" )
  151. END;
  152. mBlock := S.VAL( MemoryBlock, adr );
  153. mBlock.next := NIL;
  154. mBlock.startAdr := adr;
  155. mBlock.size := alloc;
  156. mBlock.beginBlockAdr := adr + BlockSize - AddrSize;
  157. ASSERT( (mBlock.beginBlockAdr + AddrSize) MOD BlockSize = 0 );
  158. s := adr + alloc - mBlock.beginBlockAdr - BlockSize;
  159. DEC( s, s MOD BlockSize );
  160. ASSERT( s >= size );
  161. mBlock.endBlockAdr := mBlock.beginBlockAdr + s;
  162. InsertMemoryBlock( mBlock );
  163. IF traceHeap THEN TraceHeap( mBlock ) END;
  164. a := mBlock.beginBlockAdr;
  165. S.PUT( a, a + AddrSize ); (* tag *)
  166. S.PUT( a + AddrSize, s - AddrSize ); (* size *)
  167. S.PUT( a + AddrSize + SizeSize, S.VAL( ADDRESS, 0 ) ); (* next *)
  168. beginBlockAdr := mBlock.beginBlockAdr;
  169. endBlockAdr := mBlock.endBlockAdr;
  170. END
  171. END ExpandHeap;
  172. PROCEDURE TraceHeap( new: MemoryBlock );
  173. VAR cur{UNTRACED}: MemoryBlock;
  174. BEGIN
  175. Trace.Ln;
  176. Trace.String( "Heap expanded" ); Trace.Ln;
  177. cur := memBlockHead;
  178. WHILE cur # NIL DO
  179. Trace.Hex( cur.startAdr, -8 ); Trace.String( " " ); Trace.Int( cur.size, 15 );
  180. IF cur = new THEN Trace.String( " (new)" ) END;
  181. Trace.Ln;
  182. cur := cur.next
  183. END
  184. END TraceHeap;
  185. (* returns if an address is a currently allocated heap address *)
  186. PROCEDURE ValidHeapAddress*( p: ADDRESS ): BOOLEAN;
  187. VAR mb: MemoryBlock;
  188. BEGIN
  189. mb := memBlockHead;
  190. WHILE mb # NIL DO
  191. IF (p >= mb.beginBlockAdr) & (p <= mb.endBlockAdr) THEN RETURN TRUE END;
  192. mb := mb.next;
  193. END;
  194. RETURN FALSE
  195. END ValidHeapAddress;
  196. (** Return information on free memory in Kbytes. *)
  197. PROCEDURE GetFreeK*(VAR total, lowFree, highFree: SIZE);
  198. BEGIN
  199. (* meaningless in Unix port, for interface compatibility only *)
  200. total := 0;
  201. lowFree := 0;
  202. highFree := 0
  203. END GetFreeK;
  204. (** Fill "size" bytes at "destAdr" with "filler". "size" must be multiple of 4. *)
  205. PROCEDURE Fill32* (destAdr: ADDRESS; size: SIZE; filler: LONGINT);
  206. CODE {SYSTEM.AMD64}
  207. MOV RDI, [RBP + destAdr]
  208. MOV RCX, [RBP + size]
  209. MOV EAX, [RBP + filler]
  210. TEST RCX, 3
  211. JZ ok
  212. PUSH 8 ; ASSERT failure
  213. INT 3
  214. ok:
  215. SHR RCX, 2
  216. CLD
  217. REP STOSD
  218. END Fill32;
  219. (** Return timer value of the current processor, or 0 if not available. *)
  220. (* e.g. ARM does not have a fine-grained timer *)
  221. PROCEDURE -GetTimer* (): HUGEINT;
  222. CODE {SYSTEM.AMD64}
  223. XOR RAX, RAX
  224. RDTSC ; set EDX:EAX
  225. SHL RDX, 32
  226. OR RAX, RDX
  227. END GetTimer;
  228. PROCEDURE Portin8*(port: LONGINT; VAR val: CHAR);
  229. END Portin8;
  230. PROCEDURE Portin16*(port: LONGINT; VAR val: INTEGER);
  231. END Portin16;
  232. PROCEDURE Portin32*(port: LONGINT; VAR val: LONGINT);
  233. END Portin32;
  234. PROCEDURE Portout8*(port: LONGINT; val: CHAR);
  235. END Portout8;
  236. PROCEDURE Portout16*(port: LONGINT; val: INTEGER);
  237. END Portout16;
  238. PROCEDURE Portout32*(port: LONGINT; val: LONGINT);
  239. END Portout32;
  240. PROCEDURE Is32BitAddress*( adr: ADDRESS ): BOOLEAN;
  241. BEGIN RETURN S.VAL( LONGINT, adr ) = adr;
  242. END Is32BitAddress;
  243. (** -- Atomic operations -- *)
  244. (** Atomic INC(x). *)
  245. PROCEDURE -AtomicInc*(VAR x: LONGINT);
  246. CODE {SYSTEM.AMD64}
  247. POP RAX
  248. LOCK
  249. INC DWORD [RAX]
  250. END AtomicInc;
  251. (** Atomic DEC(x). *)
  252. PROCEDURE -AtomicDec*(VAR x: LONGINT);
  253. CODE {SYSTEM.AMD64}
  254. POP RAX
  255. LOCK
  256. DEC DWORD [RAX]
  257. END AtomicDec;
  258. (** Atomic EXCL. *)
  259. PROCEDURE AtomicExcl* (VAR s: SET; bit: LONGINT);
  260. CODE {SYSTEM.AMD64}
  261. MOV EAX, [RBP + bit]
  262. MOV RBX, [RBP + s]
  263. LOCK
  264. BTR [RBX], EAX
  265. END AtomicExcl;
  266. (** Atomic INC(x, y). *)
  267. PROCEDURE -AtomicAdd*(VAR x: LONGINT; y: LONGINT);
  268. CODE {SYSTEM.AMD64}
  269. POP EBX
  270. POP RAX
  271. LOCK
  272. ADD DWORD [RAX], EBX
  273. END AtomicAdd;
  274. (** Atomic test-and-set. Set x = TRUE and return old value of x. *)
  275. PROCEDURE -AtomicTestSet*(VAR x: BOOLEAN): BOOLEAN;
  276. CODE {SYSTEM.AMD64}
  277. POP RBX
  278. MOV AL, 1
  279. XCHG [RBX], AL
  280. END AtomicTestSet;
  281. (* Atomic compare-and-swap. Set x = new if x = old and return old value of x *)
  282. PROCEDURE -AtomicCAS* (VAR x: LONGINT; old, new: LONGINT): LONGINT;
  283. CODE {SYSTEM.AMD64}
  284. POP EBX ; new
  285. POP EAX ; old
  286. POP RCX ; address of x
  287. LOCK CMPXCHG [RCX], EBX ; atomicly compare x with old and set it to new if equal
  288. END AtomicCAS;
  289. (* Return current instruction pointer *)
  290. PROCEDURE CurrentPC* (): ADDRESS;
  291. CODE {SYSTEM.AMD64}
  292. MOV RAX, [RBP + 8]
  293. END CurrentPC;
  294. (* Return current frame pointer *)
  295. PROCEDURE -CurrentBP* (): ADDRESS;
  296. CODE {SYSTEM.AMD64}
  297. MOV RAX, RBP
  298. END CurrentBP;
  299. (* Set current frame pointer *)
  300. PROCEDURE -SetBP* (bp: ADDRESS);
  301. CODE {SYSTEM.AMD64}
  302. POP RBP
  303. END SetBP;
  304. (* Return current stack pointer *)
  305. PROCEDURE -CurrentSP* (): ADDRESS;
  306. CODE {SYSTEM.AMD64}
  307. MOV RAX, RSP
  308. END CurrentSP;
  309. (* Set current stack pointer *)
  310. PROCEDURE -SetSP* (sp: ADDRESS);
  311. CODE {SYSTEM.AMD64}
  312. POP RSP
  313. END SetSP;
  314. PROCEDURE -GetRAX*(): HUGEINT;
  315. CODE{SYSTEM.AMD64}
  316. END GetRAX;
  317. PROCEDURE -GetRCX*(): HUGEINT;
  318. CODE{SYSTEM.AMD64}
  319. MOV RAX,RCX
  320. END GetRCX;
  321. PROCEDURE -GetRSI*(): HUGEINT;
  322. CODE{SYSTEM.AMD64}
  323. MOV RAX,RSI
  324. END GetRSI;
  325. PROCEDURE -GetRDI*(): HUGEINT;
  326. CODE{SYSTEM.AMD64}
  327. MOV RAX,RDI
  328. END GetRDI;
  329. PROCEDURE -SetRAX*(n: HUGEINT);
  330. CODE{SYSTEM.AMD64}
  331. NOP
  332. POP RAX
  333. END SetRAX;
  334. PROCEDURE -SetRBX*(n: HUGEINT);
  335. CODE{SYSTEM.AMD64}
  336. NOP
  337. POP RBX
  338. END SetRBX;
  339. PROCEDURE -SetRCX*(n: HUGEINT);
  340. CODE{SYSTEM.AMD64}
  341. POP RCX
  342. END SetRCX;
  343. PROCEDURE -SetRDX*(n: HUGEINT);
  344. CODE{SYSTEM.AMD64}
  345. POP RDX
  346. END SetRDX;
  347. PROCEDURE -SetRSI*(n: HUGEINT);
  348. CODE{SYSTEM.AMD64}
  349. POP RSI
  350. END SetRSI;
  351. PROCEDURE -SetRDI*(n: HUGEINT);
  352. CODE{SYSTEM.AMD64}
  353. POP RDI
  354. END SetRDI;
  355. (** -- Configuration and bootstrapping -- *)
  356. (** 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. *)
  357. PROCEDURE GetConfig* (CONST name: ARRAY OF CHAR; VAR val: ARRAY OF CHAR);
  358. VAR i, src: LONGINT; ch: CHAR;
  359. BEGIN
  360. ASSERT (name[0] # "="); (* no longer supported, use GetInit instead *)
  361. src := -1;
  362. LOOP
  363. REPEAT
  364. INC( src ); ch := config[src];
  365. IF ch = 0X THEN EXIT END;
  366. UNTIL ch > ' ';
  367. i := 0;
  368. LOOP
  369. ch := config[src];
  370. IF (ch # name[i]) OR (name[i] = 0X) THEN EXIT END;
  371. INC (i); INC (src)
  372. END;
  373. IF (ch <= ' ') & (name[i] = 0X) THEN (* found *)
  374. i := 0;
  375. REPEAT
  376. INC (src); ch := config[src]; val[i] := ch; INC (i);
  377. IF i = LEN(val) THEN val[i - 1] := 0X; RETURN END (* val too short *)
  378. UNTIL ch <= ' ';
  379. IF ch = ' ' THEN val[i -1] := 0X END;
  380. RETURN
  381. ELSE
  382. WHILE ch > ' ' DO (* skip to end of name *)
  383. INC (src); ch := config[src]
  384. END;
  385. INC (src);
  386. REPEAT (* skip to end of value *)
  387. ch := config[src]; INC (src)
  388. UNTIL ch <= ' '
  389. END
  390. END;
  391. val[0] := 0X
  392. END GetConfig;
  393. (** 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. *)
  394. PROCEDURE StrToInt* (VAR i: LONGINT; CONST s: ARRAY OF CHAR): LONGINT;
  395. VAR vd, vh, sgn, d: LONGINT; hex: BOOLEAN;
  396. BEGIN
  397. vd := 0; vh := 0; hex := FALSE;
  398. IF s[i] = "-" THEN sgn := -1; INC (i) ELSE sgn := 1 END;
  399. LOOP
  400. IF (s[i] >= "0") & (s[i] <= "9") THEN d := ORD (s[i])-ORD ("0")
  401. ELSIF (CAP (s[i]) >= "A") & (CAP (s[i]) <= "F") THEN d := ORD (CAP (s[i]))-ORD ("A") + 10; hex := TRUE
  402. ELSE EXIT
  403. END;
  404. vd := 10*vd + d; vh := 16*vh + d;
  405. INC (i)
  406. END;
  407. IF CAP (s[i]) = "H" THEN hex := TRUE; INC (i) END; (* optional H *)
  408. IF hex THEN vd := vh END;
  409. RETURN sgn * vd
  410. END StrToInt;
  411. (* function returning the number of processors that are available to Aos *)
  412. PROCEDURE NumberOfProcessors*( ): LONGINT;
  413. BEGIN
  414. RETURN 1
  415. END NumberOfProcessors;
  416. (*! non portable code, for native Aos only *)
  417. PROCEDURE SetNumberOfProcessors*( num: LONGINT );
  418. BEGIN
  419. (* numberOfProcessors := num; *)
  420. END SetNumberOfProcessors;
  421. (* function for changing byte order *)
  422. PROCEDURE ChangeByteOrder* (n: LONGINT): LONGINT;
  423. CODE {SYSTEM.AMD64}
  424. MOV EAX, [RBP + n] ; load n in eax
  425. BSWAP EAX ; swap byte order
  426. END ChangeByteOrder;
  427. (* Send and print character *)
  428. PROCEDURE TraceChar *(c: CHAR);
  429. BEGIN
  430. Trace.Char( c )
  431. END TraceChar;
  432. (** CPU identification *)
  433. PROCEDURE CPUID*(function : LONGINT; VAR eax, ebx, ecx, edx : SET);
  434. CODE {SYSTEM.AMD64}
  435. MOV EAX, [RBP+function] ; CPUID function parameter
  436. MOV RSI, [RBP+ecx] ; copy ecx into ECX (sometimes used as input parameter)
  437. MOV ECX, [RSI]
  438. CPUID ; execute CPUID
  439. MOV RSI, [RBP+eax] ; copy EAX into eax;
  440. MOV [RSI], EAX
  441. MOV RSI, [RBP+ebx] ; copy EBX into ebx
  442. MOV [RSI], EBX
  443. MOV RSI, [RBP+ecx] ; copy ECX into ecx
  444. MOV [RSI], ECX
  445. MOV RSI, [RBP+edx] ; copy EDX into edx
  446. MOV [RSI], EDX
  447. END CPUID;
  448. (* If the CPUID instruction is supported, the ID flag (bit 21) of the EFLAGS register is r/w *)
  449. PROCEDURE CpuIdSupported*() : BOOLEAN;
  450. CODE {SYSTEM.AMD64}
  451. PUSHFQ ; save RFLAGS
  452. POP RAX ; store RFLAGS in RAX
  453. MOV EBX, EAX ; save EBX for later testing
  454. XOR EAX, 00200000H ; toggle bit 21
  455. PUSH RAX ; push to stack
  456. POPFQ ; save changed RAX to RFLAGS
  457. PUSHFQ ; push RFLAGS to TOS
  458. POP RAX ; store RFLAGS in RAX
  459. CMP EAX, EBX ; see if bit 21 has changed
  460. SETNE AL; ; return TRUE if bit 21 has changed, FALSE otherwise
  461. END CpuIdSupported;
  462. (* setup MMX, SSE and SSE2..SSE5 and AVX extension *)
  463. PROCEDURE SetupSSE2Ext;
  464. CONST
  465. MMXFlag=23;(*IN features from EBX*)
  466. FXSRFlag = 24;
  467. SSEFlag = 25;
  468. SSE2Flag = 26;
  469. SSE3Flag = 0; (*IN features2 from ECX*) (*PH 04/11*)
  470. SSSE3Flag =9;
  471. SSE41Flag =19;
  472. SSE42Flag =20;
  473. SSE5Flag = 11;
  474. AVXFlag = 28;
  475. BEGIN
  476. MMXSupport := MMXFlag IN features;
  477. SSESupport := SSEFlag IN features;
  478. SSE2Support := SSESupport & (SSE2Flag IN features);
  479. SSE3Support := SSE2Support & (SSE3Flag IN features2);
  480. SSSE3Support := SSE3Support & (SSSE3Flag IN features2); (* PH 04/11*)
  481. SSE41Support := SSE3Support & (SSE41Flag IN features2);
  482. SSE42Support := SSE3Support & (SSE42Flag IN features2);
  483. SSE5Support := SSE3Support & (SSE5Flag IN features2);
  484. AVXSupport := SSE3Support & (AVXFlag IN features2);
  485. IF SSESupport & (FXSRFlag IN features) THEN
  486. (* InitSSE(); *) (*! not privileged mode in Windows and Unix not allowed *)
  487. END;
  488. END SetupSSE2Ext;
  489. (** -- Processor initialization -- *)
  490. PROCEDURE -SetFCR (s: SET);
  491. CODE {SYSTEM.AMD64, SYSTEM.FPU}
  492. FLDCW WORD [RSP] ; parameter s
  493. POP RAX
  494. END SetFCR;
  495. PROCEDURE -FCR (): SET;
  496. CODE {SYSTEM.AMD64, SYSTEM.FPU}
  497. PUSH 0
  498. FNSTCW WORD [RSP]
  499. FWAIT
  500. POP RAX
  501. END FCR;
  502. PROCEDURE -InitFPU;
  503. CODE {SYSTEM.AMD64, SYSTEM.FPU}
  504. FNINIT
  505. END InitFPU;
  506. (** Setup FPU control word of current processor. *)
  507. PROCEDURE SetupFPU*;
  508. BEGIN
  509. InitFPU; SetFCR(fcr)
  510. END SetupFPU;
  511. (* Initialize locks. *)
  512. PROCEDURE InitLocks;
  513. VAR i: LONGINT;
  514. BEGIN
  515. i := 0;
  516. WHILE i < MaxLocks DO
  517. mtx[i] := mtxInit(0); lock[i] := "N"; INC( i )
  518. END;
  519. END InitLocks;
  520. PROCEDURE CleanupLocks*;
  521. VAR i: LONGINT;
  522. BEGIN
  523. i := 0;
  524. WHILE i < MaxLocks DO mtxDestroy( mtx[i] ); INC( i ) END;
  525. END CleanupLocks;
  526. (** Acquire a spin-lock. *)
  527. PROCEDURE Acquire*( level: LONGINT ); (* non reentrant lock *)
  528. BEGIN
  529. lock[level] := "Y";
  530. mtxLock( mtx[level] );
  531. IF level = Heaps THEN saveSP END
  532. END Acquire;
  533. (** Release a spin-lock. *)
  534. PROCEDURE Release*( level: LONGINT );
  535. BEGIN
  536. mtxUnlock( mtx[level] );
  537. lock[level] := "N";
  538. END Release;
  539. PROCEDURE Shutdown*( reboot: BOOLEAN );
  540. VAR ignore: LONGINT;
  541. BEGIN
  542. ignore := Unix.close( logfile );
  543. IF reboot THEN Unix.exit( 0 ) ELSE Unix.exit( 1 ) END;
  544. END Shutdown;
  545. PROCEDURE InitHeap;
  546. VAR heapAdr, firstBlock: ADDRESS; size: SIZE;
  547. BEGIN
  548. Unix.Dlsym( 0, "heapAdr", ADDRESSOF( heapAdr ) );
  549. Unix.Dlsym( 0, "heapSize", ADDRESSOF( size ) );
  550. firstBlock := heapAdr + ((-heapAdr - AddrSize) MOD BlockSize);
  551. size := heapAdr + size - BlockSize - firstBlock; DEC( size, size MOD BlockSize + BlockSize );
  552. firstMemBlock.next := NIL;
  553. firstMemBlock.startAdr := heapAdr;
  554. firstMemBlock.beginBlockAdr := firstBlock;
  555. firstMemBlock.endBlockAdr := firstBlock + size;
  556. firstMemBlock.size := size;
  557. memBlockHead := S.VAL( MemoryBlock, ADDRESSOF( firstMemBlock ) );
  558. memBlockTail := memBlockHead;
  559. END InitHeap;
  560. PROCEDURE InitConfig;
  561. VAR a: ADDRESS; i: LONGINT; c: CHAR;
  562. BEGIN
  563. a := Unix.getenv( ADDRESSOF( "AOSCONFIG" ) );
  564. IF a = 0 THEN config := DefaultConfig
  565. ELSE
  566. REPEAT
  567. S.GET( a, c ); INC( a ); config[i] := c; INC( i )
  568. UNTIL c = 0X
  569. END
  570. END InitConfig;
  571. PROCEDURE UpdateTicks*;
  572. BEGIN
  573. ticks := SHORT( (GetTimer() - timer0) DIV (mhz * 1000) );
  574. END UpdateTicks;
  575. PROCEDURE InitThreads;
  576. VAR res: BOOLEAN;
  577. BEGIN
  578. res := thrInitialize( prioLow, prioHigh );
  579. IF ~res THEN
  580. Trace.StringLn( "Machine.InitThreads: no threads support in boot environment. teminating" );
  581. Unix.exit( 1 )
  582. END;
  583. IF Glue.debug # {} THEN
  584. Trace.String( "Threads initialized, priorities low, high: " );
  585. Trace.Int( prioLow, 0 ); Trace.String( ", " ); Trace.Int( prioHigh, 0 );
  586. Trace.Ln
  587. END
  588. END InitThreads;
  589. PROCEDURE CPUSpeed;
  590. VAR t0, t1: HUGEINT;
  591. BEGIN
  592. t0 := GetTimer(); thrSleep( 100 ); t1 := GetTimer();
  593. mhz := (t1 - t0) DIV 100000;
  594. IF Glue.debug # {} THEN
  595. Trace.String( "CPU speed: ~" ); Trace.Int( SHORT( mhz ), 0); Trace.String( " MHz" ); Trace.Ln
  596. END
  597. END CPUSpeed;
  598. PROCEDURE Log1( c: CHAR );
  599. VAR ignore: LONGINT;
  600. BEGIN
  601. ignore := Unix.write( 1, ADDRESSOF( c ), 1 );
  602. ignore := Unix.write( logfile, ADDRESSOF( c ), 1 );
  603. END Log1;
  604. PROCEDURE Log2( c: CHAR );
  605. VAR ignore: LONGINT;
  606. BEGIN
  607. ignore := Unix.write( logfile, ADDRESSOF( c ), 1 );
  608. END Log2;
  609. PROCEDURE InitLog;
  610. VAR name: ARRAY 32 OF CHAR; pid, i, d: LONGINT;
  611. BEGIN
  612. name := "AOS.xxxxx.Log";
  613. pid := Unix.getpid(); i := 8;
  614. REPEAT
  615. name[i] := CHR( pid MOD 10 + ORD( '0' ) ); DEC( i );
  616. pid := pid DIV 10;
  617. UNTIL i = 3;
  618. logfile := Unix.open( ADDRESSOF( name ), Unix.rdwr + Unix.creat + Unix.trunc, Unix.rwrwr );
  619. IF Unix.argc < 3 THEN VerboseLog
  620. ELSE standaloneAppl := TRUE; SilentLog
  621. END
  622. END InitLog;
  623. PROCEDURE SilentLog*;
  624. BEGIN
  625. Trace.Char := Log2
  626. END SilentLog;
  627. PROCEDURE VerboseLog*;
  628. BEGIN
  629. Trace.Char := Log1
  630. END VerboseLog;
  631. PROCEDURE Append( VAR a: ARRAY OF CHAR; CONST this: ARRAY OF CHAR );
  632. VAR i, j: LONGINT;
  633. BEGIN
  634. i := 0; j := 0;
  635. WHILE a[i] # 0X DO INC( i ) END;
  636. WHILE (i < LEN( a ) - 1) & (this[j] # 0X) DO a[i] := this[j]; INC( i ); INC( j ) END;
  637. a[i] := 0X
  638. END Append;
  639. PROCEDURE {REALTIME} Empty;
  640. END Empty;
  641. PROCEDURE Init;
  642. VAR eax, ebx, ecx, edx: SET;
  643. BEGIN
  644. Unix.Dlsym( 0, "thrInitialize", ADDRESSOF( thrInitialize ) );
  645. Unix.Dlsym( 0, "mtxInit", ADDRESSOF( mtxInit ) );
  646. Unix.Dlsym( 0, "mtxDestroy", ADDRESSOF( mtxDestroy ) );
  647. Unix.Dlsym( 0, "mtxLock", ADDRESSOF( mtxLock ) );
  648. Unix.Dlsym( 0, "mtxUnlock", ADDRESSOF( mtxUnlock ) );
  649. Unix.Dlsym( 0, "conInit", ADDRESSOF( conInit ) );
  650. Unix.Dlsym( 0, "conDestroy", ADDRESSOF( conDestroy ) );
  651. Unix.Dlsym( 0, "conWait", ADDRESSOF( conWait ) );
  652. Unix.Dlsym( 0, "conSignal", ADDRESSOF( conSignal ) );
  653. Unix.Dlsym( 0, "thrSleep", ADDRESSOF( thrSleep ) );
  654. Unix.Dlsym( 0, "thrThis", ADDRESSOF( thrThis ) );
  655. standaloneAppl := FALSE;
  656. saveSP := Empty; (* to be replaced in module Objects *)
  657. COPY( Unix.Version, version ); Append( version, Version );
  658. timer0 := GetTimer( ); ticks := 0;
  659. InitThreads;
  660. InitLocks;
  661. traceHeap := 1 IN Glue.debug;
  662. InitHeap;
  663. InitConfig;
  664. InitLog;
  665. CPUSpeed;
  666. (* CPUID standard function 1 returns: CPU features in ecx & edx *)
  667. CPUID( 1, eax, ebx, ecx, edx );
  668. features := S.VAL( SET, edx );
  669. features2 := S.VAL( SET, ecx );
  670. SetupSSE2Ext;
  671. fcr := (FCR() - {0,2,3,10,11}) + {0..5,8,9}; (* default FCR RC=00B *)
  672. END Init;
  673. BEGIN
  674. Init
  675. END Machine.
  676. (*
  677. 03.03.1998 pjm First version
  678. 30.06.1999 pjm ProcessorID moved to AosProcessor
  679. *)
  680. (**
  681. Notes
  682. 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.
  683. 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.
  684. 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.
  685. Config strings:
  686. 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".
  687. *)