Generic.Reflection.Mod 31 KB

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