Generic.Reflection.Mod 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238
  1. MODULE Reflection;
  2. (** (c) Felix Friedrich, ETH Zurich, 2016 -- Reflection with more structured references section emitted by FoxCompiler *)
  3. IMPORT Modules, Streams, SYSTEM, Machine, Heaps, Objects, Trace;
  4. CONST
  5. ShowAllProcs = TRUE;
  6. MaxFrames = 128;
  7. MaxString = 64;
  8. MaxArray = 8;
  9. MaxCols = 70;
  10. Sep = " ";
  11. SepLen = 2;
  12. LineDelay = 0;
  13. CONST
  14. sfTypeNone* = 0X;
  15. sfTypeCHAR* = 01X;
  16. sfTypeCHAR8* = 02X;
  17. sfTypeCHAR16* = 03X;
  18. sfTypeCHAR32* = 04X;
  19. sfTypeRANGE* = 05X;
  20. sfTypeSHORTINT* = 06X;
  21. sfTypeINTEGER* = 07X;
  22. sfTypeLONGINT* = 08X;
  23. sfTypeHUGEINT* = 09X;
  24. sfTypeWORD* = 0AX;
  25. sfTypeLONGWORD* = 0BX;
  26. sfTypeSIGNED8* = 0CX;
  27. sfTypeSIGNED16* = 0DX;
  28. sfTypeSIGNED32* = 0EX;
  29. sfTypeSIGNED64* = 0FX;
  30. sfTypeUNSIGNED8* = 10X;
  31. sfTypeUNSIGNED16* = 11X;
  32. sfTypeUNSIGNED32* = 12X;
  33. sfTypeUNSIGNED64* = 13X;
  34. sfTypeREAL* = 14X;
  35. sfTypeLONGREAL* = 15X;
  36. sfTypeCOMPLEX* = 16X;
  37. sfTypeLONGCOMPLEX* = 17X;
  38. sfTypeBOOLEAN* = 18X;
  39. sfTypeSET* = 19X;
  40. sfTypeANY* = 1AX;
  41. sfTypeOBJECT* = 1BX;
  42. sfTypeBYTE* = 1CX;
  43. sfTypeADDRESS* = 1DX;
  44. sfTypeSIZE* = 1EX;
  45. sfTypeIndirect*= 1FX;
  46. sfTypeRecord* = 20X;
  47. sfTypePointerToRecord* = 21X;
  48. sfTypePointerToArray* = 22X;
  49. sfTypeOpenArray* = 23X;
  50. sfTypeStaticArray* = 24X;
  51. sfTypeDynamicArray* = 25X;
  52. sfTypeMathStaticArray* = 26X;
  53. sfTypeMathOpenArray* = 27X;
  54. sfTypeMathTensor* = 28X;
  55. sfTypeDelegate* = 29X;
  56. sfTypeENUM* = 2AX;
  57. sfTypeCELL* = 2BX;
  58. sfTypePORT* = 2CX;
  59. sfIN* = 0X;
  60. sfOUT* = 1X;
  61. flagDelegate*=0;
  62. flagConstructor*=1;
  63. (* variable / parameter addressing modes *)
  64. sfAbsolute* = 0X; (* global vars *)
  65. sfRelative* = 1X; (* variables, value parameters *)
  66. sfIndirect* = 2X; (* var parameters *)
  67. sfScopeBegin* = 0F0X;
  68. sfScopeEnd* = 0F1X;
  69. sfProcedure* = 0F2X;
  70. sfVariable* = 0F3X;
  71. sfTypeDeclaration* = 0F4X;
  72. sfModule*= 0FFX;
  73. (*
  74. References section format:
  75. Scope = sfScopeBegin {variable:Variable} {procedure:Procedure} {typeDecl:TypeDeclaration} sfScopeEnd.
  76. Module = sfModule prevSymbolOffset:SIZE name:String Scope.
  77. Procedure = sfProcedure prevSymbolOffset:SIZE name:String start:ADR end:ADR flags:SET {parameter:Variable} returnType:Type Scope.
  78. Variable = sfVariable prevSymbolOffset:SIZE name:String (sfRelative offset: SIZE | sfIndirec offset: SIZE | sfAbsolute address:ADDRESS) type:Type.
  79. TypeDeclaration = sfTypeDeclaration prevSymbolOffset:SIZE name:String typeInfo:ADR Scope.
  80. Type =
  81. sfTypePointerToRecord
  82. | sfTypePointerToArray Type
  83. | sfTypeOpenArray Type
  84. | sfTypeDynamicArray Type
  85. | sfTypeStaticArray length:SIZE Type
  86. | sfTypeMathOpenArray Type
  87. | sfTypeMathStaticArray length:SIZE Type
  88. | sfTypeMathTensor Type
  89. | sfTypeRecord tdAdr:ADDRESS
  90. | sfTypeDelegate {Parameter} return:Type
  91. | sfTypePort (sfIN | sfOUT)
  92. | sfTypeBOOLEAN
  93. | sfTypeCHAR | sfTypeCHAR8 | sfTypeCHAR16 | sfTypeCHAR32
  94. | sfTypeSHORTINT | sfTypeINTEGER | sfTypeLONGINT | sfTypeHUGEINT
  95. | sfTypeSIGNED8 | sfTypeSIGNED16 | sfTypeSIGNED32 | sfTypeSIGNED64
  96. | sfTypeUNSIGNED8 | sfTypeUNSIGNED16 | sfTypeUNSIGNED32 | sfTypeUNSIGNED64
  97. | sfTypeWORD | sfTypeLONGWORD
  98. | sfTypeREAL | sfTypeLONGREAL
  99. | sfTypeCOMPLEX | sfTypeLONGCOMPLEX
  100. | sfTypeSET | sfTypeANY | sfTypeOBJECT | sfTypeBYTE | sfTypeADDRESS | sfTypeSIZE
  101. | sfTypeIndirect offset:SIZE.
  102. *)
  103. VAR
  104. modes: ARRAY 25 OF CHAR;
  105. TYPE
  106. Name = ARRAY 128 OF CHAR;
  107. Meta* = RECORD
  108. module-: Modules.Module;
  109. refs-: Modules.Bytes;
  110. offset*: LONGINT;
  111. END;
  112. PROCEDURE CheckHeapAddress(address: ADDRESS): BOOLEAN;
  113. BEGIN
  114. RETURN Machine.ValidHeapAddress(address);
  115. END CheckHeapAddress;
  116. PROCEDURE Wait(w: Streams.Writer);
  117. VAR i: LONGINT;
  118. BEGIN
  119. IF LineDelay > 0 THEN
  120. FOR i := 0 TO LineDelay DO END;
  121. w.Update
  122. END;
  123. END Wait;
  124. PROCEDURE Expect(b: BOOLEAN): BOOLEAN;
  125. BEGIN
  126. IF ~b THEN Trace.String("Format error in references section"); Trace.Ln END;
  127. RETURN b;
  128. END Expect;
  129. (* consume a char from the byte stream *)
  130. PROCEDURE GetChar*(refs: Modules.Bytes; VAR offset: LONGINT): CHAR;
  131. VAR c: CHAR;
  132. BEGIN
  133. IF ~Expect(offset < LEN(refs)) THEN RETURN 0X END;
  134. c := refs[offset];
  135. INC(offset);
  136. RETURN c;
  137. END GetChar;
  138. (* skip a char in the byte stream *)
  139. PROCEDURE SkipChar*(VAR offset: LONGINT);
  140. BEGIN
  141. INC(offset, SIZEOF(CHAR));
  142. END SkipChar;
  143. (* consume an address in the byte stream *)
  144. PROCEDURE GetAddress*(refs: Modules.Bytes; VAR offset: LONGINT): ADDRESS;
  145. VAR adr: ADDRESS; i: LONGINT;
  146. BEGIN
  147. IF ~Expect(offset < LEN(refs)) THEN RETURN 0 END;
  148. FOR i := 0 TO SIZEOF(ADDRESS)-1 DO
  149. SYSTEM.PUT8(ADDRESSOF(adr)+i, GetChar(refs, offset));
  150. END;
  151. RETURN adr;
  152. END GetAddress;
  153. (* skip an address in the byte stream *)
  154. PROCEDURE SkipAddress*(VAR offset: LONGINT);
  155. BEGIN
  156. INC(offset, SIZEOF(ADDRESS));
  157. END SkipAddress;
  158. (* consume a size in the byte stream *)
  159. PROCEDURE GetSize*(refs: Modules.Bytes; VAR offset: LONGINT): SIZE;
  160. VAR size: SIZE; i: LONGINT;
  161. BEGIN
  162. IF ~Expect(offset < LEN(refs)) THEN RETURN 0 END;
  163. FOR i := 0 TO SIZEOF(SIZE)-1 DO
  164. SYSTEM.PUT8(ADDRESSOF(size)+i, refs[offset]);
  165. INC(offset);
  166. END;
  167. RETURN size;
  168. END GetSize;
  169. (* skip a size in the byte stream *)
  170. PROCEDURE SkipSize*(VAR offset: LONGINT);
  171. BEGIN
  172. INC(offset, SIZEOF(SIZE));
  173. END SkipSize;
  174. (* consume a set in the byte stream *)
  175. PROCEDURE GetSet*(refs: Modules.Bytes; VAR offset: LONGINT): SET;
  176. VAR set: SET; i: LONGINT;
  177. BEGIN
  178. IF ~Expect(offset < LEN(refs)) THEN RETURN {} END;
  179. FOR i := 0 TO SIZEOF(SET)-1 DO
  180. SYSTEM.PUT8(ADDRESSOF(set)+i, refs[offset]);
  181. INC(offset);
  182. END;
  183. RETURN set;
  184. END GetSet;
  185. (* skip a set in the byte stream *)
  186. PROCEDURE SkipSet*(VAR offset: LONGINT);
  187. BEGIN
  188. INC(offset, SIZEOF(SET));
  189. END SkipSet;
  190. (* consume a string in the byte stream *)
  191. PROCEDURE GetString*(refs: Modules.Bytes; VAR offset: LONGINT; VAR string: ARRAY OF CHAR);
  192. VAR ch: CHAR; i: LONGINT;
  193. BEGIN
  194. i := 0;
  195. REPEAT
  196. ch := GetChar(refs, offset);
  197. string[i] := ch;
  198. INC(i);
  199. UNTIL ch = 0X;
  200. END GetString;
  201. (* skip a string in the byte stream *)
  202. PROCEDURE SkipString*(refs: Modules.Bytes; VAR offset: LONGINT);
  203. BEGIN
  204. WHILE(refs[offset] # 0X) DO INC(offset) END; INC(offset);
  205. END SkipString;
  206. (* extract a full name from the byte stream by traversing up the symbols in their scopes *)
  207. PROCEDURE GetFullName*(refs: Modules.Bytes; offset: LONGINT; VAR name: ARRAY OF CHAR);
  208. VAR n: LONGINT;
  209. PROCEDURE Traverse(offset: LONGINT);
  210. VAR c: CHAR;
  211. BEGIN
  212. IF offset >= 0 THEN
  213. c := GetChar(refs, offset);
  214. IF (c = sfProcedure) OR (c=sfVariable) OR (c=sfTypeDeclaration) OR (c=sfModule) THEN
  215. Traverse(GetSize(refs, offset));
  216. END;
  217. IF (n > 0) & (n<LEN(name)) THEN name[n] := "."; INC(n); END;
  218. WHILE (n<LEN(name)) & (refs[offset] # 0X) DO
  219. name[n] := refs[offset];
  220. INC(n); INC(offset);
  221. END;
  222. END;
  223. END Traverse;
  224. BEGIN
  225. n := 0;
  226. Traverse(offset);
  227. name[n] := 0X;
  228. END GetFullName;
  229. (* "lock free" version of Modules.ThisTypeByAdr *)
  230. PROCEDURE ThisTypeByAdr(adr: ADDRESS; VAR m: Modules.Module; VAR t: Modules.TypeDesc);
  231. BEGIN
  232. IF adr # 0 THEN
  233. SYSTEM.GET (adr + Heaps.TypeDescOffset, adr);
  234. IF CheckHeapAddress(adr) THEN
  235. t := SYSTEM.VAL(Modules.TypeDesc, adr);
  236. m := t.mod;
  237. ELSE
  238. m := NIL; t := NIL
  239. END
  240. ELSE
  241. m := NIL; t := NIL
  242. END
  243. END ThisTypeByAdr;
  244. (* output type descriptor information *)
  245. PROCEDURE WriteType*(w: Streams.Writer; adr: ADDRESS);
  246. VAR module: Modules.Module; typeDesc: Modules.TypeDesc;
  247. BEGIN
  248. IF CheckHeapAddress(adr) THEN
  249. ThisTypeByAdr(adr, module, typeDesc);
  250. IF module # NIL THEN
  251. w.String(module.name);
  252. ELSE
  253. w.String("NIL"); RETURN
  254. END;
  255. w.String(".");
  256. IF typeDesc # NIL THEN
  257. IF typeDesc.name = "" THEN
  258. w.String("ANONYMOUS")
  259. ELSE
  260. w.String(typeDesc.name);
  261. END;
  262. ELSE
  263. w.String("NIL");
  264. END;
  265. ELSE
  266. w.String("UNKNOWN");
  267. END;
  268. END WriteType;
  269. (* Write the specified procedure name and returns parameters for use with Variables *)
  270. PROCEDURE WriteProc0(w: Streams.Writer; mod: Modules.Module; pc, fp: ADDRESS; VAR refs: Modules.Bytes; VAR refpos: LONGINT; VAR base: ADDRESS);
  271. VAR ch: CHAR; startpc, end: ADDRESS; offset: LONGINT; name: Name;
  272. BEGIN
  273. refpos := -1;
  274. IF mod = NIL THEN
  275. IF pc = 0 THEN w.String("NIL")
  276. ELSE
  277. w.String("Unknown PC="); w.Address(pc); w.Char("H")
  278. END;
  279. IF fp # -1 THEN
  280. w.String(" FP="); w.Address(fp); w.Char("H")
  281. END
  282. ELSE
  283. refs := mod.refs;
  284. refpos := FindByAdr(refs, 0, pc);
  285. IF refpos >= 0 THEN
  286. offset := refpos;
  287. IF GetChar(refs, offset) = sfProcedure THEN
  288. SkipSize(offset);
  289. SkipString(refs, offset);
  290. GetFullName(refs, refpos, name);
  291. startpc := GetAddress(refs, offset);
  292. end := GetAddress(refs, offset);
  293. SkipSet(offset);
  294. w.String(name);
  295. w.Char(":"); w.Int(LONGINT(pc-startpc),1);
  296. base := fp; (*! only for local !! *)
  297. refpos := offset;
  298. END;
  299. ELSE
  300. w.String("procedure not found in module "); w.String(mod.name);
  301. END;
  302. w.String(" pc="); w.Int(LONGINT(pc),1); w.String(" ["); w.Address (pc); w.String("H]");
  303. w.String(" = "); w.Int(LONGINT(startpc),1); w.String(" + "); w.Int(LONGINT(pc-startpc),1);
  304. w.String(" crc="); w.Hex(mod.crc,-8);
  305. (*Wait(w);*)
  306. END
  307. END WriteProc0;
  308. PROCEDURE WriteBasicValue*(w: Streams.Writer; type: CHAR; adr: ADDRESS; VAR size: SIZE);
  309. VAR
  310. b: BOOLEAN;
  311. c: CHAR; c8: CHAR8; c16: CHAR16; c32: CHAR32;
  312. s: SHORTINT; i: INTEGER; l: LONGINT; h: HUGEINT;
  313. sz: SIZE; a: ADDRESS;
  314. word: WORD; lword: LONGWORD;
  315. s8: SIGNED8; s16: SIGNED16; s32: SIGNED32; s64: SIGNED64;
  316. u8: UNSIGNED8; u16: UNSIGNED16; u32: UNSIGNED32; u64: UNSIGNED64;
  317. r: REAL; x: LONGREAL;
  318. cplx: POINTER {UNSAFE} TO RECORD re,im: REAL END;
  319. lcplx: POINTER {UNSAFE} TO RECORD re,im: LONGREAL END;
  320. set: SET;
  321. byte: SYSTEM.BYTE;
  322. PROCEDURE Signed(i: HUGEINT);
  323. BEGIN
  324. w.Int(i,1);
  325. END Signed;
  326. PROCEDURE Unsigned(i: HUGEINT; size: LONGINT);
  327. BEGIN
  328. w.Hex(i,-2*size);
  329. END Unsigned;
  330. BEGIN
  331. CASE type OF
  332. | sfTypePointerToRecord, sfTypeANY, sfTypeOBJECT:
  333. size := SIZEOF(ADDRESS);
  334. SYSTEM.GET(adr, a); Unsigned(a, SIZEOF(ADDRESS));
  335. | sfTypeBOOLEAN:
  336. size := SIZEOF(BOOLEAN);
  337. SYSTEM.GET(adr, b); IF b THEN w.String("true") ELSE w.String("false") END;
  338. | sfTypeCHAR, sfTypeCHAR8:
  339. size := SIZEOF(CHAR);
  340. SYSTEM.GET(adr, c); IF (c > " ") & (c <= "~") THEN w.Char( c ); ELSE w.Hex( ORD( c ), -2 ); w.Char( "X" ) END;
  341. | sfTypeCHAR16:
  342. size := SIZEOF(CHAR16);
  343. SYSTEM.GET(adr, i); w.Hex(i,-4); w.Char("X");
  344. | sfTypeCHAR32:
  345. size := SIZEOF(CHAR32);
  346. SYSTEM.GET(adr, l); w.Hex(l,-8); w.Char("X");
  347. | sfTypeSHORTINT:
  348. size := SIZEOF(SHORTINT);
  349. SYSTEM.GET(adr, s); Signed(s);
  350. | sfTypeINTEGER:
  351. size := SIZEOF(INTEGER);
  352. SYSTEM.GET(adr, i); Signed(i);
  353. | sfTypeLONGINT:
  354. size := SIZEOF(LONGINT);
  355. SYSTEM.GET(adr, l); Signed(l);
  356. | sfTypeHUGEINT:
  357. size := SIZEOF(HUGEINT);
  358. SYSTEM.GET(adr, h); Signed(h);
  359. | sfTypeWORD:
  360. size := SIZEOF(WORD);
  361. SYSTEM.GET(adr, word); Signed(word);
  362. | sfTypeLONGWORD:
  363. size := SIZEOF(LONGWORD);
  364. SYSTEM.GET(adr, lword); Signed(lword);;
  365. | sfTypeSIGNED8:
  366. size := SIZEOF(SIGNED8);
  367. SYSTEM.GET(adr, s8); Signed(s8);
  368. | sfTypeSIGNED16:
  369. size := SIZEOF(SIGNED16);
  370. SYSTEM.GET(adr, s16); Signed(s16);
  371. | sfTypeSIGNED32:
  372. size := SIZEOF(SIGNED32);
  373. SYSTEM.GET(adr, s32); Signed(s32);
  374. | sfTypeSIGNED64:
  375. size := SIZEOF(SIGNED64);
  376. SYSTEM.GET(adr, s64); Signed(s64);
  377. | sfTypeUNSIGNED8:
  378. size := SIZEOF(UNSIGNED8);
  379. SYSTEM.GET(adr, u8); Unsigned(u8, SIZEOF(UNSIGNED8));
  380. | sfTypeUNSIGNED16:
  381. size := SIZEOF(UNSIGNED16);
  382. SYSTEM.GET(adr, u16); Unsigned(u16, SIZEOF(UNSIGNED16));
  383. | sfTypeUNSIGNED32:
  384. size := SIZEOF(UNSIGNED32);
  385. SYSTEM.GET(adr, u32); Unsigned(u32, SIZEOF(UNSIGNED32));
  386. | sfTypeUNSIGNED64:
  387. size := SIZEOF(UNSIGNED64);
  388. SYSTEM.GET(adr, s64); Unsigned(s64, SIZEOF(UNSIGNED64));
  389. | sfTypeREAL:
  390. size := SIZEOF(REAL);
  391. SYSTEM.GET(adr, r); w.Float(r,7);
  392. w.String(" (");
  393. SYSTEM.GET(adr, u32); Unsigned(u32, SIZEOF(UNSIGNED32));
  394. w.String(")");
  395. | sfTypeLONGREAL:
  396. size := SIZEOF(LONGREAL);
  397. SYSTEM.GET(adr, x); w.Float(x,13);
  398. w.String(" (");
  399. SYSTEM.GET(adr, s64); Unsigned(s64, SIZEOF(UNSIGNED64));
  400. w.String(")");
  401. | sfTypeCOMPLEX:
  402. size := SIZEOF(COMPLEX);
  403. cplx := adr; w.Float(cplx.re,7); w.String("+ i*"); w.Float(cplx.im,7);
  404. | sfTypeLONGCOMPLEX:
  405. size := SIZEOF(LONGCOMPLEX);
  406. lcplx := adr;
  407. w.Float(lcplx.re,13); w.String("+ i*"); w.Float(lcplx.im,13);
  408. | sfTypeSET:
  409. size := SIZEOF(SET);
  410. SYSTEM.GET(adr, set); w.Set(set);
  411. | sfTypeBYTE:
  412. size := SIZEOF(SYSTEM.BYTE);
  413. SYSTEM.GET(adr, c); Unsigned(ORD(c), 1);
  414. | sfTypeRANGE:
  415. size := SIZEOF(RANGE);
  416. SYSTEM.GET(adr, sz); Unsigned(sz,SIZEOF(SIZE)); w.String(".."); Unsigned(sz, SIZEOF(SIZE));
  417. | sfTypeADDRESS:
  418. size := SIZEOF(ADDRESS);
  419. SYSTEM.GET(adr, a); Unsigned(a, SIZEOF(ADDRESS));
  420. | sfTypeSIZE:
  421. size := SIZEOF(SIZE);
  422. SYSTEM.GET(adr, sz); Signed(sz); w.String("["); Unsigned(sz, SIZEOF(SIZE)); w.String("]");
  423. | sfTypeENUM:
  424. SYSTEM.GET(adr, word); Signed(word);
  425. | sfTypePORT:
  426. SYSTEM.GET(adr, a); Unsigned(a, SIZEOF(ADDRESS));
  427. ELSE
  428. w.String("UNKOWN TYPE "); Unsigned(ORD(type),1);
  429. END;
  430. w.Update;
  431. END WriteBasicValue;
  432. PROCEDURE OnHeapOrStack(adr: ADDRESS; low, high: ADDRESS): BOOLEAN;
  433. BEGIN
  434. RETURN (low <= adr) & (adr < high) OR CheckHeapAddress(adr);
  435. END OnHeapOrStack;
  436. PROCEDURE WriteValueString*(w: Streams.Writer; adr: ADDRESS; maxLen: LONGINT; low, high: ADDRESS);
  437. CONST MaxString = 96;
  438. VAR ch: CHAR;
  439. BEGIN
  440. IF maxLen > MaxString THEN maxLen := MaxString END;
  441. w.Char('"');
  442. LOOP
  443. IF (maxLen <= 0) OR ~OnHeapOrStack(adr, low, high) THEN
  444. EXIT END;
  445. SYSTEM.GET(adr, ch);
  446. IF (ch < " ") OR (ch > "~") THEN EXIT END;
  447. w.Char(ch);
  448. INC(adr);
  449. DEC(maxLen);
  450. END;
  451. w.Char('"');
  452. IF maxLen = 0 THEN w.String("..."); END;
  453. END WriteValueString;
  454. PROCEDURE WriteMathArray*(w: Streams.Writer; adr: ADDRESS; low, high: ADDRESS);
  455. TYPE
  456. LenInc = RECORD
  457. len: SIZE;
  458. inc: SIZE
  459. END;
  460. UnsafeArray= POINTER {UNSAFE} TO RECORD
  461. ptr: ANY;
  462. adr: ADDRESS;
  463. flags: SET;
  464. dim: SIZE;
  465. elementSize: SIZE;
  466. lens: ARRAY 8 OF LenInc;
  467. END;
  468. VAR t: UnsafeArray; i: SIZE;
  469. BEGIN
  470. IF adr = NIL THEN w.String(" (NIL)");
  471. ELSIF OnHeapOrStack(adr, low, high) THEN
  472. t := adr;
  473. w.String(" len/inc=[ ");
  474. IF t.dim < 8 THEN
  475. FOR i := 0 TO t.dim-1 DO
  476. w.Int(t.lens[i].len,0);
  477. w.String("/");
  478. w.Int(t.lens[i].inc,0);
  479. IF i < t.dim-1 THEN w.String(" , ") END;
  480. END;
  481. END;
  482. w.String(" ]");
  483. w.String(" flags = "); w.Set(t.flags);
  484. w.String(" adr = "); w.Address(t.adr);
  485. w.String(" ptr = "); w.Address(t.ptr);
  486. w.String(" elementSize = "); w.Int(t.elementSize,0);
  487. ELSE
  488. w.String("(UNKNOWN)");
  489. END;
  490. END WriteMathArray;
  491. PROCEDURE WriteValue*(w: Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT; adr: ADDRESS; low, high: ADDRESS);
  492. VAR type: CHAR; a: ADDRESS; size, ofs: SIZE; len,i: SIZE;
  493. BEGIN
  494. IF ~OnHeapOrStack(adr, low, high) THEN
  495. SkipType(refs, offset);
  496. w.String("INVALID ADR:"); w.Address(adr);
  497. RETURN;
  498. END;
  499. type := GetChar(refs, offset);
  500. IF type = sfTypeIndirect THEN
  501. ofs := GetSize(refs, offset);
  502. WriteValue(w, refs, ofs, adr, low, high);
  503. ELSE
  504. CASE type OF
  505. sfTypeNone:
  506. w.String("NONE");
  507. | sfTypePointerToRecord, sfTypeANY, sfTypeOBJECT:
  508. WriteBasicValue(w,type, adr, size);
  509. SYSTEM.GET(adr, a);
  510. IF a = 0 THEN
  511. w.String(" NIL");
  512. ELSIF CheckHeapAddress(a) THEN
  513. SYSTEM.GET(a + Heaps.TypeDescOffset, a);
  514. w.String(" (");
  515. WriteType(w,a);
  516. w.String(")");
  517. ELSE
  518. w.String(" (UNKNOWN)");
  519. END;
  520. | sfTypePointerToArray:
  521. WriteBasicValue(w, sfTypeANY, adr, size);
  522. SYSTEM.GET(adr, a);
  523. IF ~OnHeapOrStack(a,low,high) THEN w.String(" (INVALID)") END;
  524. SkipType(refs, offset);
  525. | sfTypeOpenArray:
  526. IF (refs[offset] = sfTypeCHAR) OR (refs[offset] = sfTypeCHAR8) THEN (* ARRAY OF CHAR *)
  527. SYSTEM.GET(adr, a);
  528. WriteValueString(w, a, MaxString, low, high);
  529. ELSE
  530. w.String("... (NO STRING)")
  531. END;
  532. SkipType(refs, offset);
  533. | sfTypeStaticArray:
  534. len := GetSize(refs, offset);
  535. IF (refs[offset] = sfTypeCHAR) OR (refs[offset] = sfTypeCHAR8) THEN (* ARRAY x OF CHAR *)
  536. WriteValueString(w, adr, len, low, high);
  537. ELSE
  538. w.String("... (NO STRING)");
  539. END;
  540. SkipType(refs, offset);
  541. | sfTypeDynamicArray:
  542. w.String("... (DYNAMIC ARRAY)");
  543. SkipType(refs, offset);
  544. | sfTypeMathOpenArray:
  545. w.String(" OPEN MATH ARRAY ");
  546. SkipType(refs, offset);
  547. WriteMathArray(w, adr, low, high);
  548. | sfTypeMathStaticArray:
  549. w.String(" ... (MATH STATIC ARRAY)");
  550. SkipSize(offset); SkipType(refs, offset);
  551. | sfTypeMathTensor:
  552. w.String(" TENSOR ");
  553. SYSTEM.GET(adr, a);
  554. WriteBasicValue(w, sfTypeANY, adr, size);
  555. WriteMathArray(w, a, low, high);
  556. SkipType(refs, offset);
  557. | sfTypeRecord:
  558. w.String("...");
  559. w.String("(");
  560. a := GetAddress(refs, offset);
  561. WriteType(w,a);
  562. w.String(")");
  563. | sfTypeDelegate:
  564. w.String(" (DELEGATE)");
  565. WHILE refs[offset] = sfVariable DO SkipVariable(refs, offset) END;
  566. SkipType(refs, offset);
  567. | sfTypePORT:
  568. WriteBasicValue(w, type, adr, size);
  569. SkipChar(offset);
  570. ELSE
  571. WriteBasicValue(w, type, adr, size);
  572. END;
  573. w.Update;
  574. END;
  575. END WriteValue;
  576. PROCEDURE WriteVariable*(w: Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT; base: ADDRESS; low, high: ADDRESS);
  577. VAR name: ARRAY 128 OF CHAR; adr: LONGINT; prevScope: SIZE; c: CHAR;
  578. BEGIN
  579. IF ~Expect(GetChar(refs, offset) = sfVariable) THEN RETURN END;
  580. prevScope := GetSize(refs, offset);
  581. GetString(refs, offset, name);
  582. w.String(Sep); w.String(name); w.Char("=");
  583. c := GetChar(refs, offset);
  584. IF c = sfRelative THEN
  585. adr := base + GetSize(refs, offset)
  586. ELSIF c = sfIndirect THEN
  587. adr := base + GetSize(refs, offset);
  588. w.Address(adr); w.String("->");
  589. IF ~OnHeapOrStack(adr,low,high) THEN w.String(" (INVALID)");
  590. SkipType(refs,offset);
  591. RETURN;
  592. END;
  593. SYSTEM.GET(adr, adr);
  594. ELSE (* absolute *)
  595. adr := GetAddress(refs, offset);
  596. END;
  597. w.String(" [@");w.Int(adr-base,1); w.String("] ");
  598. WriteValue(w, refs, offset, adr, low, high);
  599. END WriteVariable;
  600. (* write variables taking meta information from stream in stream at offset, potentially stored at base address
  601. *)
  602. PROCEDURE WriteVariables*(w: Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT; base: ADDRESS; low, high: ADDRESS);
  603. VAR count: LONGINT;
  604. BEGIN
  605. WHILE refs[offset] = sfVariable DO
  606. WriteVariable(w, refs, offset, base, low, high); w.Ln;
  607. (*INC(count); *)
  608. END;
  609. IF count > 0 THEN w.Ln; Wait(w); END;
  610. END WriteVariables;
  611. (* skip type metadata in stream *)
  612. PROCEDURE SkipType*(refs: Modules.Bytes; VAR offset: LONGINT);
  613. VAR size: SIZE; adr: LONGINT; c: CHAR;
  614. BEGIN
  615. c := GetChar(refs, offset);
  616. CASE c OF
  617. sfTypeNone .. sfTypeSIZE:
  618. | sfTypePointerToRecord:
  619. | sfTypePointerToArray:SkipType(refs, offset);
  620. | sfTypeOpenArray: SkipType(refs, offset);
  621. | sfTypeStaticArray: SkipSize(offset); SkipType(refs, offset);
  622. | sfTypeDynamicArray: SkipType(refs, offset);
  623. | sfTypeMathOpenArray: SkipType(refs, offset);
  624. | sfTypeMathStaticArray: SkipSize(offset); SkipType(refs, offset);
  625. | sfTypeMathTensor: SkipType(refs, offset);
  626. | sfTypeRecord: SkipSize(offset);
  627. | sfTypeDelegate:
  628. WHILE refs[offset] = sfVariable DO SkipVariable(refs, offset) END;
  629. SkipType(refs, offset);
  630. | sfTypeENUM:
  631. | sfTypePORT: SkipChar(offset);
  632. | sfTypeIndirect: SkipSize(offset);
  633. ELSE (* ?? *)
  634. END;
  635. END SkipType;
  636. (* skip procedure metadata in stream *)
  637. PROCEDURE SkipProcedure*(refs: Modules.Bytes; VAR offset: LONGINT);
  638. BEGIN
  639. IF ~Expect(GetChar(refs, offset) = sfProcedure) THEN RETURN END;
  640. SkipSize(offset);
  641. SkipString(refs, offset);
  642. SkipAddress(offset);
  643. SkipAddress(offset);
  644. SkipSet(offset);
  645. WHILE (refs[offset] = sfVariable) DO SkipVariable(refs, offset) END;
  646. SkipType(refs, offset);
  647. SkipScope(refs, offset);
  648. END SkipProcedure;
  649. (* skip variable metadata in stream *)
  650. PROCEDURE SkipVariable*(refs: Modules.Bytes; VAR offset: LONGINT);
  651. BEGIN
  652. IF ~Expect(GetChar(refs, offset) = sfVariable) THEN RETURN END;
  653. SkipSize(offset);
  654. SkipString(refs, offset);
  655. SkipChar(offset);
  656. SkipAddress(offset);
  657. SkipType(refs, offset);
  658. END SkipVariable;
  659. (* skip type declaration meta data in stream *)
  660. PROCEDURE SkipTypeDeclaration*(refs: Modules.Bytes; VAR offset: LONGINT);
  661. BEGIN
  662. IF ~Expect(GetChar(refs, offset) = sfTypeDeclaration) THEN RETURN END;
  663. SkipSize(offset);
  664. SkipString(refs, offset);
  665. SkipAddress(offset);
  666. IF refs[offset] = sfScopeBegin THEN SkipScope(refs, offset) END;
  667. END SkipTypeDeclaration;
  668. (* skip type declaration meta data in stream *)
  669. PROCEDURE SkipModule*(refs: Modules.Bytes; VAR offset: LONGINT);
  670. BEGIN
  671. IF ~Expect(GetChar(refs, offset) = sfModule) THEN RETURN END;
  672. SkipSize(offset);
  673. SkipString(refs, offset);
  674. IF refs[offset] = sfScopeBegin THEN SkipScope(refs, offset) END;
  675. END SkipModule;
  676. (* skip a scope in stream *)
  677. PROCEDURE SkipScope*(refs: Modules.Bytes; VAR offset: LONGINT);
  678. BEGIN
  679. IF ~Expect(GetChar(refs, offset) = sfScopeBegin) THEN RETURN END;
  680. WHILE (refs[offset] = sfVariable) DO SkipVariable(refs, offset) END;
  681. WHILE (refs[offset] = sfProcedure) DO SkipProcedure(refs, offset) END;
  682. WHILE (refs[offset] = sfTypeDeclaration) DO SkipTypeDeclaration(refs, offset) END;
  683. IF ~Expect(GetChar(refs, offset) = sfScopeEnd) THEN RETURN END;
  684. END SkipScope;
  685. TYPE
  686. Search = RECORD
  687. name: ARRAY 256 OF CHAR; (* for search by name *)
  688. nameOffset: LONGINT; (* to incrementally search through scopes *)
  689. minLevel: LONGINT; (* in order to stop scope search *)
  690. pc: ADDRESS; (* for search by address *)
  691. pos: LONGINT; (* symbol position in stream, -1 if not found *)
  692. found: BOOLEAN; (* if found *)
  693. END;
  694. (* check if stream contains the string part stored in search record with respective offset *)
  695. PROCEDURE FindString(refs: Modules.Bytes; VAR offset: LONGINT; level: LONGINT; VAR find: Search);
  696. VAR ofs: LONGINT;
  697. BEGIN
  698. IF find.minLevel > level THEN
  699. SkipString(refs, offset)
  700. ELSE
  701. ofs := find.nameOffset;
  702. WHILE (refs[offset] # 0X) & (find.name[ofs] = refs[offset]) DO
  703. INC(offset); INC(ofs);
  704. END;
  705. IF (refs[offset] = 0X) THEN
  706. IF find.name[ofs] = 0X THEN
  707. find.found := TRUE;
  708. ELSIF find.name[ofs] = "." THEN
  709. find.minLevel := level+1;
  710. find.nameOffset := ofs+1;
  711. END;
  712. END;
  713. WHILE(refs[offset] # 0X) DO INC(offset) END;
  714. INC(offset);
  715. END;
  716. END FindString;
  717. (* find a symbol by name or pc starting from the procedure stream section *)
  718. PROCEDURE FindInProcedure(refs: Modules.Bytes; VAR offset: LONGINT; level: LONGINT; VAR find: Search);
  719. VAR name: ARRAY 128 OF CHAR; start, end, pos: LONGINT;
  720. BEGIN
  721. pos := offset;
  722. IF ~Expect(GetChar(refs, offset) = sfProcedure) THEN RETURN END;
  723. SkipSize(offset);
  724. FindString(refs, offset, level, find);
  725. start := GetAddress(refs, offset);
  726. end := GetAddress(refs, offset);
  727. SkipSet(offset);
  728. find.found := find.found OR (start <= find.pc) & (find.pc < end);
  729. IF find.found THEN
  730. find.pos := pos;
  731. RETURN;
  732. END;
  733. WHILE (refs[offset] = sfVariable) DO
  734. IF find.minLevel <= level+1 THEN
  735. FindInVariable(refs, offset, level+1, find);
  736. IF find.found THEN RETURN END;
  737. ELSE
  738. SkipVariable(refs, offset)
  739. END;
  740. END;
  741. SkipType(refs, offset);
  742. FindInScope(refs, offset, level+1, find);
  743. END FindInProcedure;
  744. (* find a symbol by name or pc starting from the variable stream section *)
  745. PROCEDURE FindInVariable(refs: Modules.Bytes; VAR offset: LONGINT; level: LONGINT; VAR find: Search);
  746. VAR name: ARRAY 128 OF CHAR; pos: LONGINT;
  747. BEGIN
  748. pos := offset;
  749. IF ~Expect(GetChar(refs, offset) = sfVariable) THEN RETURN END;
  750. SkipSize(offset);
  751. FindString(refs, offset, level, find);
  752. IF find.found THEN
  753. find.pos := pos;
  754. RETURN;
  755. END;
  756. SkipChar(offset);
  757. SkipSize(offset);
  758. SkipType(refs, offset);
  759. END FindInVariable;
  760. (* find a symbol by name or pc starting from the type declaration stream section *)
  761. PROCEDURE FindInTypeDeclaration(refs: Modules.Bytes; VAR offset: LONGINT; level: LONGINT; VAR find: Search);
  762. VAR name: ARRAY 128 OF CHAR; adr, pos: LONGINT;
  763. BEGIN
  764. pos := offset;
  765. IF ~Expect(GetChar(refs, offset) = sfTypeDeclaration) THEN RETURN END;
  766. SkipSize(offset);
  767. FindString(refs, offset, level, find);
  768. IF find.found THEN
  769. find.pos := pos;
  770. RETURN;
  771. END;
  772. SkipAddress(offset);
  773. IF refs[offset] = sfScopeBegin THEN FindInScope(refs, offset, level+1, find) END;
  774. END FindInTypeDeclaration;
  775. PROCEDURE FindInModule(refs: Modules.Bytes; VAR offset: LONGINT; level: LONGINT; VAR find: Search);
  776. VAR pos: LONGINT;
  777. BEGIN
  778. pos := offset;
  779. IF ~Expect(GetChar(refs, offset) = sfModule) THEN RETURN END;
  780. SkipSize(offset);
  781. FindString(refs, offset, level, find);
  782. IF find.found THEN
  783. find.pos := pos;
  784. RETURN;
  785. END;
  786. FindInScope(refs, offset, level+1, find);
  787. END FindInModule;
  788. (* find a symbol by name or pc in a scope in the stream *)
  789. PROCEDURE FindInScope(refs: Modules.Bytes; VAR offset: LONGINT; level: LONGINT; VAR find: Search);
  790. VAR no,i: LONGINT;
  791. BEGIN
  792. IF ~Expect(GetChar(refs, offset) = sfScopeBegin) THEN RETURN END;
  793. WHILE ~find.found &(refs[offset] = sfVariable) & (find.minLevel <= level) DO (* Variable *)
  794. FindInVariable(refs, offset, level, find);
  795. END;
  796. WHILE ~find.found & (refs[offset] = sfProcedure) & (find.minLevel <= level) DO (* Procedure *)
  797. FindInProcedure(refs, offset, level, find);
  798. END;
  799. WHILE ~find.found & (refs[offset] = sfTypeDeclaration) & (find.minLevel <= level) DO (* TypeDeclaration *)
  800. FindInTypeDeclaration(refs, offset,level, find);
  801. END;
  802. IF find.found OR (find.minLevel > level) THEN RETURN END;
  803. IF ~Expect(GetChar(refs, offset) = sfScopeEnd) THEN RETURN END;
  804. END FindInScope;
  805. PROCEDURE InitSearch(VAR search: Search);
  806. BEGIN
  807. search.found := FALSE;
  808. search.pos := -1;
  809. search.name := "";
  810. search.nameOffset := 0;
  811. search.minLevel := 0;
  812. search.pc := 0;
  813. END InitSearch;
  814. (* Find a symbol in the stream starting at offset.
  815. If name is supposed to contain the referred to symbol, choose skipFirstSymbol = FALSE
  816. Example FindByName(m.refs, 0, "Reflection.FindByName", FALSE)
  817. otherwise choose skipFirstSymbol = TRUE
  818. Example FindByName(m.refs, 0, "FindByName", TRUE);
  819. *)
  820. PROCEDURE FindByName*(refs: Modules.Bytes; offset: LONGINT; CONST name: ARRAY OF CHAR; skipFirstSymbol: BOOLEAN): SIZE;
  821. VAR search: Search;
  822. BEGIN
  823. InitSearch(search);
  824. COPY(name, search.name);
  825. IF skipFirstSymbol THEN search.minLevel := 1 END;
  826. CASE refs[offset] OF
  827. sfModule: FindInModule(refs, offset, 0, search);
  828. |sfVariable: FindInVariable(refs, offset, 0, search);
  829. |sfProcedure: FindInProcedure(refs, offset, 0, search);
  830. |sfTypeDeclaration: FindInTypeDeclaration(refs, offset, 0, search);
  831. ELSE (* wrong position in stream *)
  832. END;
  833. RETURN search.pos;
  834. END FindByName;
  835. PROCEDURE FindByAdr*(refs: Modules.Bytes; offset: SIZE; pc: ADDRESS): SIZE;
  836. VAR search: Search;
  837. BEGIN
  838. InitSearch(search);
  839. search.pc := pc;
  840. CASE refs[offset] OF
  841. sfModule: FindInModule(refs, offset, 0, search);
  842. |sfVariable: FindInVariable(refs, offset, 0, search);
  843. |sfProcedure: FindInProcedure(refs, offset, 0, search);
  844. |sfTypeDeclaration: FindInTypeDeclaration(refs, offset, 0, search);
  845. ELSE (* wrong position in stream *)
  846. END;
  847. RETURN search.pos;
  848. END FindByAdr;
  849. (** service procedures *)
  850. (** Find procedure name and write it. *)
  851. PROCEDURE WriteProc*(w: Streams.Writer; pc: ADDRESS);
  852. VAR refs: Modules.Bytes; refpos: LONGINT; base: ADDRESS;
  853. BEGIN
  854. WriteProc0(w, Modules.ThisModuleByAdr0(pc), pc, -1, refs, refpos, base)
  855. END WriteProc;
  856. (** Write the state of the specified module. *)
  857. PROCEDURE ModuleState*(w: Streams.Writer; mod: Modules.Module);
  858. VAR offset: LONGINT; base: ADDRESS; refs: Modules.Bytes;
  859. BEGIN
  860. IF mod = NIL THEN RETURN END;
  861. refs := mod.refs;
  862. offset := 0;
  863. w.String("State "); w.String(mod.name); w.Char(":"); w.Ln; Wait(w);
  864. IF (GetChar(refs, offset) = sfModule) THEN
  865. SkipSize(offset);
  866. SkipString(refs, offset);
  867. IF (GetChar(refs, offset) = sfScopeBegin) THEN
  868. WriteVariables(w, refs, offset, 0, 0, 0)
  869. END;
  870. END;
  871. END ModuleState;
  872. PROCEDURE CheckBP(bp: ADDRESS): ADDRESS;
  873. VAR n: ADDRESS;
  874. BEGIN
  875. IF bp # NIL THEN
  876. SYSTEM.GET(bp, n);
  877. IF ODD(n) THEN INC(bp, SIZEOF(ADDRESS)) END;
  878. END;
  879. RETURN bp;
  880. END CheckBP;
  881. (* Display call trackback. *)
  882. PROCEDURE StackTraceBack*(w: Streams.Writer; pc, bp: ADDRESS; low,high: ADDRESS; long, overflow: BOOLEAN);
  883. VAR count,offset: LONGINT; stacklow: ADDRESS; base: ADDRESS; m: Modules.Module; refs: Modules.Bytes;
  884. BEGIN
  885. count := 0; (* frame count *)
  886. REPEAT
  887. m := Modules.ThisModuleByAdr0(pc);
  888. IF (ShowAllProcs OR (m # NIL) OR (count = 0)) & (bp # 0) & (bp >= low) & (bp < high) & (bp MOD SIZEOF(ADDRESS)=0) THEN
  889. IF CheckHeapAddress( pc ) THEN
  890. WriteProc0(w, m, pc, bp, refs, offset, base); w.Ln;Wait(w); w.Update;
  891. IF long & (~overflow OR (count > 0)) THEN (* show variables *)
  892. IF offset >= 0 THEN
  893. WriteVariables(w,refs,offset, base, low, high);
  894. SkipType(refs, offset);
  895. IF Expect(GetChar(refs, offset) = sfScopeBegin) THEN
  896. WriteVariables(w,refs,offset, base, low, high);
  897. END;
  898. END;
  899. IF (m # NIL) & (base # m.sb) & (count = 0) THEN ModuleState(w, m) END
  900. END;
  901. ELSE
  902. w.String( "Unknown external procedure, pc = " ); w.Address( pc ); w.Ln; Wait(w);
  903. END;
  904. bp := CheckBP(bp);
  905. SYSTEM.GET(bp + SIZEOF(ADDRESS), pc); (* return addr from stack *)
  906. SYSTEM.GET(bp, bp); (* follow dynamic link *)
  907. INC(count)
  908. ELSE
  909. bp := 0
  910. END;
  911. UNTIL (bp = 0) OR (count = MaxFrames);
  912. IF bp # 0 THEN w.String("...") END
  913. END StackTraceBack;
  914. (** Write a process's state in one line. *)
  915. PROCEDURE WriteProcess*(w: Streams.Writer; p: Objects.Process);
  916. VAR adr: ADDRESS; mode: LONGINT; m: Modules.Module;
  917. BEGIN
  918. IF p # NIL THEN
  919. w.Int(p.id, 5);
  920. mode := p.mode;
  921. IF (mode >= Objects.Ready) & (mode <= Objects.Terminated) THEN
  922. adr := (mode-Objects.Ready)*4;
  923. FOR adr := adr TO adr+3 DO w.Char(modes[adr]) END
  924. ELSE
  925. w.Char(" "); w.Int(mode, 1)
  926. END;
  927. w.Int(p.procID, 2);
  928. w.Int(p.priority, 2);
  929. w.Update;
  930. w.Address (SYSTEM.VAL(ADDRESS, p.obj));
  931. IF p.obj # NIL THEN
  932. SYSTEM.GET(SYSTEM.VAL(ADDRESS, p.obj) - SIZEOF(ADDRESS), adr);
  933. w.Char(":"); WriteType(w, adr)
  934. END;
  935. w.Update;
  936. w.Char(" "); WriteProc(w, p.state.PC);
  937. IF p.mode = Objects.AwaitingLock THEN
  938. adr := SYSTEM.VAL(ADDRESS, p.waitingOn);
  939. w.Address (adr);
  940. w.Update;
  941. IF adr # 0 THEN (* can be 0 when snapshot is taken *)
  942. SYSTEM.GET(adr - SIZEOF(ADDRESS), adr);
  943. IF adr = SYSTEM.TYPECODE(Modules.Module) THEN
  944. w.Char("-");
  945. m := SYSTEM.VAL(Modules.Module, adr);
  946. w.String(m.name)
  947. ELSE
  948. w.Char(":"); WriteType(w, adr)
  949. END;
  950. w.Update;
  951. END
  952. ELSIF p.mode = Objects.AwaitingCond THEN
  953. w.Char(" "); WriteProc(w, SYSTEM.VAL(ADDRESS, p.condition));
  954. w.Address (p.condFP)
  955. END;
  956. w.Char(" "); w.Set(p.flags)
  957. END
  958. END WriteProcess;
  959. (* for interface compatibility *)
  960. PROCEDURE GetVariableAdr*(fp, pc: ADDRESS; CONST name: ARRAY OF CHAR): SIZE;
  961. BEGIN
  962. RETURN -1;
  963. END GetVariableAdr;
  964. PROCEDURE GetProcedureName*(pc: ADDRESS; VAR name: ARRAY OF CHAR; VAR startpc: ADDRESS);
  965. VAR m: Modules.Module; offset: LONGINT;
  966. BEGIN
  967. name := "";
  968. m := Modules.ThisModuleByAdr0(pc);
  969. IF m # NIL THEN
  970. offset := FindByAdr(m.refs,0,pc);
  971. IF offset >= 0 THEN
  972. GetFullName(m.refs, offset, name);
  973. IF GetChar(m.refs, offset) = sfProcedure THEN
  974. SkipSize(offset);
  975. SkipString(m.refs,offset);
  976. startpc := GetAddress(m.refs, offset);
  977. END;
  978. END;
  979. END;
  980. END GetProcedureName;
  981. TYPE
  982. Variable* = RECORD
  983. adr-: ADDRESS;
  984. type-, size-, n-, tdadr-: LONGINT
  985. END;
  986. PROCEDURE FindVar*(mod: Modules.Module; CONST name: ARRAY OF CHAR; VAR v: Variable): BOOLEAN;
  987. BEGIN
  988. END FindVar;
  989. PROCEDURE WriteVar*(w: Streams.Writer; v: Variable; VAR col: LONGINT);
  990. BEGIN
  991. END WriteVar;
  992. PROCEDURE ReportType*(w:Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT);
  993. VAR size: SIZE; adr: LONGINT; c: CHAR;
  994. BEGIN
  995. c := GetChar(refs, offset);
  996. CASE c OF
  997. sfTypeNone: w.String("no type");
  998. | sfTypePointerToRecord: w.String("POINTER TO RECORD");
  999. | sfTypePointerToArray: w.String("POINTER TO"); ReportType(w, refs, offset);
  1000. | sfTypeOpenArray: w.String("ARRAY OF "); ReportType(w, refs, offset);
  1001. | sfTypeStaticArray: w.String("ARRAY "); w.Int(GetSize(refs, offset),1 ); w.String(" OF "); ReportType(w, refs, offset);
  1002. | sfTypeDynamicArray: w.String("DARRAY OF "); ReportType(w,refs, offset);
  1003. | sfTypeMathOpenArray: w.String("ARRAY [*] OF "); ReportType(w, refs, offset);
  1004. | sfTypeMathStaticArray: w.String("ARRAY ["); w.Int(GetSize(refs, offset),1); w.String("] OF "); ReportType(w, refs, offset);
  1005. | sfTypeMathTensor: w.String("ARRAY [?] OF "); ReportType(w, refs, offset);
  1006. | sfTypeRecord: w.String("RECORD "); w.Address(GetAddress(refs, offset));
  1007. | sfTypeDelegate:
  1008. w.String("PROCEDURE (");
  1009. WHILE refs[offset] = sfVariable DO ReportVariable(w, refs, offset) END;
  1010. w.String("):"); ReportType(w, refs, offset);
  1011. | sfTypeBOOLEAN: w.String("BOOLEAN");
  1012. | sfTypeCHAR: w.String("CHAR");
  1013. | sfTypeCHAR8: w.String("CHAR8");
  1014. | sfTypeCHAR16: w.String("CHAR16");
  1015. | sfTypeCHAR32: w.String("CHAR32");
  1016. | sfTypeSHORTINT: w.String("SHORTINT");
  1017. | sfTypeINTEGER: w.String("INTEGER");
  1018. | sfTypeLONGINT: w.String("LONGINT");
  1019. | sfTypeHUGEINT: w.String("HUGEINT");
  1020. | sfTypeWORD: w.String("WORD");
  1021. | sfTypeLONGWORD: w.String("LONGWORD");
  1022. | sfTypeSIGNED8: w.String("SIGNED8");
  1023. | sfTypeSIGNED16: w.String("SIGNED16");
  1024. | sfTypeSIGNED32: w.String("SIGNED32");
  1025. | sfTypeSIGNED64: w.String("SIGNED64");
  1026. | sfTypeUNSIGNED8: w.String("UNSIGNED8");
  1027. | sfTypeUNSIGNED16: w.String("UNSIGNED16");
  1028. | sfTypeUNSIGNED32: w.String("UNSIGNED32");
  1029. | sfTypeUNSIGNED64: w.String("UNSIGNED64");
  1030. | sfTypeREAL: w.String("REAL");
  1031. | sfTypeLONGREAL: w.String("LONGREAL");
  1032. | sfTypeCOMPLEX: w.String("COMPLEX");
  1033. | sfTypeLONGCOMPLEX: w.String("LONGCOMPLEX");
  1034. | sfTypeSET: w.String("SET");
  1035. | sfTypeANY: w.String("ANY");
  1036. | sfTypeOBJECT: w.String("OBJECT");
  1037. | sfTypeBYTE: w.String("BYTE");
  1038. | sfTypeRANGE: w.String("RANGE");
  1039. | sfTypeADDRESS: w.String("ADDRESS");
  1040. | sfTypeSIZE: w.String("SIZE");
  1041. | sfTypePORT: w.String("PORT"); IF GetChar(refs,offset) = sfIN THEN w.String("IN") ELSE w.String("OUT") END;
  1042. | sfTypeIndirect: w.String ("INDIRECT AT "); w.Int(GetSize(refs, offset),1);
  1043. ELSE w.String("????? TYPE ?????");
  1044. END;
  1045. END ReportType;
  1046. PROCEDURE ReportProcedure*(w: Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT);
  1047. VAR name: Name; start, end: LONGINT; flags: SET;
  1048. BEGIN
  1049. w.Int(offset,1); w.String(":");
  1050. w.String("PROCEDURE ");
  1051. IF ~Expect(GetChar(refs, offset) = sfProcedure) THEN RETURN END;
  1052. SkipSize(offset);
  1053. GetString(refs, offset, name);
  1054. w.String(name);
  1055. start := GetAddress(refs, offset);
  1056. end := GetAddress(refs, offset);
  1057. flags := GetSet(refs, offset);
  1058. IF flags # {} THEN
  1059. w.Set(flags);
  1060. END;
  1061. w.String("[@"); w.Address(start); w.String(" - "); w.Address(end); w.String("]");
  1062. w.String("("); w.Ln;
  1063. WHILE refs[offset] = sfVariable DO
  1064. ReportVariable(w, refs, offset);
  1065. END;
  1066. w.String(")");
  1067. w.String(":");
  1068. ReportType(w, refs, offset);
  1069. w.Ln;
  1070. ReportScope(w, refs, offset);
  1071. END ReportProcedure;
  1072. PROCEDURE ReportVariable*(w: Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT);
  1073. VAR name: ARRAY 128 OF CHAR; adr: ADDRESS; size: SIZE;
  1074. BEGIN
  1075. w.Int(offset,1); w.String(":");
  1076. w.String("VAR ");
  1077. IF ~Expect(GetChar(refs, offset) = sfVariable) THEN RETURN END;
  1078. SkipSize(offset);
  1079. GetString(refs, offset, name);
  1080. w.String(name);
  1081. IF GetChar(refs, offset) = sfRelative THEN
  1082. size := GetSize(refs, offset);
  1083. w.String("[@"); w.Int(size,1); w.String("]");
  1084. ELSE (* absolute *)
  1085. adr := GetAddress(refs, offset);
  1086. w.String("[@"); w.Address(adr); w.String("]");
  1087. END;
  1088. w.String(":");
  1089. ReportType(w, refs, offset);
  1090. w.Ln;
  1091. END ReportVariable;
  1092. PROCEDURE ReportTypeDeclaration*(w: Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT);
  1093. VAR name: ARRAY 128 OF CHAR; adr: LONGINT;
  1094. BEGIN
  1095. w.Int(offset,1); w.String(":");
  1096. w.String("TYPE ");
  1097. IF ~Expect(GetChar(refs, offset) = sfTypeDeclaration) THEN RETURN END;
  1098. SkipSize(offset);
  1099. GetString(refs, offset, name);
  1100. w.String(name);
  1101. adr := GetAddress(refs, offset);
  1102. w.String(" ");
  1103. w.Address(adr);
  1104. w.Ln;
  1105. IF refs[offset] = sfScopeBegin THEN ReportScope(w, refs, offset) END;
  1106. END ReportTypeDeclaration;
  1107. PROCEDURE ReportScope*(w:Streams.Writer; refs: Modules.Bytes; VAR offset: LONGINT);
  1108. BEGIN
  1109. IF ~Expect(GetChar(refs, offset) = sfScopeBegin) THEN RETURN END;
  1110. w.Int(offset,1); w.String(": Scope"); w.Ln;
  1111. WHILE (refs[offset] = sfVariable) DO (* Variable *)
  1112. ReportVariable(w, refs, offset);
  1113. END;
  1114. WHILE (refs[offset] = sfProcedure) DO (* Procedure *)
  1115. ReportProcedure(w, refs, offset);
  1116. END;
  1117. WHILE (refs[offset] = sfTypeDeclaration) DO (* TypeDeclaration *)
  1118. ReportTypeDeclaration(w, refs, offset);
  1119. END;
  1120. IF ~Expect(GetChar(refs, offset) = sfScopeEnd) THEN RETURN END;
  1121. w.String("END"); w.Ln;
  1122. END ReportScope;
  1123. PROCEDURE ReportModule*(w: Streams.Writer; refs: Modules.Bytes; offset: LONGINT);
  1124. VAR name: Name;
  1125. BEGIN
  1126. w.String("MODULE ");
  1127. IF ~Expect(GetChar(refs, offset) = sfModule) THEN RETURN END;
  1128. SkipSize(offset);
  1129. GetString(refs, offset, name);
  1130. w.String(name);
  1131. ReportScope(w, refs, offset);
  1132. END ReportModule;
  1133. PROCEDURE Report*(w:Streams.Writer; refs: Modules.Bytes; offset: LONGINT);
  1134. BEGIN
  1135. CASE refs[offset] OF
  1136. sfModule: ReportModule(w, refs, offset);
  1137. |sfVariable: ReportVariable(w, refs, offset);
  1138. |sfProcedure: ReportProcedure(w, refs, offset);
  1139. |sfTypeDeclaration: ReportTypeDeclaration(w, refs, offset);
  1140. ELSE (* wrong position in stream *)
  1141. END;
  1142. END Report;
  1143. BEGIN
  1144. modes := " rdy run awl awc awe rip"; (* 4 characters per mode from Objects.Ready to Objects.Terminated *)
  1145. END Reflection.
  1146. StaticLinker.Link --fileFormat=PE32 --fileName=A2.exe --extension=GofW --displacement=401000H Runtime Trace Kernel32 Machine Heaps Modules Objects Kernel KernelLog Streams Commands Files WinFS Clock Dates Reals Strings Diagnostics BitSets StringPool ObjectFile GenericLinker Reflection GenericLoader BootConsole ~