FoxPrintout.Mod 56 KB

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