FoxBasic.Mod 53 KB

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