Generic.Reflection.Mod 33 KB

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