FoxBasic.Mod 51 KB

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