FoxBasic.Mod 54 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321
  1. MODULE FoxBasic; (** AUTHOR "fof"; PURPOSE "Oberon Compiler: basic helpers: strings, lists, hash tables, graphs, indented writer"; **)
  2. (* (c) fof ETH Zürich, 2009 *)
  3. IMPORT KernelLog, StringPool, Strings, Streams, Diagnostics, Files, SYSTEM, ObjectFile, Modules, D:= Debugging;
  4. CONST
  5. (* error numbers *)
  6. (* first 255 tokens reserved for expected symbol error message *)
  7. UndeclaredIdentifier* = 256;
  8. MultiplyDefinedIdentifier* = 257;
  9. NumberIllegalCharacter* = 258;
  10. StringIllegalCharacter* = 259;
  11. NoMatchProcedureName* = 260;
  12. CommentNotClosed* = 261;
  13. IllegalCharacterValue* = 262;
  14. ValueStartIncorrectSymbol* = 263;
  15. IllegalyMarkedIdentifier* = 264;
  16. IdentifierNoType* = 265;
  17. IdentifierNoRecordType* = 266;
  18. IdentifierNoObjectType* = 267;
  19. ImportNotAvailable* = 268;
  20. RecursiveTypeDeclaration* = 269;
  21. NumberTooLarge* = 270;
  22. IdentifierTooLong* = 271;
  23. StringTooLong* = 272;
  24. InitListSize = 4;
  25. InitErrMsgSize = 300; (* initial size of array of error messages *)
  26. (* value of constant NIL *)
  27. nilval* = 0;
  28. ExportedUnicodeSupport* = FALSE;
  29. (* target machine minimum values of basic types expressed in host machine format: *)
  30. MinSInt* = -80H;
  31. MinInt* = -8000H;
  32. MinLInt* = -80000000H; (* i386: -2147483648*)
  33. (* target machine maximum values of basic types expressed in host machine format: *)
  34. MaxSInt* = 7FH;
  35. MaxInt* = 7FFFH;
  36. MaxLInt* = 7FFFFFFFH; (* i386: 2147483647*)
  37. MaxSet* = 31; (* must be >= 15, else the bootstraped compiler cannot run (IN-tests) *)
  38. invalidString*=-1;
  39. InvalidCode*=Diagnostics.Invalid;
  40. TYPE
  41. (*
  42. String* = POINTER TO ARRAY OF CHAR;
  43. *)
  44. String* = StringPool.Index;
  45. SegmentedName*= ObjectFile.SegmentedName;
  46. FileName*= Files.FileName;
  47. SectionName*= ARRAY 256 OF CHAR;
  48. MessageString*= ARRAY 256 OF CHAR;
  49. ObjectArray = POINTER TO ARRAY OF ANY;
  50. IntegerArray = POINTER TO ARRAY OF LONGINT;
  51. ErrorMsgs = POINTER TO ARRAY OF StringPool.Index;
  52. ComparisonFunction = PROCEDURE {DELEGATE} (object1, object2: ANY): BOOLEAN;
  53. Position*= RECORD
  54. start*, end*, line*, linepos*: LONGINT;
  55. END;
  56. ErrorCode*=LONGINT;
  57. List* = OBJECT (* by Luc Bläser *)
  58. VAR
  59. list: ObjectArray;
  60. count-: LONGINT;
  61. multipleAllowed*: BOOLEAN;
  62. nilAllowed*: BOOLEAN;
  63. PROCEDURE & InitList*(initialSize: LONGINT) ;
  64. BEGIN
  65. IF initialSize <= 0 THEN initialSize := 8 END;
  66. INC( lists ); NEW( list, initialSize ); count := 0; multipleAllowed := FALSE; nilAllowed := FALSE
  67. END InitList;
  68. PROCEDURE Length*( ): LONGINT;
  69. BEGIN
  70. RETURN count
  71. END Length;
  72. PROCEDURE Grow;
  73. VAR old: ObjectArray; i: LONGINT;
  74. BEGIN
  75. INC( enlarged ); old := list; NEW( list, (LEN( list ) * 3+1) DIV 2 (* more optimal for first-fit memory allocators *) ) ;
  76. FOR i := 0 TO count - 1 DO list[i] := old[i] END
  77. END Grow;
  78. PROCEDURE Get*( i: LONGINT ): ANY;
  79. BEGIN
  80. IF (i < 0) OR (i >= count) THEN HALT( 101 ) END;
  81. RETURN list[i]
  82. END Get;
  83. PROCEDURE Set*(i: LONGINT; x: ANY);
  84. BEGIN
  85. IF (i < 0) OR (i >= count) THEN HALT( 101 ) END;
  86. list[i] := x;
  87. END Set;
  88. PROCEDURE Add*( x: ANY );
  89. BEGIN
  90. IF ~nilAllowed THEN ASSERT( x # NIL ) END;
  91. IF ~multipleAllowed THEN ASSERT( ~debug OR ~Contains( x ) ) END; (* already contained *)
  92. IF count = LEN( list ) THEN Grow END;
  93. list[count] := x; INC( count )
  94. END Add;
  95. PROCEDURE Prepend*(x: ANY);
  96. VAR i: LONGINT;
  97. BEGIN
  98. IF ~nilAllowed THEN ASSERT( x # NIL ) END;
  99. IF ~multipleAllowed THEN ASSERT( debug OR ~Contains( x ) ) END; (* already contained *)
  100. IF count = LEN( list ) THEN Grow END;
  101. FOR i := count-1 TO 0 BY - 1 DO
  102. list[i+1] := list[i];
  103. END;
  104. list[0] := x; INC(count);
  105. END Prepend;
  106. PROCEDURE Append*(x: List);
  107. VAR i: LONGINT;
  108. BEGIN
  109. FOR i := 0 TO x.Length() - 1 DO
  110. IF multipleAllowed OR (~debug OR ~Contains(x.Get(i))) THEN
  111. Add(x.Get(i));
  112. END;
  113. END;
  114. END Append;
  115. PROCEDURE Remove*( x: ANY );
  116. VAR i: LONGINT;
  117. BEGIN
  118. i := 0;
  119. WHILE (i < count) & (list[i] # x) DO INC( i ) END;
  120. IF i < count THEN
  121. WHILE (i < count - 1) DO list[i] := list[i + 1]; INC( i ) END;
  122. DEC( count ); list[count] := NIL
  123. END
  124. END Remove;
  125. PROCEDURE RemoveByIndex*( i: LONGINT );
  126. BEGIN
  127. IF i < count THEN
  128. WHILE (i < count - 1) DO list[i] := list[i + 1]; INC( i ) END;
  129. DEC( count ); list[count] := NIL
  130. END
  131. END RemoveByIndex;
  132. PROCEDURE Insert*( i: LONGINT; x: ANY );
  133. VAR j: LONGINT;
  134. BEGIN
  135. IF ~nilAllowed THEN ASSERT( x # NIL ) END;
  136. IF ~multipleAllowed THEN ASSERT( ~debug OR ~Contains( x ) ) END; (* already contained *)
  137. IF count = LEN( list ) THEN Grow END; INC( count );
  138. j := count - 2;
  139. WHILE (j >= i) DO list[j+1] := list[j]; DEC( j ) END;
  140. list[i] := x;
  141. END Insert;
  142. PROCEDURE Replace*( x, y: ANY );
  143. VAR i: LONGINT;
  144. BEGIN
  145. IF ~nilAllowed THEN ASSERT( x # NIL ); ASSERT( y # NIL ) END;
  146. i := IndexOf( x );
  147. IF i >= 0 THEN list[i] := y END
  148. END Replace;
  149. PROCEDURE ReplaceByIndex*( i: LONGINT; x: ANY );
  150. BEGIN
  151. IF ~nilAllowed THEN ASSERT( x # NIL ) END;
  152. IF (i >= 0) & (i < count) THEN list[i] := x
  153. ELSE HALT( 101 ) (* out of boundaries *)
  154. END
  155. END ReplaceByIndex;
  156. (** If the object is not present, -1 is returned *)
  157. PROCEDURE IndexOf*( x: ANY ): LONGINT;
  158. VAR i: LONGINT;
  159. BEGIN
  160. i := 0;
  161. WHILE i < count DO
  162. IF list[i] = x THEN RETURN i END;
  163. INC( i )
  164. END;
  165. RETURN -1
  166. END IndexOf;
  167. PROCEDURE Contains*( x: ANY ): BOOLEAN;
  168. BEGIN
  169. RETURN IndexOf( x ) # -1
  170. END Contains;
  171. PROCEDURE Clear*;
  172. VAR i: LONGINT;
  173. BEGIN
  174. FOR i := 0 TO count - 1 DO list[i] := NIL END;
  175. count := 0
  176. END Clear;
  177. PROCEDURE GrowAndSet*(i: LONGINT; x: ANY);
  178. BEGIN
  179. IF (i<0) THEN HALT(101) END;
  180. WHILE i>=LEN(list) DO Grow END;
  181. list[i] := x;
  182. INC(i); IF count < i THEN count := i END;
  183. END GrowAndSet;
  184. PROCEDURE Sort*(comparisonFunction: ComparisonFunction);
  185. BEGIN
  186. IF count > 0 THEN
  187. QuickSort(comparisonFunction, 0, count - 1)
  188. END
  189. END Sort;
  190. PROCEDURE QuickSort(comparisonFunction: ComparisonFunction; lo, hi: LONGINT);
  191. VAR
  192. i, j: LONGINT;
  193. x, t: ANY;
  194. BEGIN
  195. i := lo; j := hi;
  196. x := list[(lo + hi) DIV 2];
  197. WHILE i <= j DO
  198. WHILE comparisonFunction(list[i], x) DO INC(i) END;
  199. WHILE comparisonFunction(x, list[j]) DO DEC(j) END;
  200. IF i <= j THEN
  201. (*IF (i < j) & comparisonFunction(list[j], list[i]) THEN*)
  202. t := list[i]; list[i] := list[j]; list[j] := t; (* swap i and j *)
  203. (*END;*)
  204. INC(i); DEC(j)
  205. END
  206. END;
  207. IF lo < j THEN QuickSort(comparisonFunction, lo, j) END;
  208. IF i < hi THEN QuickSort(comparisonFunction, i, hi) END
  209. END QuickSort;
  210. END List;
  211. IntegerList* = OBJECT
  212. VAR list: IntegerArray;
  213. count-: LONGINT;
  214. PROCEDURE & InitList*(initialSize: LONGINT) ;
  215. BEGIN
  216. INC( lists ); NEW( list, initialSize ); count := 0;
  217. END InitList;
  218. PROCEDURE Length*( ): LONGINT;
  219. BEGIN RETURN count END Length;
  220. PROCEDURE Grow;
  221. VAR old: IntegerArray; i: LONGINT;
  222. BEGIN
  223. INC( enlarged ); old := list; NEW( list, LEN( list ) * 4 );
  224. FOR i := 0 TO count - 1 DO list[i] := old[i] END
  225. END Grow;
  226. PROCEDURE Get*( i: LONGINT ): LONGINT;
  227. BEGIN
  228. IF (i < 0) OR (i >= count) THEN HALT( 101 ) END;
  229. RETURN list[i]
  230. END Get;
  231. PROCEDURE Set*(i: LONGINT; x: LONGINT);
  232. BEGIN
  233. IF (i < 0) OR (i >= count) THEN HALT( 101 ) END;
  234. list[i] := x;
  235. END Set;
  236. PROCEDURE Add*( x: LONGINT );
  237. BEGIN
  238. IF count = LEN( list ) THEN Grow END;
  239. list[count] := x; INC( count )
  240. END Add;
  241. PROCEDURE Prepend*(x: LONGINT);
  242. VAR i: LONGINT;
  243. BEGIN
  244. IF count = LEN( list ) THEN Grow END;
  245. FOR i := count-1 TO 0 BY - 1 DO
  246. list[i+1] := list[i];
  247. END;
  248. list[0] := x; INC(count);
  249. END Prepend;
  250. PROCEDURE Append*(x: IntegerList);
  251. VAR i: LONGINT;
  252. BEGIN
  253. FOR i := 0 TO x.Length() - 1 DO
  254. Add(x.Get(i));
  255. END;
  256. END Append;
  257. PROCEDURE Remove*( x: LONGINT );
  258. VAR i: LONGINT;
  259. BEGIN
  260. i := 0;
  261. WHILE (i < count) & (list[i] # x) DO INC( i ) END;
  262. IF i < count THEN
  263. WHILE (i < count - 1) DO list[i] := list[i + 1]; INC( i ) END;
  264. DEC( count );
  265. END
  266. END Remove;
  267. PROCEDURE RemoveByIndex*( i: LONGINT );
  268. BEGIN
  269. IF i < count THEN
  270. WHILE (i < count - 1) DO list[i] := list[i + 1]; INC( i ) END;
  271. DEC( count );
  272. END
  273. END RemoveByIndex;
  274. PROCEDURE Insert*( i,x: LONGINT);
  275. VAR j: LONGINT;
  276. BEGIN
  277. ASSERT((i >= 0) & (i < count));
  278. IF count = LEN( list ) THEN Grow END; INC( count );
  279. j := count - 2;
  280. WHILE (j >= i) DO list[j+1] := list[j]; DEC( j ) END;
  281. list[i] := x;
  282. END Insert;
  283. PROCEDURE Replace*( x, y: LONGINT );
  284. VAR i: LONGINT;
  285. BEGIN
  286. i := IndexOf( x );
  287. IF i >= 0 THEN list[i] := y END
  288. END Replace;
  289. PROCEDURE IndexOf*( x: LONGINT ): LONGINT;
  290. VAR i: LONGINT;
  291. BEGIN
  292. i := 0;
  293. WHILE i < count DO
  294. IF list[i] = x THEN RETURN i END;
  295. INC( i )
  296. END;
  297. RETURN -1
  298. END IndexOf;
  299. PROCEDURE Contains*( x: LONGINT ): BOOLEAN;
  300. BEGIN RETURN IndexOf( x ) # -1; END Contains;
  301. PROCEDURE Clear*;
  302. BEGIN count := 0 END Clear;
  303. END IntegerList;
  304. (* Supports get, add, contain, append in O(1) *)
  305. Bag* = OBJECT
  306. VAR
  307. count-: LONGINT;
  308. list: List;
  309. PROCEDURE & InitBag* ;
  310. BEGIN
  311. Clear();
  312. END InitBag;
  313. PROCEDURE Length*( ): LONGINT;
  314. BEGIN
  315. RETURN list.Length();
  316. END Length;
  317. PROCEDURE Get*( i: LONGINT ): ANY;
  318. BEGIN RETURN list.Get(i); END Get;
  319. PROCEDURE Add*( x: ANY );
  320. BEGIN
  321. ASSERT( x # NIL );
  322. IF ~Contains(x) THEN
  323. list.Add(x);
  324. END;
  325. END Add;
  326. PROCEDURE Append*(x: Bag);
  327. VAR i: LONGINT;
  328. BEGIN
  329. FOR i := 0 TO x.Length() - 1 DO
  330. IF ~Contains(x.Get(i)) THEN
  331. Add(x.Get(i));
  332. END;
  333. END;
  334. END Append;
  335. PROCEDURE Remove*( x: ANY );
  336. BEGIN
  337. list.Remove(x);
  338. END Remove;
  339. PROCEDURE Contains*( x: ANY ): BOOLEAN;
  340. BEGIN RETURN list.Contains(x); END Contains;
  341. PROCEDURE Clear*;
  342. BEGIN
  343. count := 0;
  344. NEW(list,InitListSize);
  345. list.multipleAllowed := TRUE; list.nilAllowed := TRUE;
  346. END Clear;
  347. END Bag;
  348. (* Supports get, add, contain, append in O(1) *)
  349. IntegerBag* = OBJECT
  350. VAR
  351. count-: LONGINT;
  352. list: IntegerList;
  353. PROCEDURE & InitBag* ;
  354. BEGIN
  355. Clear();
  356. END InitBag;
  357. PROCEDURE Length*( ): LONGINT;
  358. BEGIN
  359. RETURN list.Length();
  360. END Length;
  361. PROCEDURE Get*( i: LONGINT ):LONGINT;
  362. BEGIN RETURN list.Get(i); END Get;
  363. PROCEDURE Add*( x: LONGINT );
  364. BEGIN
  365. IF ~Contains(x) THEN
  366. list.Add(x);
  367. END;
  368. END Add;
  369. PROCEDURE Append*(x: IntegerBag);
  370. VAR i: LONGINT;
  371. BEGIN
  372. FOR i := 0 TO x.Length() - 1 DO
  373. IF ~Contains(x.Get(i)) THEN
  374. Add(x.Get(i));
  375. END;
  376. END;
  377. END Append;
  378. PROCEDURE Remove*(x: LONGINT );
  379. BEGIN
  380. list.Remove(x);
  381. END Remove;
  382. PROCEDURE Contains*( x: LONGINT ): BOOLEAN;
  383. BEGIN RETURN list.Contains(x); END Contains;
  384. PROCEDURE Clear*;
  385. BEGIN
  386. count := 0;
  387. NEW(list,InitListSize);
  388. END Clear;
  389. END IntegerBag;
  390. HashEntryAny = RECORD
  391. key, value: ANY;
  392. valueInt: LONGINT;
  393. END;
  394. HashEntryInt = RECORD
  395. key, valueInt: LONGINT;
  396. value: ANY;
  397. END;
  398. HashAnyArray = POINTER TO ARRAY OF HashEntryAny;
  399. HashIntArray = POINTER TO ARRAY OF HashEntryInt;
  400. HashTable* = OBJECT
  401. VAR
  402. table: HashAnyArray;
  403. size: LONGINT;
  404. used-: LONGINT;
  405. maxLoadFactor: REAL;
  406. (* Interface *)
  407. PROCEDURE & Init* (initialSize: LONGINT);
  408. BEGIN
  409. ASSERT(initialSize > 2);
  410. NEW(table, initialSize);
  411. size := initialSize;
  412. used := 0;
  413. maxLoadFactor := 0.75;
  414. END Init;
  415. PROCEDURE Put*(key, value: ANY);
  416. VAR hash: LONGINT;
  417. BEGIN
  418. ASSERT(used < size);
  419. ASSERT(key # NIL);
  420. hash := HashValue(key);
  421. IF table[hash].key = NIL THEN
  422. INC(used, 1);
  423. ELSE
  424. ASSERT(table[hash].key = key);
  425. END;
  426. table[hash].key := key;
  427. table[hash].value := value;
  428. IF (used / size) > maxLoadFactor THEN Grow END;
  429. END Put;
  430. PROCEDURE Get*(key: ANY):ANY;
  431. BEGIN
  432. RETURN table[HashValue(key)].value;
  433. END Get;
  434. PROCEDURE Has*(key: ANY):BOOLEAN;
  435. BEGIN
  436. RETURN table[HashValue(key)].key = key;
  437. END Has;
  438. PROCEDURE Length*():LONGINT;
  439. BEGIN RETURN used; END Length;
  440. PROCEDURE Clear*;
  441. VAR i: LONGINT;
  442. BEGIN FOR i := 0 TO size - 1 DO table[i].key := NIL; table[i].value := NIL; table[i].valueInt := 0 END; END Clear;
  443. (* Interface for integer values *)
  444. PROCEDURE PutInt*(key: ANY; value: LONGINT);
  445. VAR hash: LONGINT;
  446. BEGIN
  447. ASSERT(used < size);
  448. hash := HashValue(key);
  449. IF table[hash].key = NIL THEN
  450. INC(used, 1);
  451. END;
  452. table[hash].key := key;
  453. table[hash].valueInt := value;
  454. IF (used / size) > maxLoadFactor THEN Grow END;
  455. END PutInt;
  456. PROCEDURE GetInt*(key: ANY):LONGINT;
  457. BEGIN RETURN table[HashValue(key)].valueInt; END GetInt;
  458. (* Internals *)
  459. (* only correctly working, if NIL key cannot be entered *)
  460. PROCEDURE HashValue(key: ANY):LONGINT;
  461. VAR value, h1, h2, i: LONGINT;
  462. BEGIN
  463. value := SYSTEM.VAL(LONGINT, key) DIV SIZEOF(ADDRESS);
  464. i := 0;
  465. h1 := value MOD size;
  466. h2 := 1; (* Linear probing *)
  467. REPEAT
  468. value := (h1 + i*h2) MOD size;
  469. INC(i);
  470. UNTIL((table[value].key = NIL) OR (table[value].key = key) OR (i > size));
  471. ASSERT((table[value].key = NIL) & (table[value].value = NIL) OR (table[value].key = key));
  472. RETURN value;
  473. END HashValue;
  474. PROCEDURE Grow;
  475. VAR oldTable: HashAnyArray; oldSize, i: LONGINT; key: ANY;
  476. BEGIN
  477. oldSize := size;
  478. oldTable := table;
  479. Init(size*2);
  480. FOR i := 0 TO oldSize-1 DO
  481. key := oldTable[i].key;
  482. IF key # NIL THEN
  483. IF oldTable[i].value # NIL THEN
  484. Put(key, oldTable[i].value);
  485. ELSE
  486. PutInt(key, oldTable[i].valueInt);
  487. END;
  488. END;
  489. END;
  490. END Grow;
  491. END HashTable;
  492. IntIterator*= OBJECT
  493. VAR
  494. table: HashIntArray;
  495. count : LONGINT;
  496. PROCEDURE & Init(t: HashIntArray);
  497. BEGIN
  498. table := t;
  499. count := -1;
  500. END Init;
  501. PROCEDURE GetNext*(VAR key: LONGINT; VAR value: ANY): BOOLEAN;
  502. BEGIN
  503. REPEAT
  504. INC(count);
  505. UNTIL (count = LEN(table)) OR (table[count].value # NIL);
  506. IF count = LEN(table) THEN
  507. RETURN FALSE
  508. END;
  509. key := table[count].key;
  510. value := table[count].value;
  511. RETURN TRUE;
  512. END GetNext;
  513. END IntIterator;
  514. HashTableInt* = OBJECT
  515. VAR
  516. table: HashIntArray;
  517. size: LONGINT;
  518. used-: LONGINT;
  519. maxLoadFactor: REAL;
  520. (* Interface *)
  521. PROCEDURE & Init* (initialSize: LONGINT);
  522. BEGIN
  523. ASSERT(initialSize > 2);
  524. NEW(table, initialSize);
  525. size := initialSize;
  526. used := 0;
  527. maxLoadFactor := 0.75;
  528. END Init;
  529. PROCEDURE Put*(key: LONGINT; value: ANY);
  530. VAR hash: LONGINT;
  531. BEGIN
  532. ASSERT(key # 0);
  533. ASSERT(used < size);
  534. hash := HashValue(key);
  535. IF table[hash].key = 0 THEN
  536. INC(used, 1);
  537. END;
  538. table[hash].key := key;
  539. table[hash].value := value;
  540. IF (used / size) > maxLoadFactor THEN Grow END;
  541. END Put;
  542. PROCEDURE Get*(key: LONGINT):ANY;
  543. BEGIN
  544. RETURN table[HashValue(key)].value;
  545. END Get;
  546. PROCEDURE Has*(key: LONGINT):BOOLEAN;
  547. BEGIN
  548. RETURN table[HashValue(key)].key = key;
  549. END Has;
  550. PROCEDURE Length*():LONGINT;
  551. BEGIN RETURN used; END Length;
  552. PROCEDURE Clear*;
  553. VAR i: LONGINT;
  554. BEGIN FOR i := 0 TO size - 1 DO table[i].key := 0; END; END Clear;
  555. (* Interface for integer values *)
  556. PROCEDURE PutInt*(key, value: LONGINT);
  557. VAR hash: LONGINT;
  558. BEGIN
  559. (*ASSERT(key # 0);*)
  560. ASSERT(used < size);
  561. hash := HashValue(key);
  562. IF table[hash].key = 0 THEN
  563. INC(used, 1);
  564. END;
  565. table[hash].key := key;
  566. table[hash].valueInt := value;
  567. IF (used / size) > maxLoadFactor THEN Grow END;
  568. END PutInt;
  569. PROCEDURE GetInt*(key: LONGINT):LONGINT;
  570. BEGIN RETURN table[HashValue(key)].valueInt; END GetInt;
  571. (* Internals *)
  572. PROCEDURE HashValue(key: LONGINT):LONGINT;
  573. VAR value, h1, h2, i: LONGINT;
  574. BEGIN
  575. i := 0;
  576. value := key;
  577. h1 := key MOD size;
  578. h2 := 1; (* Linear probing *)
  579. REPEAT
  580. value := (h1 + i*h2) MOD size;
  581. INC(i);
  582. UNTIL((table[value].key = 0) OR (table[value].key = key) OR (i > size));
  583. ASSERT((table[value].key = 0) OR (table[value].key = key));
  584. RETURN value;
  585. END HashValue;
  586. PROCEDURE Grow;
  587. VAR oldTable: HashIntArray; oldSize, i, key: LONGINT;
  588. BEGIN
  589. oldSize := size;
  590. oldTable := table;
  591. Init(size*2);
  592. FOR i := 0 TO oldSize-1 DO
  593. key := oldTable[i].key;
  594. IF key # 0 THEN
  595. IF oldTable[i].value # NIL THEN
  596. Put(key, oldTable[i].value);
  597. ELSE
  598. PutInt(key, oldTable[i].valueInt);
  599. END;
  600. END;
  601. END;
  602. END Grow;
  603. PROCEDURE GetIterator*(): IntIterator;
  604. VAR iterator: IntIterator;
  605. BEGIN
  606. NEW(iterator, table);
  607. RETURN iterator;
  608. END GetIterator;
  609. END HashTableInt;
  610. HashEntrySegmentedName = RECORD
  611. key: ObjectFile.SegmentedName; (* key[0]= MIN(LONGINT) <=> empty *)
  612. value: ANY;
  613. END;
  614. HashSegmentedNameArray = POINTER TO ARRAY OF HashEntrySegmentedName;
  615. HashTableSegmentedName* = OBJECT
  616. VAR
  617. table: HashSegmentedNameArray;
  618. size: LONGINT;
  619. used-: LONGINT;
  620. maxLoadFactor: REAL;
  621. (* Interface *)
  622. PROCEDURE & Init* (initialSize: LONGINT);
  623. BEGIN
  624. ASSERT(initialSize > 2);
  625. NEW(table, initialSize);
  626. size := initialSize;
  627. used := 0;
  628. maxLoadFactor := 0.75;
  629. Clear;
  630. END Init;
  631. PROCEDURE Put*(CONST key: SegmentedName; value: ANY);
  632. VAR hash: LONGINT;
  633. BEGIN
  634. ASSERT(used < size);
  635. hash := HashValue(key);
  636. IF table[hash].key[0] < 0 THEN
  637. INC(used, 1);
  638. END;
  639. table[hash].key := key;
  640. table[hash].value := value;
  641. IF (used / size) > maxLoadFactor THEN Grow END;
  642. END Put;
  643. PROCEDURE Get*(CONST key: SegmentedName):ANY;
  644. BEGIN
  645. RETURN table[HashValue(key)].value;
  646. END Get;
  647. PROCEDURE Has*(CONST key: SegmentedName):BOOLEAN;
  648. BEGIN
  649. RETURN table[HashValue(key)].key = key;
  650. END Has;
  651. PROCEDURE Length*():LONGINT;
  652. BEGIN RETURN used; END Length;
  653. PROCEDURE Clear*;
  654. VAR i: LONGINT;
  655. BEGIN FOR i := 0 TO size - 1 DO table[i].key[0] := -1; END; END Clear;
  656. (* Internals *)
  657. PROCEDURE Hash*(CONST name: SegmentedName): LONGINT;
  658. VAR fp,i: LONGINT;
  659. BEGIN
  660. fp := name[0]; i := 1;
  661. WHILE (i<LEN(name)) & (name[i] >= 0) DO
  662. fp:=SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, ROT(fp, 7)) / SYSTEM.VAL(SET, name[i]));
  663. INC(i);
  664. END;
  665. RETURN fp
  666. END Hash;
  667. PROCEDURE HashValue(CONST key: SegmentedName):LONGINT;
  668. VAR value, h,i: LONGINT;
  669. BEGIN
  670. ASSERT(key[0] >= 0);
  671. h := Hash(key);
  672. i := 0;
  673. REPEAT
  674. value := (h + i) MOD size;
  675. INC(i);
  676. UNTIL((table[value].key[0] < 0) OR (table[value].key = key) OR (i > size));
  677. ASSERT((table[value].key[0] <0 ) OR (table[value].key = key));
  678. RETURN value;
  679. END HashValue;
  680. PROCEDURE Grow;
  681. VAR oldTable: HashSegmentedNameArray; oldSize, i: LONGINT; key: SegmentedName;
  682. BEGIN
  683. oldSize := size;
  684. oldTable := table;
  685. Init(size*2);
  686. FOR i := 0 TO oldSize-1 DO
  687. key := oldTable[i].key;
  688. IF key[0] # MIN(LONGINT) THEN
  689. IF oldTable[i].value # NIL THEN
  690. Put(key, oldTable[i].value);
  691. END;
  692. END;
  693. END;
  694. END Grow;
  695. END HashTableSegmentedName;
  696. (* Hash table supporting 2 keys *)
  697. HashTable2D* = OBJECT(HashTable);
  698. VAR
  699. initialSize: LONGINT;
  700. (* Interface *)
  701. PROCEDURE & Init* (initialSize: LONGINT);
  702. BEGIN
  703. Init^(initialSize);
  704. SELF.initialSize := initialSize;
  705. END Init;
  706. PROCEDURE Get2D*(key1, key2: ANY):ANY;
  707. VAR
  708. any: ANY;
  709. second: HashTable;
  710. BEGIN
  711. any := Get(key1);
  712. second := any(HashTable);
  713. RETURN second.Get(key2);
  714. END Get2D;
  715. PROCEDURE Put2D*(key1, key2, value: ANY);
  716. VAR
  717. any: ANY;
  718. second: HashTable;
  719. BEGIN
  720. IF ~Has(key1) THEN
  721. NEW(second, initialSize);
  722. Put(key1, second);
  723. ELSE
  724. any := Get(key1);
  725. second := any(HashTable);
  726. END;
  727. second.Put(key2, value);
  728. END Put2D;
  729. PROCEDURE Has2D*(key1, key2: ANY):BOOLEAN;
  730. VAR
  731. any: ANY;
  732. second: HashTable;
  733. BEGIN
  734. IF ~Has(key1) THEN RETURN FALSE; END;
  735. any := Get(key1);
  736. second := any(HashTable);
  737. RETURN second.Has(key2);
  738. END Has2D;
  739. END HashTable2D;
  740. (* Data structure implementing a stack using lists *)
  741. Stack* = OBJECT
  742. VAR
  743. list: List;
  744. PROCEDURE & Init*;
  745. BEGIN NEW(list,InitListSize); END Init;
  746. (* Push on top of stack *)
  747. PROCEDURE Push*(x: ANY);
  748. BEGIN list.Add(x); END Push;
  749. (* Get top element *)
  750. PROCEDURE Peek*():ANY;
  751. BEGIN RETURN list.Get(list.Length() - 1); END Peek;
  752. (* Get and remove top element *)
  753. PROCEDURE Pop*():ANY;
  754. VAR old: ANY;
  755. BEGIN
  756. old := Peek();
  757. RemoveTop();
  758. RETURN old;
  759. END Pop;
  760. (* Remove top element without reading it *)
  761. PROCEDURE RemoveTop*;
  762. BEGIN list.RemoveByIndex(list.Length() - 1); END RemoveTop;
  763. (* Check if empty *)
  764. PROCEDURE Empty*():BOOLEAN;
  765. BEGIN RETURN list.Length() = 0; END Empty;
  766. PROCEDURE Length*():LONGINT;
  767. BEGIN RETURN list.count; END Length;
  768. END Stack;
  769. (* Data structure implementing a stack using lists *)
  770. IntegerStack* = OBJECT
  771. VAR
  772. list: IntegerList;
  773. PROCEDURE & Init*;
  774. BEGIN NEW(list,InitListSize); END Init;
  775. (* Push on top of stack *)
  776. PROCEDURE Push*(x: LONGINT);
  777. BEGIN list.Add(x); END Push;
  778. (* Get top element *)
  779. PROCEDURE Peek*():LONGINT;
  780. BEGIN RETURN list.Get(list.Length() - 1); END Peek;
  781. (* Get and remove top element *)
  782. PROCEDURE Pop*():LONGINT;
  783. VAR old: LONGINT;
  784. BEGIN
  785. old := Peek();
  786. RemoveTop();
  787. RETURN old;
  788. END Pop;
  789. (* Remove top element without reading it *)
  790. PROCEDURE RemoveTop*;
  791. BEGIN list.RemoveByIndex(list.Length() - 1); END RemoveTop;
  792. (* Check if empty *)
  793. PROCEDURE Empty*():BOOLEAN;
  794. BEGIN RETURN list.Length() = 0; END Empty;
  795. PROCEDURE Length*():LONGINT;
  796. BEGIN RETURN list.count; END Length;
  797. END IntegerStack;
  798. QueueEntry = POINTER TO RECORD
  799. value: ANY;
  800. next: QueueEntry;
  801. END;
  802. Queue* = OBJECT
  803. VAR
  804. top, last: QueueEntry;
  805. PROCEDURE & Init *;
  806. BEGIN
  807. top := NIL; last := NIL;
  808. END Init;
  809. (* Add to end of queue *)
  810. PROCEDURE Append*(x: ANY);
  811. VAR entry: QueueEntry;
  812. BEGIN
  813. NEW(entry);
  814. entry.value := x;
  815. IF top = NIL THEN
  816. top := entry;
  817. ELSE
  818. last.next := entry;
  819. END;
  820. last := entry;
  821. END Append;
  822. (* Get top element *)
  823. PROCEDURE Peek*():ANY;
  824. BEGIN
  825. RETURN top.value;
  826. END Peek;
  827. (* Get and remove top element *)
  828. PROCEDURE Pop*():ANY;
  829. VAR old: QueueEntry;
  830. BEGIN
  831. ASSERT(~Empty());
  832. old := top;
  833. top := top.next;
  834. RETURN old.value;
  835. END Pop;
  836. (* Check if empty *)
  837. PROCEDURE Empty*():BOOLEAN;
  838. BEGIN
  839. RETURN top = NIL;
  840. END Empty;
  841. END Queue;
  842. PQItem = RECORD
  843. key: LONGINT;
  844. value: ANY;
  845. END;
  846. PQItemList = POINTER TO ARRAY OF PQItem;
  847. (* Priority queue using binary heap *)
  848. PriorityQueue* = OBJECT
  849. VAR
  850. heap: PQItemList;
  851. count-: LONGINT;
  852. (** Interface **)
  853. PROCEDURE & Init(size: LONGINT);
  854. BEGIN
  855. NEW(heap, size + 1);
  856. count := 0;
  857. END Init;
  858. PROCEDURE Min*():ANY; (* O(n) *)
  859. BEGIN
  860. ASSERT(count > 0);
  861. RETURN heap[1].value;
  862. END Min;
  863. PROCEDURE RemoveMin*():ANY; (* O(log n) *)
  864. VAR min: ANY;
  865. BEGIN
  866. min := Min();
  867. heap[1] := heap[count];
  868. DEC(count);
  869. IF count > 0 THEN BubbleDown(1); END;
  870. RETURN min;
  871. END RemoveMin;
  872. PROCEDURE Insert*(key: LONGINT; value: ANY); (* O(log n) *)
  873. VAR index: LONGINT;
  874. BEGIN
  875. INC(count);
  876. index := count;
  877. heap[index].key := key;
  878. heap[index].value := value;
  879. BubbleUp(index);
  880. END Insert;
  881. PROCEDURE Empty*():BOOLEAN;
  882. BEGIN
  883. RETURN count = 0;
  884. END Empty;
  885. (** Implementation **)
  886. PROCEDURE BubbleUp(VAR index: LONGINT);
  887. VAR swap: PQItem;
  888. BEGIN
  889. WHILE (index > 1) & (heap[index].key < heap[index DIV 2].key) DO
  890. swap := heap[index DIV 2];
  891. heap[index DIV 2] := heap[index];
  892. heap[index] := swap;
  893. index := index DIV 2;
  894. END;
  895. END BubbleUp;
  896. PROCEDURE BubbleDown(index: LONGINT);
  897. VAR min, minkey: LONGINT; swap: PQItem;
  898. PROCEDURE Child(child: LONGINT);
  899. BEGIN
  900. IF (child <= count) & (heap[child].key < minkey) THEN
  901. min := child;
  902. minkey := heap[child].key;
  903. END;
  904. END Child;
  905. BEGIN
  906. REPEAT
  907. min := 0;
  908. minkey := heap[index].key;
  909. Child(index * 2);
  910. Child((index * 2) + 1);
  911. IF min # 0 THEN
  912. swap := heap[min];
  913. heap[min] := heap[index];
  914. heap[index] := swap;
  915. index := min;
  916. END;
  917. UNTIL
  918. min = 0;
  919. END BubbleDown;
  920. END PriorityQueue;
  921. IndexList = POINTER TO ARRAY OF LONGINT;
  922. Edge* = OBJECT
  923. VAR
  924. from-, to-: Block;
  925. PROCEDURE Accept(v: GraphVisitor);
  926. BEGIN v.VisitEdge(SELF); END Accept;
  927. END Edge;
  928. Graph* = OBJECT
  929. VAR
  930. firstBlock*, lastBlock-: Block;
  931. blocks*: BlockList;
  932. edges-: EdgeList;
  933. edgesLookup: HashTable2D;
  934. PROCEDURE & Init *;
  935. BEGIN
  936. NEW(blocks,InitListSize);
  937. NEW(edges,InitListSize);
  938. NEW(edgesLookup, 1024);
  939. END Init;
  940. PROCEDURE AddBlock*(block: Block);
  941. BEGIN
  942. IF blocks.Length() = 0 THEN firstBlock := block; END;
  943. block.index := blocks.Length();
  944. blocks.Add(block);
  945. lastBlock := block;
  946. END AddBlock;
  947. PROCEDURE Connect*(from, to: Block);
  948. VAR edge: Edge;
  949. BEGIN
  950. IF edgesLookup.Has2D(from, to) THEN RETURN; END;
  951. from.successors.Add(to);
  952. to.predecessors.Add(from);
  953. NEW(edge);
  954. edge.from := from;
  955. edge.to := to;
  956. edges.Add(edge);
  957. edgesLookup.Put2D(from, to, edge);
  958. END Connect;
  959. PROCEDURE Split*(from, to: Block);
  960. BEGIN
  961. from.successors.Remove(to);
  962. to.predecessors.Remove(from);
  963. edges.Remove(edgesLookup.Get2D(from, to));
  964. END Split;
  965. (* Reorder blocks so that they form a reverse post order *)
  966. PROCEDURE ReIndex*;
  967. VAR b: Block; i: LONGINT; done: POINTER TO ARRAY OF BOOLEAN; new: BlockList;
  968. PROCEDURE Work(b: Block);
  969. VAR i: LONGINT; p: Block;
  970. BEGIN
  971. done[b.index] := TRUE;
  972. FOR i := 0 TO b.successors.Length() - 1 DO
  973. p := b.successors.GetBlock(i);
  974. IF ~done[p.index] THEN
  975. Work(p);
  976. END;
  977. END;
  978. new.Add(b);
  979. END Work;
  980. BEGIN
  981. NEW(new,InitListSize);
  982. NEW(done, blocks.Length());
  983. i := 0;
  984. Work(blocks.GetBlock(0));
  985. NEW(blocks,InitListSize);
  986. FOR i := new.Length() - 1 TO 0 BY -1 DO
  987. b := new.GetBlock(i);
  988. b.index := blocks.Length();
  989. blocks.Add(b);
  990. END;
  991. END ReIndex;
  992. (* Calculate dominance tree. Algorithm taken from:
  993. "A simple, fast dominance algorithm" (Cooper, Harvey, Kennedy) *)
  994. PROCEDURE CalculateDominance*;
  995. VAR
  996. doms: IndexList;
  997. i, j, len, runner, newIdom: LONGINT;
  998. changed: BOOLEAN;
  999. block, pred: Block;
  1000. PROCEDURE Intersect(b1, b2: LONGINT):LONGINT;
  1001. BEGIN
  1002. WHILE(b1 # b2) DO
  1003. WHILE(b1 > b2) DO
  1004. IF b1 = doms[b1] THEN HALT(100); END;
  1005. b1 := doms[b1];
  1006. END;
  1007. WHILE(b2 > b1) DO
  1008. IF b2 = doms[b2] THEN HALT(100); END;
  1009. b2 := doms[b2];
  1010. END;
  1011. END;
  1012. RETURN b1;
  1013. END Intersect;
  1014. BEGIN
  1015. (* Initialize the arrays *)
  1016. len := blocks.Length();
  1017. NEW(doms, len);
  1018. FOR i := 0 TO len - 1 DO
  1019. doms[i] := -1;
  1020. END;
  1021. doms[0] := 0;
  1022. (* Iteration loop *)
  1023. changed := TRUE;
  1024. WHILE(changed) DO
  1025. changed := FALSE;
  1026. FOR i := 1 TO len - 1 DO
  1027. block := blocks.GetBlock(i);
  1028. pred := block.predecessors.GetBlock(0);
  1029. newIdom := pred.index;
  1030. FOR j := 1 TO block.predecessors.Length() - 1 DO
  1031. pred := block.predecessors.GetBlock(j);
  1032. IF doms[pred.index] # -1 THEN
  1033. newIdom := Intersect(pred.index, newIdom);
  1034. END;
  1035. END;
  1036. IF doms[i] # newIdom THEN
  1037. doms[i] := newIdom;
  1038. changed := TRUE;
  1039. END;
  1040. END;
  1041. END;
  1042. FOR i := 0 TO len - 1 DO
  1043. block := blocks.GetBlock(i);
  1044. (* Set immediate dominators *)
  1045. block.immediateDominator := doms[i];
  1046. (* Calculate frontier *)
  1047. IF block.predecessors.Length() >= 2 THEN
  1048. FOR j := 0 TO block.predecessors.Length() - 1 DO
  1049. pred := block.predecessors.GetBlock(j);
  1050. runner := pred.index;
  1051. WHILE runner # doms[block.index] DO
  1052. pred := blocks.GetBlock(runner);
  1053. IF ~pred.dominanceFrontier.Contains(block) THEN
  1054. pred.dominanceFrontier.Add(block);
  1055. END;
  1056. runner := doms[runner];
  1057. END;
  1058. END;
  1059. END;
  1060. END;
  1061. END CalculateDominance;
  1062. END Graph;
  1063. BlockList* = OBJECT(List)
  1064. VAR
  1065. PROCEDURE GetBlock*(i: LONGINT):Block;
  1066. VAR block: ANY;
  1067. BEGIN
  1068. block := Get(i);
  1069. RETURN block(Block);
  1070. END GetBlock;
  1071. PROCEDURE GetIndex*(i: LONGINT):LONGINT;
  1072. VAR block: Block;
  1073. BEGIN
  1074. block := GetBlock(i);
  1075. RETURN block.index;
  1076. END GetIndex;
  1077. END BlockList;
  1078. EdgeList* = OBJECT(List)
  1079. VAR
  1080. PROCEDURE GetEdge*(i: LONGINT):Edge;
  1081. VAR
  1082. edge: ANY;
  1083. BEGIN
  1084. edge := Get(i);
  1085. RETURN edge(Edge);
  1086. END GetEdge;
  1087. END EdgeList;
  1088. Block* = OBJECT
  1089. VAR
  1090. predecessors-, successors-, dominanceFrontier-: BlockList;
  1091. index*, immediateDominator*: LONGINT;
  1092. PROCEDURE & Init*;
  1093. BEGIN
  1094. NEW(predecessors,InitListSize);
  1095. NEW(successors,InitListSize);
  1096. NEW(dominanceFrontier,InitListSize);
  1097. END Init;
  1098. PROCEDURE Accept(v: GraphVisitor);
  1099. BEGIN v.VisitBlock(SELF); END Accept;
  1100. PROCEDURE PredecessorIndex*(block: Block):LONGINT;
  1101. VAR i: LONGINT;
  1102. BEGIN
  1103. FOR i := 0 TO predecessors.Length() - 1 DO
  1104. IF predecessors.Get(i) = block THEN
  1105. RETURN i;
  1106. END;
  1107. END;
  1108. HALT(100);
  1109. END PredecessorIndex;
  1110. END Block;
  1111. ContentFunction = PROCEDURE {DELEGATE} (block: Block);
  1112. GraphVisitor* = OBJECT
  1113. VAR
  1114. block-: Block;
  1115. edge-: Edge;
  1116. graph-: Graph;
  1117. PROCEDURE VisitEdge*(edge: Edge);
  1118. BEGIN END VisitEdge;
  1119. PROCEDURE VisitBlock*(block: Block);
  1120. BEGIN END VisitBlock;
  1121. PROCEDURE VisitGraph*(graph: Graph);
  1122. VAR i: LONGINT;
  1123. BEGIN
  1124. SELF.graph := graph;
  1125. FOR i := 0 TO graph.blocks.Length() - 1 DO
  1126. block := graph.blocks.GetBlock(i);
  1127. block.Accept(SELF);
  1128. END;
  1129. FOR i := 0 TO graph.edges.Length() - 1 DO
  1130. edge := graph.edges.GetEdge(i);
  1131. edge.Accept(SELF);
  1132. END;
  1133. END VisitGraph;
  1134. END GraphVisitor;
  1135. (** Outputs a .dot file which can be parsed into a graph by GraphViz *)
  1136. GraphPrinter* = OBJECT(GraphVisitor)
  1137. VAR
  1138. active-: Block;
  1139. writer-: Streams.Writer;
  1140. content: ContentFunction;
  1141. PROCEDURE VisitEdge*(edge: Edge);
  1142. BEGIN
  1143. writer.String("block"); writer.Int(edge.from.index, 0);
  1144. writer.String("->");
  1145. writer.String("block"); writer.Int(edge.to.index, 0);
  1146. writer.String(";"); writer.Ln;
  1147. END VisitEdge;
  1148. PROCEDURE VisitBlock(block: Block);
  1149. VAR
  1150. i: LONGINT;
  1151. dom: Block;
  1152. BEGIN
  1153. writer.String("block");
  1154. writer.Int(block.index, 0);
  1155. writer.String(' [ label=<<table border="0" cellpadding="1" cellspacing="1"><tr><td>#');
  1156. writer.Int(block.index, 0);
  1157. writer.String("</td><td>idom=");
  1158. writer.Int(block.immediateDominator, 0);
  1159. writer.String("</td><td>df=");
  1160. FOR i := 0 TO block.dominanceFrontier.Length() - 1 DO
  1161. dom := block.dominanceFrontier.GetBlock(i);
  1162. writer.Int(dom.index, 0);
  1163. writer.String(" ");
  1164. END;
  1165. writer.String("</td></tr>");
  1166. content(block);
  1167. writer.String('</table>>]; ');
  1168. writer.Ln;
  1169. END VisitBlock;
  1170. PROCEDURE VisitGraph*(graph: Graph);
  1171. BEGIN
  1172. SELF.graph := graph;
  1173. (* Print header of dot file *)
  1174. writer.String("digraph G {"); writer.Ln;
  1175. (* Print all blocks *)
  1176. writer.String("node [shape=box]; ");
  1177. VisitGraph^(graph);
  1178. (* Footer *)
  1179. writer.Ln;
  1180. writer.String("overlap=false;"); writer.Ln;
  1181. writer.String('label=" Created with OC";'); writer.Ln;
  1182. writer.String("fontsize=12;"); writer.Ln;
  1183. writer.String("}");
  1184. END VisitGraph;
  1185. PROCEDURE SetWriter*(w: Streams.Writer);
  1186. BEGIN
  1187. writer := w;
  1188. END SetWriter;
  1189. PROCEDURE & Init*(c: ContentFunction);
  1190. BEGIN
  1191. content := c;
  1192. END Init;
  1193. END GraphPrinter;
  1194. IntegerObject = OBJECT
  1195. END IntegerObject;
  1196. Writer* = OBJECT (Streams.Writer)
  1197. VAR
  1198. indent-: LONGINT;
  1199. doindent: BOOLEAN;
  1200. w-: Streams.Writer;
  1201. PROCEDURE InitBasicWriter*( w: Streams.Writer );
  1202. BEGIN
  1203. SELF.w := w; indent := 0; doindent := TRUE;
  1204. END InitBasicWriter;
  1205. PROCEDURE & InitW(w: Streams.Writer); (* protect against use of NEW *)
  1206. BEGIN InitBasicWriter(w);
  1207. END InitW;
  1208. PROCEDURE Reset*;
  1209. BEGIN w.Reset;
  1210. END Reset;
  1211. PROCEDURE CanSetPos*( ): BOOLEAN;
  1212. BEGIN RETURN w.CanSetPos();
  1213. END CanSetPos;
  1214. PROCEDURE SetPos*( pos: LONGINT );
  1215. BEGIN w.SetPos(pos);
  1216. END SetPos;
  1217. PROCEDURE Update*;
  1218. BEGIN w.Update;
  1219. END Update;
  1220. PROCEDURE Pos*( ): LONGINT;
  1221. BEGIN RETURN w.Pos()
  1222. END Pos;
  1223. PROCEDURE Indent;
  1224. VAR i: LONGINT;
  1225. BEGIN
  1226. IF doindent THEN
  1227. FOR i := 0 TO indent-1 DO
  1228. w.Char(9X);
  1229. END;
  1230. doindent := FALSE
  1231. END;
  1232. END Indent;
  1233. PROCEDURE Char*( x: CHAR );
  1234. BEGIN Indent; w.Char(x);
  1235. END Char;
  1236. PROCEDURE Bytes*(CONST x: ARRAY OF CHAR; ofs, len: LONGINT );
  1237. BEGIN w.Bytes(x,ofs,len);
  1238. END Bytes;
  1239. PROCEDURE RawSInt*( x: SHORTINT );
  1240. BEGIN w.RawSInt(x)
  1241. END RawSInt;
  1242. PROCEDURE RawInt*( x: INTEGER );
  1243. BEGIN w.RawInt(x)
  1244. END RawInt;
  1245. PROCEDURE RawLInt*( x: LONGINT );
  1246. BEGIN w.RawLInt(x)
  1247. END RawLInt;
  1248. PROCEDURE RawHInt*( x: HUGEINT );
  1249. BEGIN w.RawHInt(x)
  1250. END RawHInt;
  1251. PROCEDURE Net32*( x: LONGINT );
  1252. BEGIN w.Net32(x)
  1253. END Net32;
  1254. PROCEDURE Net16*( x: LONGINT );
  1255. BEGIN w.Net16(x)
  1256. END Net16;
  1257. PROCEDURE Net8*( x: LONGINT );
  1258. BEGIN w.Net8(x)
  1259. END Net8;
  1260. PROCEDURE RawSet*( x: SET );
  1261. BEGIN w.RawSet(x)
  1262. END RawSet;
  1263. PROCEDURE RawBool*( x: BOOLEAN );
  1264. BEGIN w.RawBool(x)
  1265. END RawBool;
  1266. PROCEDURE RawReal*( x: REAL );
  1267. BEGIN w.RawReal(x)
  1268. END RawReal;
  1269. PROCEDURE RawLReal*( x: LONGREAL );
  1270. BEGIN w.RawLReal(x)
  1271. END RawLReal;
  1272. PROCEDURE RawString*(CONST x: ARRAY OF CHAR );
  1273. BEGIN w.RawString(x)
  1274. END RawString;
  1275. PROCEDURE RawNum*( x: LONGINT );
  1276. BEGIN w.RawNum(x)
  1277. END RawNum;
  1278. PROCEDURE Ln*;
  1279. BEGIN w.Ln; doindent := TRUE;
  1280. END Ln;
  1281. PROCEDURE String*(CONST x: ARRAY OF CHAR );
  1282. BEGIN Indent; w.String(x)
  1283. END String;
  1284. PROCEDURE Int*( x: HUGEINT; wd: LONGINT );
  1285. BEGIN Indent; w.Int(x,wd)
  1286. END Int;
  1287. PROCEDURE Set*( s: SET ); (* from P. Saladin *)
  1288. BEGIN Indent; w.Set(s)
  1289. END Set;
  1290. PROCEDURE Hex*(x: HUGEINT; wd: LONGINT );
  1291. BEGIN Indent; w.Hex(x,wd)
  1292. END Hex;
  1293. PROCEDURE Address* (x: ADDRESS);
  1294. BEGIN Indent; w.Address(x)
  1295. END Address;
  1296. PROCEDURE Date*( t, d: LONGINT );
  1297. BEGIN Indent; w.Date(t,d)
  1298. END Date;
  1299. PROCEDURE Date822*( t, d, tz: LONGINT );
  1300. BEGIN Indent; w.Date822(t,d,tz)
  1301. END Date822;
  1302. PROCEDURE Float*( x: LONGREAL; n: LONGINT );
  1303. BEGIN Indent; w.Float(x,n)
  1304. END Float;
  1305. PROCEDURE FloatFix*( x: LONGREAL; n, f, D: LONGINT );
  1306. BEGIN Indent; w.FloatFix(x,n,f,D)
  1307. END FloatFix;
  1308. PROCEDURE SetIndent*(i: LONGINT);
  1309. BEGIN
  1310. indent := i
  1311. END SetIndent;
  1312. PROCEDURE IncIndent*;
  1313. BEGIN INC(indent)
  1314. END IncIndent;
  1315. PROCEDURE DecIndent*;
  1316. BEGIN DEC(indent)
  1317. END DecIndent;
  1318. PROCEDURE BeginAlert*;
  1319. END BeginAlert;
  1320. PROCEDURE EndAlert*;
  1321. END EndAlert;
  1322. PROCEDURE BeginKeyword*;
  1323. BEGIN
  1324. END BeginKeyword;
  1325. PROCEDURE EndKeyword*;
  1326. BEGIN
  1327. END EndKeyword;
  1328. PROCEDURE BeginComment*;
  1329. END BeginComment;
  1330. PROCEDURE EndComment*;
  1331. END EndComment;
  1332. PROCEDURE AlertString*(CONST s: ARRAY OF CHAR);
  1333. BEGIN
  1334. BeginAlert; w.String(s); EndAlert;
  1335. END AlertString;
  1336. END Writer;
  1337. CRC32Stream* = OBJECT(Streams.Writer) (* from CRC.Mod *)
  1338. VAR
  1339. crc : LONGINT;
  1340. PROCEDURE &InitStream*;
  1341. BEGIN
  1342. crc := LONGINT(0FFFFFFFFH);
  1343. InitWriter(Send, 256)
  1344. END InitStream;
  1345. PROCEDURE Send*(CONST buf: ARRAY OF CHAR; ofs, len: LONGINT; propagate: BOOLEAN; VAR res: LONGINT);
  1346. VAR idx: LONGINT;
  1347. BEGIN
  1348. WHILE len > 0 DO
  1349. idx := SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, crc) / SYSTEM.VAL(SET, LONG(ORD(buf[ofs])))) MOD 100H;
  1350. crc := SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, CRC32Table[idx])/SYSTEM.VAL(SET, LSH(crc, -8)));
  1351. DEC(len); INC(ofs)
  1352. END;
  1353. res := Streams.Ok
  1354. END Send;
  1355. PROCEDURE GetCRC*():LONGINT;
  1356. BEGIN
  1357. Update();
  1358. RETURN SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, crc)/{0..31})
  1359. END GetCRC;
  1360. END CRC32Stream;
  1361. TracingDiagnostics=OBJECT (Diagnostics.Diagnostics)
  1362. VAR diagnostics: Diagnostics.Diagnostics;
  1363. PROCEDURE &InitDiagnostics(diagnostics: Diagnostics.Diagnostics);
  1364. BEGIN
  1365. SELF.diagnostics := diagnostics
  1366. END InitDiagnostics;
  1367. PROCEDURE Error(CONST source: ARRAY OF CHAR; position, errorCode : LONGINT; CONST message : ARRAY OF CHAR);
  1368. BEGIN
  1369. IF diagnostics # NIL THEN
  1370. diagnostics.Error(source,position,errorCode,message);
  1371. END;
  1372. D.Ln;
  1373. D.String(" ---------------------- TRACE for COMPILER ERROR < ");
  1374. D.String(source);
  1375. IF position # Diagnostics.Invalid THEN D.String("@"); D.Int(position,1) END;
  1376. IF errorCode # Diagnostics.Invalid THEN D.String(" "); D.Int(errorCode,1); END;
  1377. D.String(" "); D.String(message);
  1378. D.String(" > ---------------------- ");
  1379. D.TraceBack
  1380. END Error;
  1381. PROCEDURE Warning*(CONST source : ARRAY OF CHAR; position, errorCode : LONGINT; CONST message : ARRAY OF CHAR);
  1382. BEGIN
  1383. IF diagnostics # NIL THEN
  1384. diagnostics.Warning(source,position,errorCode,message);
  1385. END;
  1386. END Warning;
  1387. PROCEDURE Information*(CONST source : ARRAY OF CHAR; position, errorCode : LONGINT; CONST message : ARRAY OF CHAR);
  1388. BEGIN
  1389. IF diagnostics # NIL THEN
  1390. diagnostics.Information(source,position,errorCode,message);
  1391. END;
  1392. END Information;
  1393. END TracingDiagnostics;
  1394. DebugWriterFactory*= PROCEDURE{DELEGATE} (CONST title: ARRAY OF CHAR): Streams.Writer;
  1395. WriterFactory*=PROCEDURE{DELEGATE} (w: Streams.Writer): Writer;
  1396. DiagnosticsFactory*=PROCEDURE{DELEGATE} (w: Streams.Writer): Diagnostics.Diagnostics;
  1397. VAR
  1398. lists-: LONGINT; enlarged-: LONGINT; strings-: LONGINT; integerObjects: HashTableInt;
  1399. errMsg: ErrorMsgs; (*error messages*)
  1400. emptyString-: String;
  1401. debug: BOOLEAN;
  1402. getDebugWriter: DebugWriterFactory;
  1403. getWriter: WriterFactory;
  1404. getDiagnostics: DiagnosticsFactory;
  1405. CRC32Table: ARRAY 256 OF SET;
  1406. invalidPosition-: Position;
  1407. (* Make a string out of a series of characters. *)
  1408. PROCEDURE MakeString*( CONST s: ARRAY OF CHAR ): String;
  1409. (* VAR str: String; *)
  1410. BEGIN
  1411. INC( strings );
  1412. (*
  1413. (* allocation based *)
  1414. NEW( str, Strings.Length( s ) +1); COPY( s, str^ ); RETURN str;
  1415. *)
  1416. RETURN StringPool.GetIndex1( s )
  1417. END MakeString;
  1418. PROCEDURE GetString*(s: String; VAR str: ARRAY OF CHAR);
  1419. BEGIN
  1420. StringPool.GetString(s,str);
  1421. END GetString;
  1422. PROCEDURE StringEqual*( s, t: String ): BOOLEAN;
  1423. BEGIN
  1424. RETURN s = t;
  1425. (*
  1426. (* allocation based *)
  1427. RETURN s^ = t^
  1428. *)
  1429. END StringEqual;
  1430. PROCEDURE GetErrorMessage*(err: LONGINT; CONST msg: ARRAY OF CHAR; VAR res: ARRAY OF CHAR);
  1431. VAR str: ARRAY 128 OF CHAR;
  1432. BEGIN
  1433. res := "";
  1434. IF (errMsg # NIL) & (0 <= err) & (err < LEN(errMsg)) THEN
  1435. StringPool.GetString(errMsg[err], str);
  1436. Strings.Append(res,str);
  1437. Strings.Append(res, " ");
  1438. END;
  1439. Strings.Append(res, msg);
  1440. END GetErrorMessage;
  1441. PROCEDURE AppendPosition*(VAR msg: ARRAY OF CHAR; pos: Position);
  1442. BEGIN
  1443. IF pos.line >= 0 THEN
  1444. Strings.Append(msg, " in line ");
  1445. Strings.AppendInt(msg, pos.line);
  1446. Strings.Append(msg, ", col ");
  1447. Strings.AppendInt(msg, pos.start- pos.linepos);
  1448. END;
  1449. END AppendPosition;
  1450. PROCEDURE MakeMessage(pos: Position; code: ErrorCode; CONST msg: ARRAY OF CHAR; VAR message: ARRAY OF CHAR);
  1451. BEGIN
  1452. GetErrorMessage(code, msg, message);
  1453. AppendPosition(message, pos);
  1454. END MakeMessage;
  1455. (* error message with code *)
  1456. PROCEDURE ErrorC*(diagnostics: Diagnostics.Diagnostics; CONST source: ARRAY OF CHAR; pos: Position; code: ErrorCode; CONST msg: ARRAY OF CHAR);
  1457. VAR message,err: ARRAY 512 OF CHAR; ch: CHAR; b: BOOLEAN; oldpos: LONGINT;
  1458. file: Files.File; reader: Files.Reader;
  1459. BEGIN
  1460. IF diagnostics # NIL THEN
  1461. MakeMessage(pos, code, msg,message);
  1462. IF (source # "") & (pos.linepos >= 0) THEN
  1463. err[0] := 0AX;
  1464. err[1] := 0X;
  1465. file := Files.Old(source);
  1466. IF file # NIL THEN
  1467. NEW(reader, file, pos.linepos);
  1468. reader.SetPos(pos.linepos);
  1469. reader.Char(ch);
  1470. WHILE (ch # 0X) & (ch # 0AX) DO
  1471. Strings.AppendChar(err, ch);
  1472. IF reader.Pos() = pos.start THEN
  1473. Strings.Append(err,"(*!*)");
  1474. END;
  1475. reader.Char(ch);
  1476. END;
  1477. Strings.Append(message, err);
  1478. (*diagnostics.Error(source, Diagnostics.Invalid, Diagnostics.Invalid, err);*)
  1479. END;
  1480. END;
  1481. diagnostics.Error(source, pos.start, code, message);
  1482. END;
  1483. END ErrorC;
  1484. (* error message without code *)
  1485. PROCEDURE Error*(diagnostics: Diagnostics.Diagnostics; CONST source: ARRAY OF CHAR; pos: Position; CONST msg: ARRAY OF CHAR);
  1486. BEGIN
  1487. ErrorC(diagnostics, source, pos, InvalidCode, msg);
  1488. END Error;
  1489. PROCEDURE Warning*(diagnostics: Diagnostics.Diagnostics; CONST source: ARRAY OF CHAR; pos: Position; CONST msg: ARRAY OF CHAR);
  1490. VAR message: ARRAY 256 OF CHAR;
  1491. BEGIN
  1492. IF diagnostics # NIL THEN
  1493. MakeMessage(pos, InvalidCode, msg,message);
  1494. diagnostics.Warning(source, pos.start, InvalidCode, message);
  1495. END;
  1496. END Warning;
  1497. PROCEDURE Information*(diagnostics: Diagnostics.Diagnostics; CONST source: ARRAY OF CHAR; pos: Position;CONST msg: ARRAY OF CHAR);
  1498. VAR message: ARRAY 256 OF CHAR;
  1499. BEGIN
  1500. IF diagnostics # NIL THEN
  1501. MakeMessage(pos, InvalidCode, msg,message);
  1502. diagnostics.Information(source, pos.start, InvalidCode, message);
  1503. END;
  1504. END Information;
  1505. (** SetErrorMsg - Set message for error n *)
  1506. PROCEDURE SetErrorMessage*(n: LONGINT; CONST msg: ARRAY OF CHAR);
  1507. BEGIN
  1508. IF errMsg = NIL THEN NEW(errMsg, InitErrMsgSize) END;
  1509. WHILE LEN(errMsg^) < n DO Expand(errMsg) END;
  1510. StringPool.GetIndex(msg, errMsg[n])
  1511. END SetErrorMessage;
  1512. PROCEDURE SetErrorExpected*(n: LONGINT; CONST msg: ARRAY OF CHAR);
  1513. VAR err: ARRAY 256 OF CHAR;
  1514. BEGIN
  1515. err := "missing '";
  1516. Strings.Append(err,msg);
  1517. Strings.Append(err, "'");
  1518. SetErrorMessage(n,err);
  1519. END SetErrorExpected;
  1520. PROCEDURE AppendNumber*(VAR s: ARRAY OF CHAR; num: LONGINT);
  1521. VAR nums: ARRAY 32 OF CHAR;
  1522. BEGIN
  1523. Strings.IntToStr(num,nums);
  1524. Strings.Append(s,nums);
  1525. END AppendNumber;
  1526. PROCEDURE InitSegmentedName*(VAR name: SegmentedName);
  1527. VAR i: LONGINT;
  1528. BEGIN FOR i := 0 TO LEN(name)-1 DO name[i] := -1 END;
  1529. END InitSegmentedName;
  1530. PROCEDURE ToSegmentedName*(CONST name: ARRAY OF CHAR; VAR pooledName: SegmentedName);
  1531. BEGIN
  1532. ObjectFile.StringToSegmentedName(name,pooledName);
  1533. END ToSegmentedName;
  1534. PROCEDURE SegmentedNameToString*(CONST pooledName: SegmentedName; VAR name: ARRAY OF CHAR);
  1535. BEGIN
  1536. ObjectFile.SegmentedNameToString(pooledName, name);
  1537. END SegmentedNameToString;
  1538. PROCEDURE WriteSegmentedName*(w: Streams.Writer; name: SegmentedName);
  1539. VAR sectionName: ObjectFile.SectionName;
  1540. BEGIN
  1541. SegmentedNameToString(name, sectionName);
  1542. w.String(sectionName);
  1543. END WriteSegmentedName;
  1544. PROCEDURE AppendToSegmentedName*(VAR name: SegmentedName; CONST this: ARRAY OF CHAR);
  1545. VAR i,j: LONGINT; string: ObjectFile.SectionName;
  1546. BEGIN
  1547. i := 0;
  1548. WHILE (i<LEN(name)) & (name[i] >= 0) DO
  1549. INC(i)
  1550. END;
  1551. IF (this[0] = ".") & (i < LEN(name)) THEN (* suffix *)
  1552. j := 0;
  1553. WHILE this[j+1] # 0X DO
  1554. string[j] := this[j+1];
  1555. INC(j);
  1556. END;
  1557. string[j] := 0X;
  1558. name[i] := StringPool.GetIndex1(string);
  1559. IF i<LEN(name)-1 THEN name[i+1] := -1 END;
  1560. ELSE
  1561. StringPool.GetString(name[i-1], string);
  1562. Strings.Append(string, this);
  1563. name[i-1] := StringPool.GetIndex1(string);
  1564. END;
  1565. END AppendToSegmentedName;
  1566. (* suffix using separation character "." *)
  1567. PROCEDURE SuffixSegmentedName*(VAR name: SegmentedName; this: StringPool.Index);
  1568. VAR string, suffix: ObjectFile.SectionName; i: LONGINT;
  1569. BEGIN
  1570. i := 0;
  1571. WHILE (i < LEN(name)) & (name[i] >= 0) DO
  1572. INC(i);
  1573. END;
  1574. IF i < LEN(name) THEN (* suffix *)
  1575. name[i] := this;
  1576. IF i<LEN(name)-1 THEN name[i+1] := -1 END;
  1577. ELSE
  1578. StringPool.GetString(name[i-1], string);
  1579. StringPool.GetString(this, suffix);
  1580. Strings.Append(string,".");
  1581. Strings.Append(string, suffix);
  1582. name[i-1] := StringPool.GetIndex1(string);
  1583. END;
  1584. END SuffixSegmentedName;
  1585. PROCEDURE SegmentedNameEndsWith*(CONST name: SegmentedName; CONST this: ARRAY OF CHAR): BOOLEAN;
  1586. VAR string: ObjectFile.SectionName; i: LONGINT;
  1587. BEGIN
  1588. i := 0;
  1589. WHILE (i< LEN(name)) & (name[i] >= 0) DO
  1590. INC(i);
  1591. END;
  1592. DEC(i);
  1593. IF i < 0 THEN
  1594. RETURN FALSE
  1595. ELSE
  1596. StringPool.GetString(name[i],string);
  1597. RETURN Strings.EndsWith(this, string);
  1598. END
  1599. END SegmentedNameEndsWith;
  1600. PROCEDURE RemoveSuffix*(VAR name: SegmentedName);
  1601. VAR i,pos,pos0: LONGINT;string: ObjectFile.SectionName;
  1602. BEGIN
  1603. i := 0;
  1604. WHILE (i< LEN(name)) & (name[i] >= 0) DO
  1605. INC(i);
  1606. END;
  1607. ASSERT(i>0);
  1608. IF i < LEN(name) THEN (* name[i] = empty *) name[i-1] := -1
  1609. ELSE (* i = LEN(name), name[i] = nonempty *)
  1610. DEC(i);
  1611. StringPool.GetString(name[i],string);
  1612. pos0 := 0; pos := 0;
  1613. WHILE (pos0 < LEN(string)) & (string[pos0] # 0X) DO
  1614. IF string[pos0] = "." THEN pos := pos0 END;
  1615. INC(pos0);
  1616. END;
  1617. IF pos = 0 THEN (* no dot in name or name starts with dot *)
  1618. name[i] := -1
  1619. ELSE (* remove last part in name *)
  1620. string[pos] := 0X;
  1621. name[i] := StringPool.GetIndex1(string);
  1622. END;
  1623. END;
  1624. END RemoveSuffix;
  1625. PROCEDURE GetSuffix*(CONST name: SegmentedName; VAR string: ARRAY OF CHAR);
  1626. VAR i,pos,pos0: LONGINT;
  1627. BEGIN
  1628. i := 0;
  1629. WHILE (i< LEN(name)) & (name[i] >= 0) DO
  1630. INC(i);
  1631. END;
  1632. ASSERT(i>0);
  1633. StringPool.GetString(name[i-1],string);
  1634. IF i = LEN(name) THEN
  1635. pos0 := 0; pos := 0;
  1636. WHILE (pos0 < LEN(string)) & (string[pos0] # 0X) DO
  1637. IF string[pos0] = "." THEN pos := pos0 END;
  1638. INC(pos0);
  1639. END;
  1640. IF pos # 0 THEN (* no dot in name or name starts with dot *)
  1641. pos0 := 0;
  1642. REPEAT
  1643. INC(pos); (* start with character after "." *)
  1644. string[pos0] := string[pos];
  1645. INC(pos0);
  1646. UNTIL string[pos] = 0X;
  1647. END;
  1648. END;
  1649. END GetSuffix;
  1650. PROCEDURE IsPrefix*(CONST prefix, of: SegmentedName): BOOLEAN;
  1651. VAR prefixS, ofS: SectionName; i: LONGINT;
  1652. BEGIN
  1653. i := 0;
  1654. WHILE (i< LEN(prefix)) & (prefix[i] = of[i]) DO INC(i) END;
  1655. IF i = LEN(prefix) THEN RETURN TRUE (* identical *)
  1656. ELSE (* prefix[i] # of[i] *)
  1657. IF prefix[i] < 0 THEN RETURN TRUE
  1658. ELSIF of[i] < 0 THEN RETURN FALSE
  1659. ELSIF (i<LEN(prefix)-1) THEN RETURN FALSE
  1660. ELSE
  1661. StringPool.GetString(prefix[i], prefixS);
  1662. StringPool.GetString(of[i], ofS);
  1663. RETURN Strings.StartsWith(prefixS, 0, ofS)
  1664. END
  1665. END;
  1666. END IsPrefix;
  1667. PROCEDURE Expand(VAR oldAry: ErrorMsgs);
  1668. VAR
  1669. len, i: LONGINT;
  1670. newAry: ErrorMsgs;
  1671. BEGIN
  1672. IF oldAry = NIL THEN RETURN END;
  1673. len := LEN(oldAry^);
  1674. NEW(newAry, len * 2);
  1675. FOR i := 0 TO len-1 DO
  1676. newAry[i] := oldAry[i];
  1677. END;
  1678. oldAry := newAry;
  1679. END Expand;
  1680. PROCEDURE Concat*(VAR result: ARRAY OF CHAR; CONST prefix, name, suffix: ARRAY OF CHAR);
  1681. VAR i, j: LONGINT;
  1682. BEGIN
  1683. i := 0; WHILE prefix[i] # 0X DO result[i] := prefix[i]; INC(i) END;
  1684. j := 0; WHILE name[j] # 0X DO result[i+j] := name[j]; INC(j) END;
  1685. INC(i, j);
  1686. j := 0; WHILE suffix[j] # 0X DO result[i+j] := suffix[j]; INC(j) END;
  1687. result[i+j] := 0X;
  1688. END Concat;
  1689. PROCEDURE Lowercase*(CONST name: ARRAY OF CHAR; VAR result: ARRAY OF CHAR);
  1690. VAR ch: CHAR; i: LONGINT;
  1691. BEGIN
  1692. i := 0;
  1693. REPEAT
  1694. ch := name[i];
  1695. IF (ch >= 'A') & (ch <= 'Z') THEN
  1696. ch := CHR(ORD(ch)-ORD('A')+ORD('a'));
  1697. END;
  1698. result[i] := ch; INC(i);
  1699. UNTIL ch = 0X;
  1700. END Lowercase;
  1701. PROCEDURE Uppercase*(CONST name: ARRAY OF CHAR; VAR result: ARRAY OF CHAR);
  1702. VAR ch: CHAR; i: LONGINT;
  1703. BEGIN
  1704. i := 0;
  1705. REPEAT
  1706. ch := name[i];
  1707. IF (ch >= 'a') & (ch <= 'z') THEN
  1708. ch := CHR(ORD(ch)-ORD('a')+ORD('A'));
  1709. END;
  1710. result[i] := ch; INC(i);
  1711. UNTIL ch = 0X;
  1712. END Uppercase;
  1713. PROCEDURE GetIntegerObj*(value: LONGINT):ANY;
  1714. VAR obj: IntegerObject;
  1715. BEGIN
  1716. IF integerObjects.Has(value) THEN
  1717. RETURN integerObjects.Get(value);
  1718. END;
  1719. NEW(obj);
  1720. integerObjects.Put(value, obj);
  1721. RETURN obj;
  1722. END GetIntegerObj;
  1723. PROCEDURE Align*(VAR offset: LONGINT; alignment: LONGINT);
  1724. BEGIN
  1725. IF alignment >0 THEN
  1726. INC(offset,(-offset) MOD alignment);
  1727. ELSIF alignment < 0 THEN
  1728. DEC(offset,offset MOD (-alignment));
  1729. END;
  1730. END Align;
  1731. PROCEDURE InitErrorMessages;
  1732. BEGIN
  1733. SetErrorMessage(UndeclaredIdentifier, "undeclared identifier");
  1734. SetErrorMessage(MultiplyDefinedIdentifier, "multiply defined identifier");
  1735. SetErrorMessage(NumberIllegalCharacter, "illegal character in number");
  1736. SetErrorMessage(StringIllegalCharacter, "illegal character in string");
  1737. SetErrorMessage(NoMatchProcedureName, "procedure name does not match");
  1738. SetErrorMessage(CommentNotClosed, "comment not closed");
  1739. SetErrorMessage(IllegalCharacterValue, "illegal character value");
  1740. SetErrorMessage(ValueStartIncorrectSymbol, "value starts with incorrect symbol");
  1741. SetErrorMessage(IllegalyMarkedIdentifier, "illegaly marked identifier");
  1742. SetErrorMessage(IdentifierNoType, "identifier is not a type");
  1743. SetErrorMessage(IdentifierNoRecordType, "identifier is not a record type");
  1744. SetErrorMessage(IdentifierNoObjectType, "identifier is not an object type");
  1745. SetErrorMessage(ImportNotAvailable, "import is not available");
  1746. SetErrorMessage(RecursiveTypeDeclaration, "recursive type declaration");
  1747. SetErrorMessage(NumberTooLarge, "number too large");
  1748. SetErrorMessage(IdentifierTooLong, "identifier too long");
  1749. SetErrorMessage(StringTooLong, "string too long");
  1750. END InitErrorMessages;
  1751. PROCEDURE ActivateDebug*;
  1752. BEGIN
  1753. debug := TRUE;
  1754. END ActivateDebug;
  1755. PROCEDURE Test*;
  1756. VAR table: HashTableInt; dump: LONGINT;
  1757. BEGIN
  1758. NEW(table, 32);
  1759. table.PutInt(32, -4);
  1760. dump := table.GetInt(32);
  1761. HALT(100);
  1762. END Test;
  1763. PROCEDURE GetFileReader*(CONST filename: ARRAY OF CHAR): Streams.Reader;
  1764. VAR
  1765. file: Files.File; fileReader: Files.Reader; offset: LONGINT;
  1766. BEGIN
  1767. (* Optimisation: skip header of oberon files and return a file reader instead of default text reader*)
  1768. file := Files.Old (filename);
  1769. IF file = NIL THEN RETURN NIL END;
  1770. NEW (fileReader, file, 0);
  1771. IF (fileReader.Get () = 0F0X) & (fileReader.Get () = 001X) THEN
  1772. offset := ORD (fileReader.Get ());
  1773. INC (offset, LONG (ORD (fileReader.Get ())) * 0100H);
  1774. fileReader.SetPos(offset);
  1775. ELSE fileReader.SetPos(0)
  1776. END;
  1777. RETURN fileReader
  1778. END GetFileReader;
  1779. PROCEDURE GetWriter*(w: Streams.Writer): Writer;
  1780. VAR writer: Writer;
  1781. BEGIN
  1782. ASSERT(w # NIL);
  1783. IF w IS Writer THEN RETURN w(Writer)
  1784. ELSIF getWriter = NIL THEN
  1785. NEW(writer,w); RETURN writer
  1786. ELSE RETURN getWriter(w)
  1787. END;
  1788. END GetWriter;
  1789. PROCEDURE GetDebugWriter*(CONST title: ARRAY OF CHAR): Streams.Writer;
  1790. VAR w: Streams.Writer;
  1791. BEGIN
  1792. IF getDebugWriter # NIL THEN w:= getDebugWriter(title)
  1793. ELSE NEW(w, KernelLog.Send,1024)
  1794. END;
  1795. RETURN w;
  1796. END GetDebugWriter;
  1797. PROCEDURE GetDiagnostics*(w: Streams.Writer): Diagnostics.Diagnostics;
  1798. VAR diagnostics: Diagnostics.StreamDiagnostics;
  1799. BEGIN
  1800. IF getDiagnostics # NIL THEN RETURN getDiagnostics(w)
  1801. ELSE NEW(diagnostics,w); RETURN diagnostics
  1802. END;
  1803. END GetDiagnostics;
  1804. PROCEDURE GetDefaultDiagnostics*(): Diagnostics.Diagnostics;
  1805. VAR w: Streams.Writer;
  1806. BEGIN
  1807. NEW(w, KernelLog.Send,128);
  1808. RETURN GetDiagnostics(w);
  1809. END GetDefaultDiagnostics;
  1810. PROCEDURE InitWindowWriter;
  1811. VAR install: PROCEDURE;
  1812. BEGIN
  1813. getDebugWriter := NIL; getWriter := NIL;
  1814. IF Modules.ModuleByName("WindowManager") # NIL THEN
  1815. GETPROCEDURE("FoxA2Interface","Install",install);
  1816. END;
  1817. IF install # NIL THEN install END;
  1818. END InitWindowWriter;
  1819. PROCEDURE InstallWriterFactory*(writer: WriterFactory; debug: DebugWriterFactory; diagnostics: DiagnosticsFactory);
  1820. BEGIN
  1821. getWriter := writer;
  1822. getDebugWriter := debug;
  1823. getDiagnostics := diagnostics;
  1824. END InstallWriterFactory;
  1825. PROCEDURE Replace(VAR in: ARRAY OF CHAR; CONST this, by: ARRAY OF CHAR);
  1826. VAR pos: LONGINT;
  1827. BEGIN
  1828. pos := Strings.Pos(this,in);
  1829. IF pos >= 0 THEN
  1830. Strings.Delete(in,pos,Strings.Length(this));
  1831. Strings.Insert(by, in, pos);
  1832. END;
  1833. END Replace;
  1834. PROCEDURE MessageS*(CONST format, s0: ARRAY OF CHAR): MessageString;
  1835. VAR message: MessageString;
  1836. BEGIN
  1837. COPY(format, message);
  1838. Replace(message,"%0",s0);
  1839. RETURN message
  1840. END MessageS;
  1841. PROCEDURE MessageSS*(CONST format, s0, s1: ARRAY OF CHAR): MessageString;
  1842. VAR message: MessageString;
  1843. BEGIN
  1844. COPY(format, message);
  1845. Replace(message,"%0",s0);
  1846. Replace(message,"%1",s1);
  1847. RETURN message
  1848. END MessageSS;
  1849. PROCEDURE MessageI*(CONST format: ARRAY OF CHAR; i0: LONGINT): MessageString;
  1850. VAR message: MessageString; number: ARRAY 32 OF CHAR;
  1851. BEGIN
  1852. COPY(format, message);
  1853. Strings.IntToStr(i0,number);
  1854. Replace(message,"%0",number);
  1855. END MessageI;
  1856. PROCEDURE MessageSI*(CONST format: ARRAY OF CHAR; CONST s0: ARRAY OF CHAR; i1: LONGINT): MessageString;
  1857. VAR message: MessageString; number: ARRAY 32 OF CHAR;
  1858. BEGIN
  1859. COPY(format, message);
  1860. Replace(message,"%0",s0);
  1861. Strings.IntToStr(i1,number);
  1862. Replace(message,"%1",number);
  1863. END MessageSI;
  1864. (*
  1865. Get next available name from stream ignoring comments and end of comment brackets
  1866. Returns TRUE on success, returns FALSE on end of stream, on error or if "~" or ";" encountered.
  1867. Scanner based on Peek() feature of stream. Necessary to make it restartable.
  1868. *)
  1869. PROCEDURE GetStringParameter*(r: Streams.Reader; VAR string: ARRAY OF CHAR): BOOLEAN;
  1870. VAR ch: CHAR; i: LONGINT; done,error: BOOLEAN;
  1871. PROCEDURE Next;
  1872. BEGIN r.Char(ch); ch := r.Peek();
  1873. END Next;
  1874. PROCEDURE Append(ch: CHAR);
  1875. BEGIN string[i] := ch; INC(i)
  1876. END Append;
  1877. PROCEDURE SkipWhitespace;
  1878. BEGIN WHILE (ch <= " ") & (ch # 0X) DO Next END;
  1879. END SkipWhitespace;
  1880. PROCEDURE Comment;
  1881. VAR done: BOOLEAN;
  1882. BEGIN
  1883. done := FALSE;
  1884. Next;
  1885. REPEAT
  1886. CASE ch OF
  1887. |"(": Next; IF ch = "*" THEN Comment; SkipWhitespace END;
  1888. |"*": Next; IF ch =")" THEN Next; done:= TRUE END;
  1889. | 0X: done := TRUE;
  1890. ELSE Next;
  1891. END;
  1892. UNTIL done;
  1893. END Comment;
  1894. PROCEDURE String(delimiter: CHAR);
  1895. VAR done: BOOLEAN;
  1896. BEGIN
  1897. done := FALSE; Next;
  1898. REPEAT
  1899. IF ch = delimiter THEN done := TRUE; Next;
  1900. ELSIF ch = 0X THEN done := TRUE; error := TRUE;
  1901. ELSE Append(ch); Next;
  1902. END;
  1903. UNTIL done OR (i=LEN(string)-1);
  1904. END String;
  1905. BEGIN
  1906. i := 0; done := FALSE;
  1907. ch := r.Peek(); (* restart scanning *)
  1908. SkipWhitespace;
  1909. REPEAT
  1910. CASE ch OF
  1911. "(": Next; IF ch = "*" THEN Comment ; SkipWhitespace ELSE Append(ch) END;
  1912. | "*": Next; IF ch = ")" THEN Next; SkipWhitespace ELSE Append(ch) END;
  1913. | '"', "'": done := TRUE; IF i = 0 THEN String(ch) END;
  1914. | 0X .. ' ', '~', ';': done := TRUE;
  1915. ELSE
  1916. Append(ch);
  1917. Next;
  1918. END;
  1919. UNTIL done OR (i = LEN(string)-1);
  1920. string[i] := 0X;
  1921. RETURN (i > 0) & done & ~error;
  1922. END GetStringParameter;
  1923. PROCEDURE GetTracingDiagnostics*(diagnostics: Diagnostics.Diagnostics): Diagnostics.Diagnostics;
  1924. VAR tracing: TracingDiagnostics;
  1925. BEGIN
  1926. NEW(tracing, diagnostics); RETURN tracing
  1927. END GetTracingDiagnostics;
  1928. PROCEDURE InitTable32;
  1929. CONST poly = LONGINT(0EDB88320H);
  1930. VAR n, c, k: LONGINT;
  1931. BEGIN
  1932. FOR n := 0 TO 255 DO
  1933. c := n;
  1934. FOR k := 0 TO 7 DO
  1935. IF ODD(c) THEN c := SYSTEM.VAL(LONGINT, SYSTEM.VAL(SET, poly) / SYSTEM.VAL(SET, LSH(c, -1)))
  1936. ELSE c := LSH(c, -1)
  1937. END
  1938. END;
  1939. CRC32Table[n] := SYSTEM.VAL(SET, c)
  1940. END
  1941. END InitTable32;
  1942. BEGIN
  1943. InitErrorMessages;
  1944. InitWindowWriter;
  1945. InitTable32;
  1946. lists := 0; enlarged := 0; strings := 0;
  1947. emptyString := MakeString("");
  1948. debug := FALSE;
  1949. invalidPosition.start := -1;
  1950. invalidPosition.end := -1;
  1951. invalidPosition.line := -1;
  1952. invalidPosition.linepos := -1;
  1953. NEW(integerObjects, 128);
  1954. END FoxBasic.
  1955. FoxBasic.ActivateDebug ~
  1956. FoxBasic.Test ~