FoxPrintout.Mod 55 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015
  1. MODULE FoxPrintout; (** AUTHOR "fof & fn"; PURPOSE "Oberon Compiler Module Output for SymbolFile, Pretty Printing and Testing"; *)
  2. (* (c) fof ETHZ 2009 *)
  3. IMPORT
  4. Scanner := FoxScanner, SyntaxTree := FoxSyntaxTree, Global := FoxGlobal, Basic := FoxBasic, Streams, D:=Debugging, SYSTEM;
  5. CONST
  6. (* print modes *)
  7. Exported*=0; SymbolFile*=1; SourceCode*=2; All*=3;
  8. TYPE
  9. Printer*= OBJECT (SyntaxTree.Visitor)
  10. VAR
  11. w-: Basic.Writer; mode: LONGINT; singleStatement: BOOLEAN;
  12. currentScope: SyntaxTree.Scope; ws: Streams.StringWriter;
  13. info: BOOLEAN; case: LONGINT;
  14. useCase: BOOLEAN; (* TRUE to enable case conversion in "Identifier" *)
  15. alertCount, commentCount: LONGINT;
  16. PROCEDURE Small(CONST name: ARRAY OF CHAR; VAR result: ARRAY OF CHAR);
  17. VAR ch: CHAR; i: LONGINT;
  18. BEGIN
  19. i := 0;
  20. REPEAT
  21. ch := name[i];
  22. IF (ch >= 'A') & (ch <= 'Z') THEN
  23. ch := CHR(ORD(ch)-ORD('A')+ORD('a'));
  24. END;
  25. result[i] := ch; INC(i);
  26. UNTIL ch = 0X;
  27. END Small;
  28. PROCEDURE Big(CONST name: ARRAY OF CHAR; VAR result: ARRAY OF CHAR);
  29. VAR ch: CHAR; i: LONGINT;
  30. BEGIN
  31. i := 0;
  32. REPEAT
  33. ch := name[i];
  34. IF (ch >= 'a') & (ch <= 'z') THEN
  35. ch := CHR(ORD(ch)-ORD('a')+ORD('A'));
  36. END;
  37. result[i] := ch; INC(i);
  38. UNTIL ch = 0X;
  39. END Big;
  40. PROCEDURE Keyword(CONST a: ARRAY OF CHAR);
  41. VAR str: ARRAY 64 OF CHAR;
  42. BEGIN
  43. IF case= Scanner.Lowercase THEN Small(a,str) ELSE COPY(a,str) END;
  44. w.BeginKeyword;
  45. w.String(str);
  46. w.EndKeyword;
  47. END Keyword;
  48. PROCEDURE AlertString(CONST s: ARRAY OF CHAR);
  49. BEGIN
  50. w.BeginAlert; w.String(s); w.EndAlert;
  51. END AlertString;
  52. PROCEDURE Indent;
  53. BEGIN w.Ln;
  54. END Indent;
  55. PROCEDURE Identifier*(x: SyntaxTree.Identifier);
  56. VAR str: Scanner.IdentifierString;
  57. BEGIN
  58. Basic.GetString(x,str);
  59. IF useCase THEN
  60. IF case = Scanner.Lowercase THEN Small(str,str); ELSE Big(str,str); END;
  61. END;
  62. w.String(str);
  63. END Identifier;
  64. PROCEDURE QualifiedIdentifier*(x: SyntaxTree.QualifiedIdentifier);
  65. BEGIN
  66. IF x.prefix # SyntaxTree.invalidIdentifier THEN Identifier(x.prefix); w.String("."); END;
  67. Identifier(x.suffix);
  68. END QualifiedIdentifier;
  69. PROCEDURE Type*(x: SyntaxTree.Type);
  70. BEGIN
  71. IF x= NIL THEN
  72. AlertString("nil type");
  73. ELSE
  74. x.Accept(SELF);
  75. END;
  76. END Type;
  77. PROCEDURE VisitType(x: SyntaxTree.Type);
  78. BEGIN
  79. IF x = SyntaxTree.importType THEN w.String("importType")
  80. ELSIF x = SyntaxTree.typeDeclarationType THEN w.String("typeDeclarationType");
  81. ELSE
  82. AlertString("InvalidType");
  83. END;
  84. END VisitType;
  85. PROCEDURE VisitBasicType(x: SyntaxTree.BasicType);
  86. BEGIN
  87. IF x.typeDeclaration # NIL THEN
  88. Identifier(x.typeDeclaration.name);
  89. ELSE
  90. Identifier(x.name);
  91. END
  92. END VisitBasicType;
  93. PROCEDURE VisitBooleanType(x: SyntaxTree.BooleanType);
  94. BEGIN
  95. VisitBasicType(x);
  96. END VisitBooleanType;
  97. PROCEDURE VisitSetType(x: SyntaxTree.SetType);
  98. BEGIN
  99. VisitBasicType(x);
  100. END VisitSetType;
  101. PROCEDURE VisitSizeType(x: SyntaxTree.SizeType);
  102. BEGIN
  103. VisitBasicType(x);
  104. END VisitSizeType;
  105. PROCEDURE VisitCharacterType(x: SyntaxTree.CharacterType);
  106. BEGIN
  107. VisitBasicType(x);
  108. END VisitCharacterType;
  109. PROCEDURE VisitIntegerType(x: SyntaxTree.IntegerType);
  110. BEGIN
  111. VisitBasicType(x);
  112. END VisitIntegerType;
  113. PROCEDURE VisitFloatType(x: SyntaxTree.FloatType);
  114. BEGIN
  115. VisitBasicType(x);
  116. END VisitFloatType;
  117. PROCEDURE VisitComplexType(x: SyntaxTree.ComplexType);
  118. BEGIN
  119. VisitBasicType(x);
  120. END VisitComplexType;
  121. PROCEDURE VisitByteType(x: SyntaxTree.ByteType);
  122. BEGIN
  123. VisitBasicType(x);
  124. END VisitByteType;
  125. PROCEDURE VisitQualifiedType(x: SyntaxTree.QualifiedType);
  126. BEGIN
  127. IF x.resolved = SyntaxTree.invalidType THEN
  128. AlertString("(*unresolved*)");
  129. END;
  130. IF x.qualifiedIdentifier # NIL THEN
  131. (* Problem: how to distinguish betwteen type aliases, e.g. Status = LONGINT and actual use of LONGINT?
  132. This tries to use scope level: if the type is declared in the global scope, it should be a basic type use. *)
  133. IF x.resolved # NIL THEN
  134. useCase := (x.resolved IS SyntaxTree.BasicType) & (x.scope.Level() = 0);
  135. END;
  136. QualifiedIdentifier(x.qualifiedIdentifier);
  137. useCase := FALSE;
  138. ELSE
  139. AlertString("NIL (* missing qualified identifier *)");
  140. END;
  141. END VisitQualifiedType;
  142. PROCEDURE VisitStringType(x: SyntaxTree.StringType);
  143. BEGIN
  144. w.String("STRING"); w.String("(* len = "); w.Int(x.length,1); w.String(" *)");
  145. END VisitStringType;
  146. PROCEDURE VisitEnumerationType(x: SyntaxTree.EnumerationType);
  147. VAR e: SyntaxTree.Constant; first: BOOLEAN;
  148. BEGIN
  149. Keyword("ENUM ");
  150. IF x.enumerationBase # NIL THEN
  151. w.String("(");
  152. Type(x.enumerationBase);
  153. w.String(") ");
  154. END;
  155. e := x.enumerationScope.firstConstant; first := TRUE;
  156. WHILE (e # NIL) DO
  157. IF ~first THEN w.String(", ") ELSE first := FALSE END;
  158. VisitConstant(e);
  159. e := e.nextConstant;
  160. END;
  161. Keyword(" END");
  162. END VisitEnumerationType;
  163. PROCEDURE VisitRangeType(x: SyntaxTree.RangeType);
  164. BEGIN VisitBasicType(x);
  165. END VisitRangeType;
  166. PROCEDURE VisitArrayType(x: SyntaxTree.ArrayType);
  167. BEGIN
  168. Keyword("ARRAY " );
  169. IF x.length # NIL THEN Expression(x.length);
  170. w.String( " " ); END;
  171. Keyword("OF " );
  172. Type(x.arrayBase);
  173. END VisitArrayType;
  174. PROCEDURE VisitNilType(x: SyntaxTree.NilType);
  175. BEGIN
  176. w.String("NILTYPE");
  177. END VisitNilType;
  178. PROCEDURE VisitAddressType(x: SyntaxTree.AddressType);
  179. BEGIN
  180. w.String("ADDRESSTYPE");
  181. END VisitAddressType;
  182. PROCEDURE VisitObjectType(x: SyntaxTree.ObjectType);
  183. BEGIN
  184. VisitBasicType(x);
  185. END VisitObjectType;
  186. PROCEDURE VisitAnyType(x: SyntaxTree.AnyType);
  187. BEGIN
  188. VisitBasicType(x);
  189. END VisitAnyType;
  190. PROCEDURE VisitMathArrayType(x: SyntaxTree.MathArrayType);
  191. BEGIN
  192. Keyword("ARRAY " );
  193. IF x.form = SyntaxTree.Tensor THEN w.String("[?] ");
  194. ELSE
  195. w.String("[");
  196. IF x.length = NIL THEN
  197. w.String("*")
  198. ELSE
  199. Expression(x.length);
  200. END;
  201. WHILE(x.arrayBase # NIL) & (x.arrayBase IS SyntaxTree.MathArrayType) DO
  202. x := x.arrayBase(SyntaxTree.MathArrayType);
  203. w.String(", ");
  204. IF x.length = NIL THEN
  205. w.String("*")
  206. ELSE
  207. Expression(x.length);
  208. END;
  209. END;
  210. w.String("] ");
  211. END;
  212. IF x.arrayBase # NIL THEN
  213. Keyword("OF " );
  214. Type(x.arrayBase);
  215. END;
  216. END VisitMathArrayType;
  217. PROCEDURE PointerFlags(x: SyntaxTree.PointerType);
  218. VAR first: BOOLEAN;
  219. BEGIN
  220. first := TRUE;
  221. IF x.isUnsafe THEN Flag(Global.NameUnsafe,first) END;
  222. IF x.isRealtime THEN Flag(Global.NameRealtime,first) END;
  223. IF x.isDisposable THEN Flag(Global.NameDisposable,first) END;
  224. IF x.isPlain THEN Flag(Global.NamePlain,first) END;
  225. FlagEnd(first);
  226. END PointerFlags;
  227. PROCEDURE ObjectFlags ( rec: SyntaxTree.RecordType; x: SyntaxTree.PointerType);
  228. VAR first: BOOLEAN;
  229. BEGIN
  230. first := TRUE;
  231. IF x.isUnsafe THEN Flag(Global.NameUnsafe,first) END;
  232. IF x.isRealtime THEN Flag(Global.NameRealtime,first) END;
  233. IF x.isDisposable THEN Flag(Global.NameDisposable,first) END;
  234. IF x.isPlain THEN Flag(Global.NamePlain,first) END;
  235. IF rec.IsProtected() THEN Flag(Global.NameExclusive, first) END;
  236. FlagEnd(first);
  237. END ObjectFlags;
  238. PROCEDURE VisitPointerType(x: SyntaxTree.PointerType);
  239. VAR pointerBase: SyntaxTree.Type;
  240. BEGIN
  241. IF x.pointerBase = NIL THEN
  242. w.BeginAlert; Keyword("POINTER TO NIL"); w.EndAlert;
  243. ELSE
  244. pointerBase := x.pointerBase;
  245. IF x.isHidden THEN
  246. Type(x.pointerBase);
  247. ELSIF (pointerBase IS SyntaxTree.RecordType) & (pointerBase(SyntaxTree.RecordType).isObject) THEN
  248. VisitRecordType(pointerBase(SyntaxTree.RecordType))
  249. ELSE
  250. Keyword("POINTER "); PointerFlags(x); Keyword("TO " ); Type(x.pointerBase);
  251. END;
  252. END;
  253. END VisitPointerType;
  254. PROCEDURE VisitPortType(x: SyntaxTree.PortType);
  255. BEGIN
  256. Keyword("PORT");
  257. IF x.direction = SyntaxTree.OutPort THEN
  258. Keyword(" OUT")
  259. ELSE
  260. ASSERT(x.direction = SyntaxTree.InPort);
  261. Keyword(" IN");
  262. END;
  263. IF x.sizeExpression # NIL THEN
  264. w.String(" ("); Expression(x.sizeExpression); w.String(")");
  265. END;
  266. END VisitPortType;
  267. PROCEDURE VisitCellType(x: SyntaxTree.CellType);
  268. BEGIN
  269. IF x.isCellNet THEN
  270. Keyword("CELLNET ")
  271. ELSE
  272. Keyword("CELL ");
  273. END;
  274. Modifiers(x.modifiers);
  275. IF x.firstParameter # NIL THEN ParameterList(x.firstParameter) END;
  276. Scope(x.cellScope);
  277. IF (x.cellScope IS SyntaxTree.CellScope) & (x.cellScope(SyntaxTree.CellScope).bodyProcedure # NIL) THEN
  278. Body(x.cellScope(SyntaxTree.CellScope).bodyProcedure.procedureScope.body)
  279. END;
  280. Indent; Keyword("END ");
  281. IF (x.typeDeclaration # NIL) THEN
  282. Identifier(x.typeDeclaration.name);
  283. END;
  284. END VisitCellType;
  285. PROCEDURE VisitRecordType(x: SyntaxTree.RecordType);
  286. VAR prevScope: SyntaxTree.Scope;
  287. BEGIN
  288. IF x.isObject THEN
  289. Keyword("OBJECT ");
  290. IF x.pointerType # NIL THEN ObjectFlags(x, x.pointerType) END;
  291. IF info THEN
  292. BeginComment; w.String("ObjectType");
  293. IF x.HasArrayStructure() THEN
  294. w.String(" (array structure: ");
  295. VisitMathArrayType(x.arrayStructure);
  296. w.String(")");
  297. END;
  298. EndComment;
  299. END;
  300. IF (x.baseType # NIL) THEN
  301. w.String( "(" );
  302. IF (x.baseType IS SyntaxTree.RecordType) & (x.baseType(SyntaxTree.RecordType).pointerType # NIL) THEN
  303. Type(x.baseType(SyntaxTree.RecordType).pointerType)
  304. ELSE
  305. Type(x.baseType);
  306. END;
  307. w.String( ")" );
  308. END;
  309. Scope(x.recordScope);
  310. IF (x.recordScope.bodyProcedure # NIL) THEN
  311. Body(x.recordScope.bodyProcedure.procedureScope.body)
  312. END;
  313. Indent; Keyword("END ");
  314. IF (x.pointerType # NIL) & (x.pointerType.typeDeclaration # NIL) THEN
  315. Identifier(x.pointerType.typeDeclaration.name);
  316. END;
  317. ELSE
  318. Keyword("RECORD ");
  319. IF (x.baseType # NIL) THEN
  320. w.String( "(" );
  321. IF (x.baseType IS SyntaxTree.RecordType) & (x.baseType(SyntaxTree.RecordType).pointerType # NIL) THEN
  322. Type(x.baseType(SyntaxTree.RecordType).pointerType)
  323. ELSE
  324. Type(x.baseType);
  325. END;
  326. w.String( ")" );
  327. END;
  328. prevScope := currentScope;
  329. currentScope := x.recordScope;
  330. VariableList(x.recordScope.firstVariable);
  331. currentScope := prevScope;
  332. Indent; Keyword("END" );
  333. END;
  334. END VisitRecordType;
  335. PROCEDURE Flags((*flags: SET*));
  336. (*
  337. VAR first: BOOLEAN;
  338. PROCEDURE Flag(CONST name: ARRAY OF CHAR);
  339. BEGIN
  340. IF first THEN w.String("{"); first := FALSE; ELSE w.String(", ") END;
  341. w.String(name);
  342. END Flag;
  343. BEGIN
  344. first := TRUE;
  345. IF SyntaxTree.ActiveFlag IN flags THEN Flag("ACTIVE") END;
  346. IF SyntaxTree.ExclusiveFlag IN flags THEN Flag("EXCLUSIVE") END;
  347. IF SyntaxTree.SafeFlag IN flags THEN Flag("SAFE") END;
  348. IF SyntaxTree.RealtimeFlag IN flags THEN Flag("REALTIME") END;
  349. IF SyntaxTree.WinAPIFlag IN flags THEN Flag("WINAPI") END;
  350. IF SyntaxTree.CFlag IN flags THEN Flag("C") END;
  351. IF SyntaxTree.DelegateFlag IN flags THEN Flag("DELEGATE") END;
  352. IF SyntaxTree.UntracedFlag IN flags THEN Flag("UNTRACED") END;
  353. IF ~first THEN w.String("} ") END;
  354. IF info THEN
  355. BeginComment;
  356. IF SyntaxTree.AccessedFlag IN flags THEN Flag("ACCESSED") END;
  357. IF SyntaxTree.WrittenFlag IN flags THEN Flag("WRITTEN") END;
  358. IF SyntaxTree.UnreachableFlag IN flags THEN Flag("UNREACHABLE") END;
  359. IF ~first THEN w.String("}") END;
  360. EndComment;
  361. END;
  362. *)
  363. END Flags;
  364. PROCEDURE Flag(identifier: SyntaxTree.Identifier; VAR first: BOOLEAN);
  365. VAR name: SyntaxTree.IdentifierString;
  366. BEGIN
  367. IF first THEN w.String("{") ELSE w.String(", ") END;
  368. first := FALSE;
  369. Basic.GetString(identifier,name);
  370. w.String(name);
  371. END Flag;
  372. PROCEDURE FlagEnd(first: BOOLEAN);
  373. BEGIN
  374. IF ~first THEN w.String("} ") END;
  375. END FlagEnd;
  376. PROCEDURE Value(identifier: SyntaxTree.Identifier; value: LONGINT; VAR first: BOOLEAN);
  377. BEGIN
  378. Flag(identifier,first);
  379. w.String("("); w.Int(value,1); w.String(")");
  380. END Value;
  381. PROCEDURE VisitProcedureType(x: SyntaxTree.ProcedureType);
  382. VAR first: BOOLEAN;
  383. BEGIN
  384. Keyword("PROCEDURE " );
  385. first := TRUE;
  386. IF x.isDelegate THEN Flag(Global.NameDelegate,first) END;
  387. IF x.isInterrupt THEN Flag(Global.NameInterrupt,first) END;
  388. IF x.noPAF THEN Flag(Global.NameNoPAF,first) END;
  389. IF x.noReturn THEN Flag(Global.NameNoReturn,first) END;
  390. IF x.callingConvention = SyntaxTree.WinAPICallingConvention THEN
  391. Flag(Global.NameWinAPI,first)
  392. ELSIF x.callingConvention = SyntaxTree.CCallingConvention THEN
  393. Flag(Global.NameC,first)
  394. END;
  395. IF x.stackAlignment > 1 THEN Value(Global.NameStackAligned,x.stackAlignment,first) END;
  396. IF ~first THEN w.String("}") END;
  397. IF (x.modifiers # NIL) & info THEN
  398. BeginComment;
  399. Modifiers(x.modifiers);
  400. EndComment;
  401. END;
  402. (*
  403. CallingConvention(x.callingConvention);
  404. IF x.isDelegate THEN w.String("{DELEGATE}") END;
  405. *)
  406. IF (x.firstParameter # NIL) OR (x.returnType # NIL) THEN
  407. ParameterList(x.firstParameter)
  408. END;
  409. IF x.returnType # NIL THEN
  410. w.String( ":" );
  411. IF x.hasUntracedReturn THEN
  412. first := TRUE;
  413. Flag(Global.NameUntraced, first);
  414. FlagEnd(first);
  415. END;
  416. Type(x.returnType)
  417. END;
  418. IF info & (x.returnParameter # NIL) THEN
  419. BeginComment;
  420. VisitParameter(x.returnParameter);
  421. EndComment;
  422. END;
  423. END VisitProcedureType;
  424. (*** expressions ****)
  425. PROCEDURE ExpressionList(x: SyntaxTree.ExpressionList);
  426. VAR i: LONGINT; expression: SyntaxTree.Expression;
  427. BEGIN
  428. FOR i := 0 TO x.Length() - 1 DO
  429. expression := x.GetExpression( i ); Expression(expression);
  430. IF i < x.Length() - 1 THEN w.String( ", " ); END;
  431. END;
  432. END ExpressionList;
  433. PROCEDURE Expression*(x: SyntaxTree.Expression);
  434. BEGIN
  435. IF x = NIL THEN
  436. AlertString("nil expression");
  437. ELSE
  438. x.Accept(SELF);
  439. IF info & (x.resolved # NIL) & (x.resolved # x) THEN
  440. BeginComment; w.String("value = "); Expression(x.resolved); EndComment;
  441. END;
  442. END;
  443. w.Update;
  444. END Expression;
  445. PROCEDURE VisitExpression(x: SyntaxTree.Expression);
  446. BEGIN
  447. AlertString("InvalidExpression");
  448. END VisitExpression;
  449. PROCEDURE VisitSet(x: SyntaxTree.Set);
  450. BEGIN
  451. w.String( "{" ); ExpressionList(x.elements); w.String( "}" );
  452. END VisitSet;
  453. PROCEDURE VisitMathArrayExpression(x: SyntaxTree.MathArrayExpression);
  454. BEGIN
  455. w.String( "[" ); ExpressionList(x.elements); w.String( "]" );
  456. END VisitMathArrayExpression;
  457. PROCEDURE VisitUnaryExpression(x: SyntaxTree.UnaryExpression);
  458. VAR identifier: SyntaxTree.Identifier;
  459. BEGIN
  460. w.String(" ");
  461. IF x.operator = Scanner.Transpose THEN
  462. identifier := Global.GetIdentifier(x.operator,case);
  463. Expression(x.left);
  464. Identifier(identifier);
  465. ELSIF (x.operator = Scanner.Address) OR (x.operator = Scanner.Size) OR (x.operator = Scanner.Alias) THEN
  466. identifier := Global.GetIdentifier(x.operator,case);
  467. Identifier(identifier);
  468. Keyword(" OF ");
  469. Expression(x.left);
  470. ELSE
  471. identifier := Global.GetIdentifier(x.operator,case);
  472. Identifier(identifier);
  473. Expression(x.left);
  474. END;
  475. END VisitUnaryExpression;
  476. PROCEDURE VisitBinaryExpression(x: SyntaxTree.BinaryExpression);
  477. VAR identifier: SyntaxTree.Identifier;
  478. BEGIN
  479. w.String( "(" );
  480. Expression(x.left);
  481. identifier := Global.GetIdentifier(x.operator,case);
  482. w.String(" "); Identifier(identifier); w.String(" ");
  483. Expression(x.right);
  484. w.String(")");
  485. END VisitBinaryExpression;
  486. PROCEDURE VisitRangeExpression(x: SyntaxTree.RangeExpression);
  487. BEGIN
  488. IF x.missingFirst & x.missingLast & x.missingStep THEN
  489. (* open range expression *)
  490. (* the surrounding spaces prevent the asterisk from being next to a parenthesis,
  491. which could be confused with the beginning or end of a comment *)
  492. w.String(" * ")
  493. ELSE
  494. IF ~x.missingFirst THEN Expression(x.first) END;
  495. w.String(" .. ");
  496. IF ~x.missingLast THEN Expression(x.last) END;
  497. IF ~x.missingStep THEN
  498. Keyword(" BY ");
  499. Expression(x.step)
  500. END
  501. END;
  502. IF info THEN
  503. BeginComment;
  504. w.String("<RangeExpression:");
  505. ShortType(x.type);
  506. w.String(">");
  507. EndComment
  508. END
  509. END VisitRangeExpression;
  510. PROCEDURE VisitTensorRangeExpression(x: SyntaxTree.TensorRangeExpression);
  511. BEGIN
  512. w.String(" ? ");
  513. END VisitTensorRangeExpression;
  514. PROCEDURE VisitConversion(x: SyntaxTree.Conversion);
  515. BEGIN
  516. IF x.typeExpression # NIL THEN Expression(x.typeExpression); w.String("(");
  517. ELSIF info THEN BeginComment; ShortType(x.type); w.String("<-"); EndComment;
  518. END;
  519. Expression(x.expression);
  520. IF x.typeExpression # NIL THEN w.String(")") END;
  521. END VisitConversion;
  522. PROCEDURE VisitDesignator(x: SyntaxTree.Designator);
  523. BEGIN
  524. AlertString("InvalidDesignator");
  525. END VisitDesignator;
  526. PROCEDURE VisitIdentifierDesignator(x: SyntaxTree.IdentifierDesignator);
  527. BEGIN
  528. IF info THEN AlertString("(*<IdentifierDesignator>*)") END;
  529. Identifier(x.identifier)
  530. END VisitIdentifierDesignator;
  531. PROCEDURE VisitSelectorDesignator(x: SyntaxTree.SelectorDesignator);
  532. BEGIN
  533. Expression(x.left);
  534. w.String(".");
  535. IF info THEN AlertString("(*<SelectorDesignator>*)") END;
  536. Identifier(x.identifier);
  537. END VisitSelectorDesignator;
  538. PROCEDURE VisitBracketDesignator(x: SyntaxTree.BracketDesignator);
  539. BEGIN
  540. Expression(x.left);
  541. IF info THEN AlertString("(*<BracketDesignator>*)") END;
  542. w.String("["); ExpressionList(x.parameters); w.String("]");
  543. END VisitBracketDesignator;
  544. PROCEDURE VisitParameterDesignator(x: SyntaxTree.ParameterDesignator);
  545. BEGIN
  546. Expression(x.left);
  547. IF info THEN AlertString("(*<ParameterDesignator>*)") END;
  548. w.String("("); ExpressionList(x.parameters); w.String(")");
  549. END VisitParameterDesignator;
  550. PROCEDURE VisitIndexDesignator(x: SyntaxTree.IndexDesignator);
  551. BEGIN
  552. Expression(x.left);
  553. w.String("["); ExpressionList(x.parameters); w.String("]");
  554. IF info THEN
  555. BeginComment;
  556. w.String("<IndexDesignator:");
  557. ShortType(x.type);
  558. w.String(">");
  559. EndComment
  560. END;
  561. END VisitIndexDesignator;
  562. PROCEDURE VisitArrowDesignator(x: SyntaxTree.ArrowDesignator);
  563. BEGIN
  564. Expression(x.left);
  565. IF info THEN AlertString("(*<ArrowDesignator>*)") END;
  566. w.String( "^" );
  567. END VisitArrowDesignator;
  568. PROCEDURE ShortType(x: SyntaxTree.Type); (* for debug information, to prevent recursion *)
  569. BEGIN
  570. IF x = NIL THEN w.String("NIL TYPE")
  571. ELSIF x IS SyntaxTree.QualifiedType THEN Type(x)
  572. ELSIF x IS SyntaxTree.BasicType THEN Type(x)
  573. ELSIF x IS SyntaxTree.ProcedureType THEN w.String("ProcedureType:");ShortType(x(SyntaxTree.ProcedureType).returnType);
  574. ELSE w.String("(other)") END;
  575. END ShortType;
  576. PROCEDURE VisitSymbolDesignator(x: SyntaxTree.SymbolDesignator);
  577. BEGIN
  578. IF (x.left # NIL) & ~x.left.isHidden THEN
  579. Expression(x.left); w.String(".");
  580. END;
  581. IF x.symbol IS SyntaxTree.Operator THEN
  582. w.String('"'); Identifier(x.symbol.name); w.String('"');
  583. ELSE
  584. useCase :=
  585. (x.symbol IS SyntaxTree.Builtin)
  586. OR ((x.symbol IS SyntaxTree.TypeDeclaration) & (x.symbol(SyntaxTree.TypeDeclaration).declaredType IS SyntaxTree.BasicType))
  587. OR (x.symbol IS SyntaxTree.Module) & ((x.symbol.name = Global.systemName) OR (x.symbol.name = Global.SystemName));
  588. Identifier(x.symbol.name);
  589. useCase := FALSE;
  590. END;
  591. IF info THEN
  592. BeginComment;
  593. w.String("<SymbolDesignator:");
  594. ShortType(x.symbol.type);
  595. w.String(">");
  596. EndComment
  597. END;
  598. END VisitSymbolDesignator;
  599. PROCEDURE VisitSupercallDesignator(x: SyntaxTree.SupercallDesignator);
  600. BEGIN
  601. Expression(x.left);
  602. w.String( "^" );
  603. IF info THEN
  604. BeginComment;
  605. w.String("<SupercallDesignator:");
  606. ShortType(x.type);
  607. w.String(">");
  608. EndComment
  609. END;
  610. END VisitSupercallDesignator;
  611. PROCEDURE VisitSelfDesignator(x: SyntaxTree.SelfDesignator);
  612. BEGIN
  613. ASSERT(x.left = NIL);
  614. IF case = Scanner.Lowercase THEN w.String("self"); ELSE w.String("SELF"); END;
  615. IF info THEN
  616. BeginComment;
  617. w.String("<SelfDesignator:");
  618. ShortType(x.type);
  619. w.String(">");
  620. EndComment
  621. END;
  622. END VisitSelfDesignator;
  623. PROCEDURE VisitResultDesignator(x: SyntaxTree.ResultDesignator);
  624. BEGIN
  625. ASSERT(x.left = NIL);
  626. w.String("RESULT");
  627. IF info THEN
  628. BeginComment;
  629. w.String("<ResultDesignator:");
  630. ShortType(x.type);
  631. w.String(">");
  632. EndComment
  633. END;
  634. END VisitResultDesignator;
  635. PROCEDURE VisitDereferenceDesignator(x: SyntaxTree.DereferenceDesignator);
  636. BEGIN
  637. Expression(x.left);
  638. w.String( "^" );
  639. IF info THEN
  640. BeginComment;
  641. w.String("<DereferenceDesignator:");
  642. ShortType(x.type);
  643. w.String(">");
  644. EndComment
  645. END;
  646. END VisitDereferenceDesignator;
  647. PROCEDURE VisitTypeGuardDesignator(x: SyntaxTree.TypeGuardDesignator);
  648. BEGIN
  649. Expression(x.left);
  650. IF info THEN
  651. BeginComment;
  652. w.String("<TypeGuardDesignator:");
  653. ShortType(x.type);
  654. w.String(">");
  655. EndComment
  656. END;
  657. w.String("(");
  658. IF x.typeExpression # NIL THEN Expression(x.typeExpression) ELSE Type(x.type) END;
  659. w.String(")");
  660. END VisitTypeGuardDesignator;
  661. PROCEDURE VisitProcedureCallDesignator(x: SyntaxTree.ProcedureCallDesignator);
  662. BEGIN
  663. Expression(x.left);
  664. IF info THEN
  665. BeginComment;
  666. w.String("<ProcedureCallDesignator:");
  667. ShortType(x.type);
  668. w.String(">");
  669. EndComment
  670. END;
  671. w.String("("); ExpressionList(x.parameters); w.String(")");
  672. END VisitProcedureCallDesignator;
  673. PROCEDURE VisitStatementDesignator(x: SyntaxTree.StatementDesignator);
  674. BEGIN
  675. Indent; Keyword("STATEMENT-DESIGNATOR ");
  676. IF x.result # NIL THEN
  677. Keyword("RETURNS ");
  678. Expression(x.result)
  679. END;
  680. Indent; Statement(x.statement);
  681. END VisitStatementDesignator;
  682. PROCEDURE VisitBuiltinCallDesignator(x: SyntaxTree.BuiltinCallDesignator);
  683. BEGIN
  684. IF x.left # NIL THEN
  685. Expression(x.left);
  686. ELSE
  687. w.String("BUILTIN(");
  688. w.Int(x.id,1);
  689. w.String(")");
  690. END;
  691. IF info THEN
  692. BeginComment;
  693. w.String("<BuiltinCallDesignator:");
  694. ShortType(x.type);
  695. w.String(">");
  696. EndComment
  697. END;
  698. w.String("("); ExpressionList(x.parameters); w.String(")");
  699. END VisitBuiltinCallDesignator;
  700. PROCEDURE VisitValue(x: SyntaxTree.Value);
  701. BEGIN
  702. AlertString("InvalidValue");
  703. END VisitValue;
  704. PROCEDURE VisitBooleanValue(x: SyntaxTree.BooleanValue);
  705. BEGIN
  706. IF Scanner.Uppercase = case THEN
  707. IF x.value THEN w.String("TRUE" ) ELSE w.String( "FALSE" ) END
  708. ELSE
  709. IF x.value THEN w.String("true" ) ELSE w.String( "false" ) END
  710. END
  711. END VisitBooleanValue;
  712. PROCEDURE Hex(x: HUGEINT);
  713. VAR i: LONGINT; a: ARRAY 20 OF CHAR; y: HUGEINT;
  714. BEGIN
  715. i := 0;
  716. REPEAT
  717. y := x MOD 10H;
  718. IF y < 10 THEN a[i] := CHR(y+ORD('0'))
  719. ELSE a[i] := CHR(y-10+ORD('A'))
  720. END;
  721. x := x DIV 10H;
  722. INC(i);
  723. UNTIL (x=0) OR (i=16);
  724. IF y >=10 THEN w.Char("0") END;
  725. REPEAT DEC( i ); w.Char( a[i] ) UNTIL i = 0
  726. END Hex;
  727. PROCEDURE VisitIntegerValue(x: SyntaxTree.IntegerValue);
  728. PROCEDURE InBounds(val: HUGEINT; bits: LONGINT): BOOLEAN;
  729. VAR m: HUGEINT;
  730. BEGIN
  731. m := ASH(HUGEINT(1),bits-1);
  732. RETURN (val < m) & (-val <= m)
  733. END InBounds;
  734. BEGIN
  735. (*! use subtype for representation form ? *)
  736. IF x.hvalue = MIN(HUGEINT) THEN
  737. (* special case: display 8000000000000000H without leading minus sign
  738. to avoid double minus sign for unary expression -8000000000000000H
  739. *)
  740. w.Char("0"); w.Hex(x.hvalue,-16); w.Char("H");
  741. ELSIF InBounds(x.hvalue,32) THEN
  742. w.Int(SHORT(x.hvalue),1);
  743. ELSE
  744. Hex(x.hvalue); w.Char("H");
  745. END;
  746. END VisitIntegerValue;
  747. PROCEDURE VisitCharacterValue(x: SyntaxTree.CharacterValue);
  748. BEGIN
  749. Hex( ORD(x.value)); w.String( "X" );
  750. END VisitCharacterValue;
  751. PROCEDURE VisitSetValue(x: SyntaxTree.SetValue);
  752. VAR i: LONGINT;
  753. BEGIN
  754. w.String("{");
  755. i := 0;
  756. WHILE (i<MAX(SET)) & ~(i IN x.value) DO
  757. INC(i);
  758. END;
  759. IF i<MAX(SET) THEN
  760. w.Int(i,1);
  761. INC(i);
  762. WHILE i < MAX(SET) DO
  763. IF i IN x.value THEN w.String(","); w.Int(i,1); END;
  764. INC(i)
  765. END
  766. END;
  767. w.String("}");
  768. END VisitSetValue;
  769. PROCEDURE VisitMathArrayValue(x: SyntaxTree.MathArrayValue);
  770. BEGIN
  771. VisitMathArrayExpression(x.array);
  772. END VisitMathArrayValue;
  773. PROCEDURE FormatedFloat(value: LONGREAL; subtype: LONGINT);
  774. VAR string: ARRAY 128 OF CHAR; i: LONGINT;
  775. BEGIN
  776. IF subtype = Scanner.Real THEN
  777. ws.SetPos(0); ws.Float(value,11(*mantissa X.XXXXXXX *)+5(*exponent E+XXX *)); ws.Get(string);
  778. i := 0;
  779. WHILE(i<LEN(string)) & (string[i] # 0X) DO
  780. IF string[i] = "D" THEN string[i] := "E" END;
  781. INC(i);
  782. END;
  783. w.String(string);
  784. ELSIF subtype = Scanner.Longreal THEN
  785. ws.SetPos(0); ws.Float(value,20(*mantissa X.X..(16)..X *)+5(*exponent E+XXX *) ); ws.Get(string);
  786. i := 0;
  787. WHILE(i<LEN(string)) & (string[i] # 0X) DO
  788. IF string[i] = "E" THEN string[i] := "D" END;
  789. INC(i);
  790. END;
  791. w.String(string);
  792. ELSE
  793. w.Float(value,64)
  794. END;
  795. END FormatedFloat;
  796. PROCEDURE VisitRealValue(x: SyntaxTree.RealValue);
  797. BEGIN FormatedFloat(x.value, x.subtype)
  798. END VisitRealValue;
  799. PROCEDURE VisitComplexValue(x: SyntaxTree.ComplexValue);
  800. BEGIN
  801. IF (x.realValue = 0) & (x.imagValue = 1) THEN
  802. w.String("IMAG")
  803. ELSE
  804. w.String("(");
  805. FormatedFloat(x.realValue, x.subtype) ;
  806. w.String(" ");
  807. IF x.imagValue > 0 THEN w.String("+") END;
  808. FormatedFloat(x.imagValue, x.subtype);
  809. w.String("*IMAG)")
  810. END
  811. END VisitComplexValue;
  812. PROCEDURE VisitStringValue(x: SyntaxTree.StringValue);
  813. VAR i: LONGINT; ch: CHAR;
  814. BEGIN
  815. i := 0;
  816. w.Char('\');
  817. w.Char('"');
  818. WHILE (i < LEN( x.value )) & (x.value[i] # 0X) DO
  819. ch := x.value[i];
  820. IF ch = Scanner.CR THEN w.String("\n")
  821. ELSIF ch = Scanner.LF THEN (* ignore *)
  822. ELSIF ch = Scanner.TAB THEN w.String("\t")
  823. ELSIF ch = '\' THEN w.String("\\")
  824. ELSIF ch = '"' THEN w.String(\"""\); (* " *)
  825. ELSE w.Char(ch)
  826. END;
  827. INC( i );
  828. END;
  829. w.Char('"');
  830. w.Char('\');
  831. END VisitStringValue;
  832. PROCEDURE VisitNilValue(x: SyntaxTree.NilValue);
  833. BEGIN IF case = Scanner.Lowercase THEN w.String( "nil" ); ELSE w.String( "NIL" ); END; IF info THEN BeginComment; Type(x.type); EndComment; END;
  834. END VisitNilValue;
  835. PROCEDURE VisitEnumerationValue(x: SyntaxTree.EnumerationValue);
  836. BEGIN w.Int(x.value,1);
  837. END VisitEnumerationValue;
  838. (**** symbols ****)
  839. PROCEDURE Symbol*(x: SyntaxTree.Symbol);
  840. BEGIN
  841. IF x = NIL THEN
  842. AlertString("nil symbol");
  843. ELSE
  844. x.Accept(SELF);
  845. END
  846. END Symbol;
  847. PROCEDURE VisitSymbol(x: SyntaxTree.Symbol);
  848. BEGIN
  849. AlertString("InvalidSymbol");
  850. END VisitSymbol;
  851. PROCEDURE Visible(symbol: SyntaxTree.Symbol): BOOLEAN;
  852. BEGIN
  853. RETURN TRUE (* (SyntaxTree.Public * symbol.access # {}) OR (mode > SymbolFile) *)
  854. (* using only exported symbols does not work since there might be dependencies ... *)
  855. END Visible;
  856. PROCEDURE PrintSymbol(x: SyntaxTree.Symbol);
  857. VAR first: BOOLEAN;
  858. BEGIN
  859. IF x IS SyntaxTree.Operator THEN
  860. w.String('"');Identifier(x.name); w.String('"')
  861. ELSE
  862. Identifier(x.name)
  863. END;
  864. IF SyntaxTree.PublicWrite IN x.access THEN w.String( "*" )
  865. ELSIF SyntaxTree.PublicRead IN x.access THEN w.String( "-" )
  866. ELSIF x.access = {} THEN ASSERT(mode > SourceCode);
  867. IF info THEN BeginComment; w.String("<- hidden"); EndComment END;
  868. END;
  869. IF x.externalName # NIL THEN
  870. Keyword(" EXTERN " ); w.Char('"');
  871. w.String(x.externalName^); w.Char('"');
  872. END;
  873. IF info THEN
  874. BeginComment;
  875. w.String("access= {");
  876. Access(x.access);
  877. w.String("}");
  878. IF x.offsetInBits # MIN(LONGINT) THEN
  879. w.String("@"); w.Hex(x.offsetInBits,1);
  880. END;
  881. IF x.type # NIL THEN
  882. IF x.type.resolved.alignmentInBits >=0 THEN
  883. w.String("@@"); w.Hex(x.type.resolved.alignmentInBits,1);
  884. END;
  885. END;
  886. EndComment;
  887. END;
  888. END PrintSymbol;
  889. PROCEDURE VisitTypeDeclaration(x: SyntaxTree.TypeDeclaration);
  890. BEGIN
  891. IF Visible(x) THEN
  892. IF (x.access # SyntaxTree.Hidden) OR (mode > SourceCode) THEN
  893. Comments(x.comment,x,FALSE);
  894. PrintSymbol(x);
  895. w.String(" = ");
  896. IF x.access # SyntaxTree.Hidden THEN
  897. Type(x.declaredType);
  898. ELSE ShortType(x.declaredType)
  899. END;
  900. Comments(x.comment,x,TRUE);
  901. END;
  902. END;
  903. END VisitTypeDeclaration;
  904. PROCEDURE TypeDeclarationList(x: SyntaxTree.TypeDeclaration);
  905. BEGIN
  906. Indent;
  907. Keyword("TYPE " );
  908. w.IncIndent;
  909. WHILE(x # NIL) DO
  910. Indent;
  911. Symbol(x);
  912. w.String( "; " );
  913. x := x.nextTypeDeclaration;
  914. IF x # NIL THEN w.Ln END;
  915. END;
  916. w.DecIndent;
  917. END TypeDeclarationList;
  918. PROCEDURE VisitConstant(x: SyntaxTree.Constant);
  919. BEGIN
  920. IF Visible(x) THEN
  921. IF (mode > SourceCode) OR (x.access # SyntaxTree.Hidden) THEN
  922. Comments(x.comment,x,FALSE);
  923. PrintSymbol(x);
  924. IF x.value # NIL THEN
  925. w.String( " = " ); Expression(x.value);
  926. END;
  927. IF info THEN BeginComment; ShortType(x.type); EndComment; END;
  928. IF info & (x.value.resolved = NIL) THEN AlertString("(*NOT A CONSTANT*)") END;
  929. Comments(x.comment,x,TRUE);
  930. END;
  931. END;
  932. END VisitConstant;
  933. PROCEDURE ConstantList(x: SyntaxTree.Constant);
  934. BEGIN
  935. Indent; Keyword("CONST " );
  936. w.IncIndent;
  937. WHILE(x # NIL) DO
  938. IF (x.access # SyntaxTree.Hidden) OR (mode > SourceCode) THEN
  939. Indent;
  940. Symbol(x);
  941. w.String( "; " );
  942. END;
  943. x := x.nextConstant;
  944. END;
  945. w.DecIndent;
  946. END ConstantList;
  947. PROCEDURE VisitVariable(x: SyntaxTree.Variable);
  948. VAR first: BOOLEAN;
  949. BEGIN
  950. IF Visible(x) THEN
  951. IF (x.access # SyntaxTree.Hidden) THEN
  952. Comments(x.comment,x,FALSE);
  953. PrintSymbol(x);
  954. IF x.modifiers # NIL THEN w.String(" "); Modifiers(x.modifiers); END;
  955. IF x.initializer # NIL THEN
  956. w.String( " := " ); Expression (x.initializer);
  957. END;
  958. w.String( ": " );
  959. Type(x.type);
  960. Comments(x.comment,x,TRUE);
  961. ELSIF mode>SourceCode THEN
  962. Comments(x.comment,x,FALSE);
  963. PrintSymbol(x);
  964. IF x.initializer # NIL THEN
  965. w.String( " := " ); Expression (x.initializer);
  966. END;
  967. Comments(x.comment,x,TRUE);
  968. END
  969. END;
  970. END VisitVariable;
  971. PROCEDURE VariableList(x: SyntaxTree.Variable);
  972. VAR next: SyntaxTree.Variable;
  973. PROCEDURE Flags(x: SyntaxTree.Variable);
  974. VAR first: BOOLEAN;
  975. BEGIN
  976. first := TRUE;
  977. IF x.fixed THEN
  978. Value(Global.NameFixed,x.alignment,first)
  979. ELSIF x.alignment > 1 THEN
  980. Value(Global.NameAligned,x.alignment,first)
  981. ELSIF x.fictive THEN
  982. Value(Global.NameFictive, x.fictiveOffset, first);
  983. END;
  984. IF x.untraced THEN
  985. Flag(Global.NameUntraced,first)
  986. END;
  987. FlagEnd(first);
  988. END Flags;
  989. BEGIN
  990. w.IncIndent;
  991. WHILE(x # NIL) DO
  992. next := x.nextVariable;
  993. IF (x.access # SyntaxTree.Hidden) OR (mode > SourceCode) THEN
  994. Indent;
  995. Comments(x.comment, x, FALSE);
  996. PrintSymbol(x); Flags(x);
  997. WHILE(next # NIL) & (next.type = x.type) & ((next.access # SyntaxTree.Hidden) OR (mode > SourceCode)) DO
  998. w.String(", "); PrintSymbol(next); Flags(next);
  999. next := next.nextVariable;
  1000. END;
  1001. IF x.access # SyntaxTree.Hidden THEN
  1002. w.String(": ");
  1003. Type(x.type);
  1004. ELSE
  1005. w.String(": ");
  1006. ShortType(x.type);
  1007. END;
  1008. w.String("; ");
  1009. Comments(x.comment,x, TRUE);
  1010. END;
  1011. x := next;
  1012. END;
  1013. w.DecIndent
  1014. END VariableList;
  1015. PROCEDURE VisitParameter(x: SyntaxTree.Parameter);
  1016. BEGIN
  1017. IF (x.access # SyntaxTree.Hidden) THEN
  1018. Comments(x.comment,x,TRUE);
  1019. IF x.kind = SyntaxTree.VarParameter THEN Keyword("VAR " );
  1020. ELSIF x.kind = SyntaxTree.ConstParameter THEN Keyword("CONST " );
  1021. END;
  1022. PrintSymbol(x);
  1023. IF x.modifiers # NIL THEN w.String(" "); Modifiers(x.modifiers); END;
  1024. IF x.defaultValue # NIL THEN
  1025. w.String("= "); Expression(x.defaultValue);
  1026. END;
  1027. w.String( ": " );
  1028. Type(x.type);
  1029. Comments(x.comment,x,TRUE);
  1030. ELSIF (mode > SourceCode) THEN
  1031. Comments(x.comment,x,FALSE);
  1032. PrintSymbol(x);
  1033. Comments(x.comment,x,TRUE);
  1034. END;
  1035. END VisitParameter;
  1036. PROCEDURE ParameterList*(x: SyntaxTree.Parameter);
  1037. VAR next: SyntaxTree.Parameter; first: BOOLEAN;
  1038. PROCEDURE Flags(x: SyntaxTree.Parameter);
  1039. VAR first: BOOLEAN;
  1040. BEGIN
  1041. IF x.modifiers # NIL THEN
  1042. Modifiers(x.modifiers)
  1043. ELSE
  1044. first := TRUE;
  1045. IF x.untraced THEN
  1046. Flag(Global.NameUntraced,first)
  1047. END;
  1048. FlagEnd(first);
  1049. END;
  1050. END Flags;
  1051. BEGIN
  1052. first := TRUE;
  1053. w.String( "(" );
  1054. WHILE(x # NIL) DO
  1055. next := x.nextParameter;
  1056. IF (x.access # SyntaxTree.Hidden) OR (mode > SourceCode) THEN
  1057. IF ~first THEN w.String("; ") END;
  1058. first := FALSE;
  1059. IF x.kind = SyntaxTree.VarParameter THEN Keyword("VAR " );
  1060. ELSIF x.kind = SyntaxTree.ConstParameter THEN Keyword("CONST " );
  1061. END;
  1062. PrintSymbol(x); Flags(x);
  1063. IF x.defaultValue # NIL THEN
  1064. w.String("= "); Expression(x.defaultValue);
  1065. END;
  1066. WHILE (next # NIL) & (next.type = x.type) & (next.kind = x.kind) & ((next.access # SyntaxTree.Hidden) OR (mode > SourceCode)) DO
  1067. w.String(", ");
  1068. PrintSymbol(next); Flags(next);
  1069. IF next.defaultValue # NIL THEN
  1070. w.String("= "); Expression(next.defaultValue);
  1071. END;
  1072. next := next.nextParameter;
  1073. END;
  1074. IF x.access # SyntaxTree.Hidden THEN
  1075. w.String(": ");
  1076. Type(x.type);
  1077. ELSE
  1078. w.String(": ");
  1079. ShortType(x.type);
  1080. END;
  1081. END;
  1082. x := next;
  1083. END;
  1084. w.String( ")" );
  1085. END ParameterList;
  1086. PROCEDURE Access(access: SET);
  1087. BEGIN
  1088. IF SyntaxTree.PublicWrite IN access THEN w.String(" PublicWrite") END;
  1089. IF SyntaxTree.ProtectedWrite IN access THEN w.String(" ProtectedWrite") END;
  1090. IF SyntaxTree.InternalWrite IN access THEN w.String(" InternalWrite") END;
  1091. IF SyntaxTree.PublicRead IN access THEN w.String(" PublicRead") END;
  1092. IF SyntaxTree.ProtectedRead IN access THEN w.String(" ProtectedRead") END;
  1093. IF SyntaxTree.InternalRead IN access THEN w.String(" InternalRead") END;
  1094. END Access;
  1095. PROCEDURE VisitProcedure(x: SyntaxTree.Procedure);
  1096. VAR type: SyntaxTree.ProcedureType; flags: SET; first: BOOLEAN;
  1097. BEGIN
  1098. IF Visible(x) THEN
  1099. Indent;
  1100. Comments(x.comment,x,FALSE);
  1101. Keyword("PROCEDURE " );
  1102. IF (mode = SymbolFile) & ~x.isInline THEN
  1103. w.String("^ ");
  1104. END;
  1105. (*
  1106. CallingConvention(x.type(SyntaxTree.ProcedureType).callingConvention);
  1107. *)
  1108. type := x.type(SyntaxTree.ProcedureType);
  1109. (*
  1110. flags := type.flags;
  1111. *)
  1112. first := TRUE;
  1113. IF type.stackAlignment > 1 THEN Value(Global.NameStackAligned,type.stackAlignment,first) END;
  1114. IF (type.isRealtime) THEN Flag(Global.NameRealtime,first) END;
  1115. IF (type.noReturn) THEN Flag(Global.NameNoReturn,first) END;
  1116. IF (x.fixed) THEN Value(Global.NameFixed, x.alignment,first)
  1117. ELSIF (x.alignment >1) THEN Value(Global.NameAligned, x.alignment, first)
  1118. END;
  1119. FlagEnd(first);
  1120. IF x.isInline THEN w.String(" - ") END;
  1121. IF x.isConstructor THEN w.String(" & ") END;
  1122. IF x.isFinalizer THEN w.String(" ~ ") END;
  1123. IF info THEN
  1124. BeginComment;
  1125. Modifiers(x.type(SyntaxTree.ProcedureType).modifiers);
  1126. EndComment;
  1127. END;
  1128. PrintSymbol(x);
  1129. IF (type.firstParameter # NIL) OR (type.returnType # NIL ) THEN (* print parentheses only if not parameterless procedure *)
  1130. ParameterList(type.firstParameter);
  1131. END;
  1132. IF type.returnType # NIL THEN
  1133. w.String( ": " );
  1134. IF type.hasUntracedReturn THEN
  1135. first := TRUE;
  1136. Flag(Global.NameUntraced, first);
  1137. FlagEnd(first);
  1138. END;
  1139. Type(type.returnType);
  1140. END;
  1141. IF info & (type.returnParameter # NIL) THEN
  1142. BeginComment;
  1143. w.String("retPar = ");
  1144. Symbol(type.returnParameter);
  1145. EndComment;
  1146. END;
  1147. IF x.externalName = NIL THEN
  1148. IF (mode > SymbolFile) OR (mode = SymbolFile) & (x.isInline) THEN
  1149. w.String( ";" );
  1150. Comments(x.comment,x,TRUE);
  1151. IF mode >= SymbolFile THEN
  1152. ProcedureScope(x.procedureScope);
  1153. END;
  1154. Indent; Keyword("END " ); Identifier(x.name);
  1155. END;
  1156. END;
  1157. END;
  1158. END VisitProcedure;
  1159. PROCEDURE VisitOperator(x: SyntaxTree.Operator);
  1160. VAR type: SyntaxTree.ProcedureType;
  1161. recordType: SyntaxTree.RecordType;
  1162. i: LONGINT;
  1163. valid, first: BOOLEAN;
  1164. BEGIN
  1165. IF Visible(x) THEN
  1166. Indent;
  1167. Comments(x.comment,x,FALSE);
  1168. (* mark array access operators for array-structured object types *)
  1169. IF info THEN
  1170. IF (x.scope # NIL) & (x.scope IS SyntaxTree.RecordScope) THEN
  1171. recordType := x.scope(SyntaxTree.RecordScope).ownerRecord;
  1172. IF recordType.HasArrayStructure() THEN
  1173. BeginComment;
  1174. valid := FALSE;
  1175. IF x = recordType.arrayAccessOperators.len THEN w.String("the length operator: "); valid := TRUE;
  1176. ELSIF x = recordType.arrayAccessOperators.generalRead THEN w.String("the general read operator"); valid := TRUE;
  1177. ELSIF x = recordType.arrayAccessOperators.generalWrite THEN w.String("the general write operator"); valid := TRUE;
  1178. ELSE
  1179. FOR i := 0 TO LEN(recordType.arrayAccessOperators.read, 0) - 1 DO
  1180. IF x = recordType.arrayAccessOperators.read[i] THEN w.String("a read operator (hash="); w.Int(i, 1); w.String("):"); valid := TRUE;
  1181. ELSIF x = recordType.arrayAccessOperators.write[i] THEN w.String("a write operator (hash="); w.Int(i, 1); w.String("):"); valid := TRUE;
  1182. END
  1183. END
  1184. END;
  1185. IF ~valid THEN w.String("an invalid operator:") END;
  1186. EndComment;
  1187. w.String(" ");
  1188. END
  1189. END
  1190. END;
  1191. Keyword("OPERATOR ");
  1192. first := TRUE;
  1193. IF x.isInline THEN
  1194. ASSERT(~x.isDynamic);
  1195. w.String("-");
  1196. ELSE
  1197. IF mode = SymbolFile THEN w.String("^ ") END;
  1198. IF x.isDynamic THEN Flag(Global.NameDynamic, first) END;
  1199. IF ~first THEN w.String("}") END;
  1200. END;
  1201. type := x.type(SyntaxTree.ProcedureType);
  1202. PrintSymbol(x);
  1203. ParameterList(type.firstParameter);
  1204. IF type.returnType # NIL THEN
  1205. w.String( ": " );
  1206. IF type.hasUntracedReturn THEN
  1207. first := TRUE;
  1208. Flag(Global.NameUntraced, first);
  1209. FlagEnd(first);
  1210. END;
  1211. Type(type.returnType);
  1212. END;
  1213. IF info & (type.returnParameter # NIL) THEN
  1214. BeginComment;
  1215. (*w.String("retPar = ");*) (*! this is present in VisitProcedure - should it be present here as well??? *)
  1216. Symbol(type.returnParameter);
  1217. EndComment;
  1218. END;
  1219. IF x.externalName = NIL THEN
  1220. IF (mode > SymbolFile) OR (mode = SymbolFile) & (x.isInline) THEN
  1221. w.String( ";" );
  1222. Comments(x.comment,x,TRUE);
  1223. IF mode >= SymbolFile THEN
  1224. ProcedureScope(x.procedureScope);
  1225. END;
  1226. Indent; Keyword("END " ); w.String( '"' ); Identifier(x.name); w.String( '"' );
  1227. END;
  1228. END;
  1229. END
  1230. END VisitOperator;
  1231. PROCEDURE ProcedureList(x: SyntaxTree.Procedure);
  1232. BEGIN
  1233. w.IncIndent;
  1234. WHILE(x # NIL) DO
  1235. IF (x.access # SyntaxTree.Hidden) & ~(x.isBodyProcedure) OR (mode > SourceCode) THEN
  1236. Symbol(x);
  1237. w.String( "; " );
  1238. END;
  1239. x := x.nextProcedure;
  1240. IF (x# NIL) & (mode > SymbolFile) & ((x.access # SyntaxTree.Hidden) OR (mode > SourceCode)) THEN w.Ln END;
  1241. END;
  1242. w.DecIndent;
  1243. END ProcedureList;
  1244. PROCEDURE VisitImport(x: SyntaxTree.Import);
  1245. VAR context: SyntaxTree.Identifier;
  1246. BEGIN
  1247. IF x.moduleName # x.name THEN Identifier(x.name); w.String( " := " ); END;
  1248. IF (x.scope = NIL) OR (x.scope.ownerModule = NIL) THEN context := SyntaxTree.invalidIdentifier ELSE context := x.scope.ownerModule.context END;
  1249. Identifier(x.moduleName);
  1250. IF (x.context # SyntaxTree.invalidIdentifier) & (x.context#context) THEN
  1251. w.String(" IN ");
  1252. Identifier(x.context)
  1253. END;
  1254. END VisitImport;
  1255. PROCEDURE ImportList(x: SyntaxTree.Import);
  1256. VAR first: BOOLEAN;
  1257. BEGIN
  1258. first := TRUE;
  1259. WHILE(x # NIL) DO
  1260. IF x.direct & (x.module # NIL) OR (mode > SymbolFile) THEN
  1261. IF ~first THEN w.String(", ") ELSE Indent; Keyword("IMPORT "); first := FALSE END;
  1262. Symbol(x);
  1263. END;
  1264. x := x.nextImport;
  1265. END;
  1266. IF ~first THEN w.String( ";" ); END;
  1267. END ImportList;
  1268. PROCEDURE VisitBuiltin(x: SyntaxTree.Builtin);
  1269. BEGIN
  1270. Indent; Keyword("BUILTIN ");
  1271. Identifier(x.name);
  1272. END VisitBuiltin;
  1273. PROCEDURE BuiltinList(x: SyntaxTree.Builtin);
  1274. BEGIN
  1275. WHILE(x # NIL) DO
  1276. VisitBuiltin(x);
  1277. x := x.nextBuiltin;
  1278. END;
  1279. END BuiltinList;
  1280. PROCEDURE BeginComment;
  1281. BEGIN
  1282. w.BeginComment; w.String("(*");
  1283. END BeginComment;
  1284. PROCEDURE EndComment;
  1285. BEGIN
  1286. w.String("*)");w.EndComment
  1287. END EndComment;
  1288. PROCEDURE Comment(x: SyntaxTree.Comment);
  1289. VAR i: LONGINT; ch: CHAR;
  1290. BEGIN
  1291. BeginComment;
  1292. WHILE (i<LEN(x.source^)) & (x.source[i] # 0X) DO
  1293. ch := x.source[i];
  1294. IF ch = 0DX THEN w.Ln
  1295. ELSE w.Char(ch)
  1296. END;
  1297. INC(i);
  1298. END;
  1299. EndComment;
  1300. END Comment;
  1301. PROCEDURE Comments(c: SyntaxTree.Comment; x: ANY; sameLine: BOOLEAN);
  1302. BEGIN
  1303. IF mode >= SourceCode THEN
  1304. WHILE (c # NIL) & (c.item = x) DO
  1305. IF c.sameLine = sameLine THEN
  1306. Comment(c);
  1307. IF ~sameLine THEN
  1308. Indent;
  1309. END;
  1310. END;
  1311. c := c.nextComment;
  1312. END;
  1313. END;
  1314. END Comments;
  1315. PROCEDURE CommentList(x: SyntaxTree.Comment);
  1316. BEGIN
  1317. IF info THEN
  1318. WHILE (x#NIL) DO
  1319. Indent;
  1320. w.String("comment at position "); w.Int(x.position.start,1);
  1321. IF x.sameLine THEN w.String("(in line with item)") END;
  1322. IF x.item = NIL THEN w.String("(no item)"); END;
  1323. w.String(":");
  1324. Comment(x);
  1325. x := x.nextComment;
  1326. END;
  1327. END;
  1328. END CommentList;
  1329. (*** scopes ****)
  1330. PROCEDURE Scope*(x: SyntaxTree.Scope);
  1331. VAR prevScope: SyntaxTree.Scope;
  1332. BEGIN
  1333. prevScope := currentScope;
  1334. currentScope := x;
  1335. (* ASSERT(currentScope.outerScope = prevScope); (* sanity check *) *)
  1336. WITH x: SyntaxTree.CellScope DO
  1337. IF x.firstImport # NIL THEN ImportList(x.firstImport) END;
  1338. ELSE
  1339. END;
  1340. IF x.firstConstant # NIL THEN ConstantList(x.firstConstant); END;
  1341. IF x.firstTypeDeclaration # NIL THEN TypeDeclarationList(x.firstTypeDeclaration); END;
  1342. IF x.firstVariable # NIL THEN Indent; Keyword("VAR " ); VariableList(x.firstVariable); END;
  1343. IF x.firstProcedure # NIL THEN w.Ln; ProcedureList(x.firstProcedure) END;
  1344. currentScope := prevScope;
  1345. END Scope;
  1346. PROCEDURE ProcedureScope(x: SyntaxTree.ProcedureScope);
  1347. VAR prevScope: SyntaxTree.Scope;
  1348. BEGIN
  1349. prevScope := currentScope;
  1350. currentScope := x;
  1351. IF (mode >= SourceCode) OR (x.ownerProcedure.isInline) THEN Scope(x) END;
  1352. IF (mode >= SymbolFile) & (x.body # NIL) THEN Body(x.body)
  1353. END;
  1354. currentScope := prevScope;
  1355. END ProcedureScope;
  1356. PROCEDURE Statement*(x: SyntaxTree.Statement);
  1357. BEGIN
  1358. IF x = NIL THEN
  1359. AlertString("nil statement")
  1360. ELSE
  1361. Comments(x.comment, x, FALSE);
  1362. x.Accept(SELF);
  1363. Comments(x.comment,x,TRUE);
  1364. END
  1365. END Statement;
  1366. PROCEDURE StatementSequence*(x: SyntaxTree.StatementSequence);
  1367. VAR statement: SyntaxTree.Statement; i: LONGINT;
  1368. BEGIN
  1369. IF singleStatement THEN
  1370. w.String("...")
  1371. ELSE
  1372. FOR i := 0 TO x.Length() - 1 DO
  1373. statement := x.GetStatement( i );
  1374. Indent; Statement(statement);
  1375. IF i < x.Length() - 1 THEN w.String( "; " ); END;
  1376. END;
  1377. END;
  1378. END StatementSequence;
  1379. PROCEDURE VisitStatement(x: SyntaxTree.Statement);
  1380. BEGIN
  1381. AlertString("InvalidStatement");
  1382. END VisitStatement;
  1383. PROCEDURE VisitProcedureCallStatement(x: SyntaxTree.ProcedureCallStatement);
  1384. BEGIN Expression(x.call) END VisitProcedureCallStatement;
  1385. PROCEDURE VisitAssignment(x: SyntaxTree.Assignment);
  1386. BEGIN
  1387. Expression(x.left); w.String( " := " ); Expression(x.right);
  1388. END VisitAssignment;
  1389. PROCEDURE VisitCommunicationStatement(x: SyntaxTree.CommunicationStatement);
  1390. VAR identifier: SyntaxTree.Identifier;
  1391. BEGIN
  1392. Expression(x.left);
  1393. identifier := Global.GetIdentifier(x.op,case);
  1394. w.String(" "); Identifier(identifier); w.String(" ");
  1395. Expression(x.right);
  1396. END VisitCommunicationStatement;
  1397. PROCEDURE IfPart(x: SyntaxTree.IfPart);
  1398. BEGIN
  1399. Comments(x.comment, x, FALSE);
  1400. Keyword("IF " );
  1401. Expression(x.condition);
  1402. Keyword(" THEN " );
  1403. Comments(x.comment,x,TRUE);
  1404. w.IncIndent;
  1405. StatementSequence(x.statements);
  1406. w.DecIndent;
  1407. END IfPart;
  1408. PROCEDURE VisitIfStatement(x: SyntaxTree.IfStatement);
  1409. VAR i: LONGINT; elsif: SyntaxTree.IfPart;
  1410. BEGIN
  1411. IfPart(x.ifPart);
  1412. FOR i := 0 TO x.ElsifParts() - 1 DO
  1413. elsif := x.GetElsifPart( i );
  1414. Indent; Keyword("ELS");
  1415. IfPart(elsif);
  1416. END;
  1417. IF x.elsePart # NIL THEN
  1418. Indent; Keyword("ELSE" );
  1419. w.IncIndent;
  1420. StatementSequence(x.elsePart);
  1421. w.DecIndent;
  1422. END;
  1423. Indent; Keyword("END" );
  1424. END VisitIfStatement;
  1425. PROCEDURE WithPart(x: SyntaxTree.WithPart);
  1426. BEGIN
  1427. Comments(x.comment, x, FALSE);
  1428. Expression(x.variable);
  1429. w.String(" : ");
  1430. Type(x.type);
  1431. Keyword(" DO " );
  1432. Comments(x.comment,x, TRUE);
  1433. w.IncIndent; StatementSequence(x.statements); w.DecIndent;
  1434. END WithPart;
  1435. PROCEDURE VisitWithStatement(x: SyntaxTree.WithStatement);
  1436. VAR i: LONGINT;
  1437. BEGIN
  1438. Indent; Keyword("WITH " );
  1439. WithPart(x.GetWithPart(0));
  1440. FOR i := 1 TO x.WithParts()-1 DO
  1441. Indent; w.String("| ");
  1442. WithPart(x.GetWithPart(i));
  1443. END;
  1444. IF x.elsePart # NIL THEN
  1445. Indent; w.String("ELSE ");
  1446. w.IncIndent; StatementSequence(x.elsePart); w.DecIndent;
  1447. END;
  1448. Indent; Keyword("END" );
  1449. END VisitWithStatement;
  1450. PROCEDURE CasePart(x: SyntaxTree.CasePart);
  1451. VAR case: SyntaxTree.CaseConstant;
  1452. BEGIN
  1453. Comments(x.comment, x, FALSE);
  1454. ExpressionList(x.elements);
  1455. IF info THEN
  1456. w.BeginComment;
  1457. case := x.firstConstant;
  1458. WHILE(case # NIL) DO
  1459. IF case # x.firstConstant THEN w.String(",") END;
  1460. w.Int(case.min,1); w.String(".."); w.Int(case.max,1);
  1461. case := case.next;
  1462. END;
  1463. EndComment;
  1464. END;
  1465. w.String( ":" );
  1466. Comments(x.comment,x,TRUE);
  1467. w.IncIndent; StatementSequence(x.statements); w.DecIndent;
  1468. END CasePart;
  1469. PROCEDURE VisitCaseStatement(x: SyntaxTree.CaseStatement);
  1470. VAR i: LONGINT; case: SyntaxTree.CasePart;
  1471. BEGIN
  1472. Keyword("CASE " );
  1473. Expression(x.variable);
  1474. Keyword(" OF " );
  1475. FOR i := 0 TO x.CaseParts() - 1 DO
  1476. case := x.GetCasePart( i );
  1477. Indent;
  1478. w.String( "| " );
  1479. CasePart(case);
  1480. END;
  1481. IF x.elsePart # NIL THEN
  1482. Indent;
  1483. Keyword("ELSE" );
  1484. w.IncIndent;
  1485. StatementSequence(x.elsePart);
  1486. w.DecIndent;
  1487. END;
  1488. Indent;
  1489. Keyword("END" );
  1490. END VisitCaseStatement;
  1491. PROCEDURE VisitWhileStatement(x: SyntaxTree.WhileStatement);
  1492. BEGIN
  1493. Keyword("WHILE " );
  1494. Expression(x.condition);
  1495. Keyword(" DO " );
  1496. w.IncIndent;
  1497. StatementSequence(x.statements);
  1498. w.DecIndent;
  1499. Indent;
  1500. Keyword("END" );
  1501. END VisitWhileStatement;
  1502. PROCEDURE VisitRepeatStatement(x: SyntaxTree.RepeatStatement);
  1503. BEGIN
  1504. Keyword("REPEAT " );
  1505. w.IncIndent;
  1506. StatementSequence(x.statements);
  1507. w.DecIndent;
  1508. Indent; Keyword("UNTIL " );
  1509. Expression(x.condition);
  1510. END VisitRepeatStatement;
  1511. PROCEDURE VisitForStatement(x: SyntaxTree.ForStatement);
  1512. BEGIN
  1513. Keyword("FOR " );
  1514. Expression(x.variable);
  1515. w.String( " := " );
  1516. Expression(x.from);
  1517. Keyword(" TO " );
  1518. Expression(x.to);
  1519. IF x.by # NIL THEN
  1520. Keyword(" BY " );
  1521. Expression(x.by);
  1522. END;
  1523. Keyword(" DO " );
  1524. w.IncIndent;
  1525. StatementSequence(x.statements);
  1526. w.DecIndent;
  1527. Indent;
  1528. Keyword("END" );
  1529. END VisitForStatement;
  1530. PROCEDURE VisitLoopStatement(x: SyntaxTree.LoopStatement);
  1531. BEGIN
  1532. Keyword("LOOP " );
  1533. w.IncIndent; StatementSequence(x.statements); w.DecIndent;
  1534. Indent; Keyword("END" );
  1535. END VisitLoopStatement;
  1536. PROCEDURE VisitExitableBlock(x: SyntaxTree.ExitableBlock);
  1537. BEGIN
  1538. Keyword("EXITABLE " );
  1539. w.IncIndent; StatementSequence(x.statements); w.DecIndent;
  1540. Indent; Keyword("END " );
  1541. END VisitExitableBlock;
  1542. PROCEDURE VisitExitStatement(x: SyntaxTree.ExitStatement);
  1543. BEGIN Keyword("EXIT" ) END VisitExitStatement;
  1544. PROCEDURE VisitReturnStatement(x: SyntaxTree.ReturnStatement);
  1545. BEGIN
  1546. Keyword("RETURN " );
  1547. IF x.returnValue # NIL THEN Expression(x.returnValue) END
  1548. END VisitReturnStatement;
  1549. PROCEDURE VisitAwaitStatement(x: SyntaxTree.AwaitStatement);
  1550. BEGIN
  1551. Keyword("AWAIT (" ); Expression(x.condition); w.String( ")" );
  1552. END VisitAwaitStatement;
  1553. PROCEDURE Modifiers(x: SyntaxTree.Modifier);
  1554. VAR name: Scanner.IdentifierString; first: BOOLEAN;
  1555. BEGIN
  1556. first := TRUE;
  1557. WHILE x # NIL DO
  1558. IF first THEN w.String("{"); first := FALSE ELSE w.String(", ") END;
  1559. Basic.GetString(x.identifier,name);
  1560. w.String(name);
  1561. IF x.expression # NIL THEN
  1562. w.String("(");
  1563. Expression(x.expression);
  1564. w.String(")");
  1565. END;
  1566. x := x.nextModifier;
  1567. END;
  1568. IF ~first THEN w.String("} ") END;
  1569. END Modifiers;
  1570. (*
  1571. PROCEDURE BlockModifier(x: SyntaxTree.StatementBlock);
  1572. VAR first: BOOLEAN;
  1573. PROCEDURE Comma;
  1574. BEGIN
  1575. IF first THEN first := FALSE ELSE w.String(", "); END;
  1576. END Comma;
  1577. BEGIN
  1578. first := TRUE;
  1579. IF x.flags # {} THEN
  1580. w.String("{");
  1581. IF SyntaxTree.ActiveFlag IN x.flags THEN Comma; w.String("ACTIVE") END;
  1582. IF SyntaxTree.PriorityFlag IN x.flags THEN Comma; w.String("PRIORITY("); Expression(x(SyntaxTree.Body).priority); w.String(")"); first := FALSE; END;
  1583. IF SyntaxTree.SafeFlag IN x.flags THEN Comma; w.String("SAFE") END;
  1584. IF SyntaxTree.ExclusiveFlag IN x.flags THEN Comma; w.String("EXCLUSIVE") END;
  1585. w.String("}");
  1586. END;
  1587. END BlockModifier;
  1588. *)
  1589. PROCEDURE VisitStatementBlock(x: SyntaxTree.StatementBlock);
  1590. BEGIN
  1591. Keyword("BEGIN"); Modifiers(x.blockModifiers);
  1592. w.IncIndent;
  1593. IF x.statements # NIL THEN StatementSequence(x.statements); END;
  1594. w.DecIndent;
  1595. Indent; Keyword("END");
  1596. END VisitStatementBlock;
  1597. PROCEDURE Code(x: SyntaxTree.Code);
  1598. VAR i: LONGINT; ch: CHAR; cr: BOOLEAN; procedure: SyntaxTree.Procedure; procedureType: SyntaxTree.ProcedureType;
  1599. CONST CR=0DX;
  1600. BEGIN
  1601. IF (currentScope # NIL) & (currentScope IS SyntaxTree.ProcedureScope) THEN
  1602. procedure := currentScope(SyntaxTree.ProcedureScope).ownerProcedure;
  1603. procedureType := procedure.type(SyntaxTree.ProcedureType);
  1604. END;
  1605. IF (mode >= SourceCode) OR (procedure = NIL) OR (procedure.access * SyntaxTree.Public # {}) & procedure.isInline THEN
  1606. (*
  1607. IF x.inlineCode # NIL THEN
  1608. unit := 8;
  1609. w.String(" D"); w.Int(unit,1);
  1610. i := 0; size := x.inlineCode.GetSize();
  1611. WHILE i < size DO
  1612. value := x.inlineCode.GetBits(i,unit);
  1613. w.String(" "); w.Int(value,1);
  1614. INC(i,unit);
  1615. END;
  1616. ELS*)
  1617. IF (x.sourceCode # NIL) THEN
  1618. i := 0;
  1619. ch := x.sourceCode[0];
  1620. WHILE (ch # 0X) DO
  1621. IF ch = CR THEN
  1622. cr := TRUE;
  1623. ELSE
  1624. IF cr THEN Indent; cr := FALSE END;
  1625. w.Char(ch);
  1626. END;
  1627. INC(i); ch := x.sourceCode[i];
  1628. END;
  1629. END;
  1630. (*
  1631. IF x.inlineCode # NIL THEN
  1632. w.String("; ");
  1633. size := x.inlineCode.GetSize() DIV 8;
  1634. FOR i := 0 TO size-1 DO
  1635. value := x.inlineCode.GetBits(i*8,8);
  1636. w.Hex(value,-2); w.String(" ");
  1637. END;
  1638. END;
  1639. *)
  1640. END;
  1641. END Code;
  1642. PROCEDURE VisitCode(x: SyntaxTree.Code);
  1643. VAR in, out: BOOLEAN;
  1644. BEGIN
  1645. Indent; Keyword("CODE");
  1646. Code(x);
  1647. in := x.inRules.Length()>0;
  1648. out := x.outRules.Length() >0;
  1649. IF in OR out THEN
  1650. Indent; Keyword("WITH ");
  1651. IF in THEN
  1652. Indent; Keyword("IN "); StatementSequence(x.inRules)
  1653. END;
  1654. IF out THEN
  1655. Indent; Keyword("OUT "); StatementSequence(x.outRules)
  1656. END;
  1657. END;
  1658. Indent; Keyword("END");
  1659. END VisitCode;
  1660. PROCEDURE Body(x: SyntaxTree.Body);
  1661. VAR
  1662. BEGIN
  1663. IF x.code # NIL THEN
  1664. Indent; Keyword("CODE");
  1665. Code(x.code);
  1666. ELSE
  1667. Indent; Keyword("BEGIN" ); Modifiers(x.blockModifiers);
  1668. IF mode >= SourceCode THEN
  1669. IF x.statements # NIL THEN
  1670. w.IncIndent;
  1671. StatementSequence(x.statements);
  1672. w.DecIndent;
  1673. END;
  1674. IF x.finally # NIL THEN
  1675. Indent; Keyword("FINALLY" );
  1676. w.IncIndent;
  1677. StatementSequence(x.finally);
  1678. w.DecIndent
  1679. END;
  1680. END;
  1681. END;
  1682. (* "END" written by caller *)
  1683. END Body;
  1684. PROCEDURE Module*(x: SyntaxTree.Module);
  1685. BEGIN
  1686. IF x = NIL THEN
  1687. AlertString("(* no module *)");
  1688. ELSE
  1689. case := x.case;
  1690. currentScope := x.moduleScope.outerScope;
  1691. Comments(x.comment,x,FALSE);
  1692. Keyword("MODULE ");
  1693. Identifier(x.name);
  1694. IF (x.context # SyntaxTree.invalidIdentifier) & (x.context#Global.A2Name) THEN
  1695. w.String(" IN ");
  1696. Identifier(x.context)
  1697. END;
  1698. IF (x.type IS SyntaxTree.CellType) & (x.type(SyntaxTree.CellType).firstParameter # NIL) THEN (* for actors *)
  1699. ParameterList(x.type(SyntaxTree.CellType).firstParameter);
  1700. END;
  1701. w.String(";");
  1702. Comments(x.comment,x,TRUE);
  1703. w.IncIndent;
  1704. IF x.moduleScope.firstImport # NIL THEN
  1705. ImportList(x.moduleScope.firstImport)
  1706. END;
  1707. w.DecIndent;
  1708. Scope(x.moduleScope);
  1709. IF x.moduleScope.firstBuiltin # NIL THEN
  1710. BuiltinList(x.moduleScope.firstBuiltin)
  1711. END;
  1712. IF (x.moduleScope.bodyProcedure # NIL) & (x.moduleScope.bodyProcedure.procedureScope.body # NIL) THEN
  1713. Body(x.moduleScope.bodyProcedure.procedureScope.body)
  1714. END;
  1715. Indent; Keyword("END "); Identifier(x.name); w.String( "." ); w.Ln; w.Update;
  1716. Comments(x.closingComment,x, FALSE);
  1717. IF (mode > SourceCode) & (x.moduleScope.firstComment # NIL) THEN w.Ln; CommentList(x.moduleScope.firstComment) END;
  1718. END
  1719. END Module;
  1720. PROCEDURE SingleStatement*(b: BOOLEAN);
  1721. BEGIN singleStatement := b
  1722. END SingleStatement;
  1723. PROCEDURE &Init*(w: Streams.Writer; mode: LONGINT; info: BOOLEAN);
  1724. BEGIN
  1725. SELF.w := Basic.GetWriter(w);
  1726. SELF.mode := mode; NEW(ws,128); SELF.info := info; case := Scanner.Uppercase;
  1727. commentCount := 0; alertCount := 0; singleStatement := FALSE;
  1728. END Init;
  1729. END Printer;
  1730. (* debugging helper *)
  1731. VAR
  1732. debug: Printer;
  1733. PROCEDURE NewPrinter*(w: Streams.Writer; mode: LONGINT; info: BOOLEAN): Printer;
  1734. VAR p: Printer;
  1735. BEGIN
  1736. NEW(p,w,mode,info); RETURN p
  1737. END NewPrinter;
  1738. PROCEDURE Info*(CONST info: ARRAY OF CHAR; a: ANY);
  1739. VAR symbol: SyntaxTree.Symbol; scope: SyntaxTree.Scope;
  1740. BEGIN
  1741. debug.w := Basic.GetWriter(D.Log);
  1742. D.Ln;
  1743. D.Str(" --------> ");
  1744. D.Str(info);
  1745. D.Str(" ");
  1746. D.Hex(SYSTEM.VAL(LONGINT,a),8);
  1747. D.Str(" : ");
  1748. IF a = NIL THEN
  1749. D.Str("NIL");
  1750. ELSIF a IS SyntaxTree.Expression THEN
  1751. debug.Expression(a(SyntaxTree.Expression));
  1752. Info("with type",a(SyntaxTree.Expression).type);
  1753. ELSIF a IS SyntaxTree.Type THEN
  1754. IF a IS SyntaxTree.QualifiedType THEN
  1755. D.Str("[QualifiedType] ");
  1756. END;
  1757. debug.Type(a(SyntaxTree.Type))
  1758. ELSIF a IS SyntaxTree.Symbol THEN
  1759. debug.Symbol(a(SyntaxTree.Symbol))
  1760. ELSIF a IS SyntaxTree.Statement THEN
  1761. debug.Statement(a(SyntaxTree.Statement))
  1762. ELSIF a IS SyntaxTree.StatementSequence THEN
  1763. debug.StatementSequence(a(SyntaxTree.StatementSequence));
  1764. ELSIF a IS SyntaxTree.Scope THEN
  1765. scope := a(SyntaxTree.Scope);
  1766. WHILE(scope # NIL) DO
  1767. D.Ln; D.Str(" ");
  1768. IF scope IS SyntaxTree.ModuleScope THEN D.Str("ModuleScope: ")
  1769. ELSIF scope IS SyntaxTree.ProcedureScope THEN D.Str("ProcedureScope: ");
  1770. ELSIF scope IS SyntaxTree.RecordScope THEN D.Str("RecordScope: ");
  1771. ELSE D.Str("Scope: ");
  1772. END;
  1773. symbol := scope.firstSymbol;
  1774. WHILE(symbol # NIL) DO
  1775. debug.Identifier(symbol.name); D.Str(" ");
  1776. symbol := symbol.nextSymbol;
  1777. END;
  1778. scope := scope.outerScope;
  1779. END;
  1780. (*
  1781. ELSIF a IS SyntaxTree.Identifier THEN
  1782. debug.Identifier(a(SyntaxTree.Identifier));
  1783. *)
  1784. ELSIF a IS SyntaxTree.QualifiedIdentifier THEN
  1785. debug.QualifiedIdentifier(a(SyntaxTree.QualifiedIdentifier));
  1786. ELSIF a IS SyntaxTree.Module THEN
  1787. debug.Module(a(SyntaxTree.Module))
  1788. ELSE
  1789. debug.w.String("unknown");
  1790. END;
  1791. D.Update();
  1792. END Info;
  1793. PROCEDURE IsIntegerValue(x: SyntaxTree.Expression; VAR value: LONGINT): BOOLEAN;
  1794. VAR result: BOOLEAN;
  1795. BEGIN
  1796. IF (x.resolved # NIL) & (x.resolved IS SyntaxTree.IntegerValue) THEN
  1797. value := x.resolved(SyntaxTree.IntegerValue).value;
  1798. result := TRUE
  1799. ELSE
  1800. result := FALSE
  1801. END;
  1802. RETURN result
  1803. END IsIntegerValue;
  1804. PROCEDURE Init;
  1805. BEGIN
  1806. NEW(debug,D.Log,All,TRUE);
  1807. debug.case := Scanner.Uppercase;
  1808. END Init;
  1809. BEGIN
  1810. Init;
  1811. END FoxPrintout.