2
0

CPP.txt 58 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650
  1. MODULE DevCPP;
  2. (* THIS IS TEXT COPY OF BlackBox 1.6-rc6 Dev/Mod/CPP.odc *)
  3. (* DO NOT EDIT *)
  4. IMPORT
  5. DevCPM, DevCPT, DevCPB, DevCPS;
  6. CONST
  7. anchorVarPar = TRUE;
  8. (* numtyp values *)
  9. char = 1; integer = 2; real = 4; int64 = 5; real32 = 6; real64 = 7;
  10. (*symbol values*)
  11. null = 0; times = 1; slash = 2; div = 3; mod = 4;
  12. and = 5; plus = 6; minus = 7; or = 8; eql = 9;
  13. neq = 10; lss = 11; leq = 12; gtr = 13; geq = 14;
  14. in = 15; is = 16; arrow = 17; dollar = 18; period = 19;
  15. comma = 20; colon = 21; upto = 22; rparen = 23; rbrak = 24;
  16. rbrace = 25; of = 26; then = 27; do = 28; to = 29;
  17. by = 30; not = 33;
  18. lparen = 40; lbrak = 41; lbrace = 42; becomes = 44;
  19. number = 45; nil = 46; string = 47; ident = 48; semicolon = 49;
  20. bar = 50; end = 51; else = 52; elsif = 53; until = 54;
  21. if = 55; case = 56; while = 57; repeat = 58; for = 59;
  22. loop = 60; with = 61; exit = 62; return = 63; array = 64;
  23. record = 65; pointer = 66; begin = 67; const = 68; type = 69;
  24. var = 70; out = 71; procedure = 72; close = 73; import = 74;
  25. module = 75; eof = 76;
  26. (* object modes *)
  27. Var = 1; VarPar = 2; Con = 3; Fld = 4; Typ = 5; LProc = 6; XProc = 7;
  28. SProc = 8; CProc = 9; IProc = 10; Mod = 11; Head = 12; TProc = 13; Attr = 20;
  29. (* Structure forms *)
  30. Undef = 0; Byte = 1; Bool = 2; Char8 = 3; Int8 = 4; Int16 = 5; Int32 = 6;
  31. Real32 = 7; Real64 = 8; Set = 9; String8 = 10; NilTyp = 11; NoTyp = 12;
  32. Pointer = 13; ProcTyp = 14; Comp = 15;
  33. Char16 = 16; String16 = 17; Int64 = 18;
  34. intSet = {Int8..Int32, Int64}; charSet = {Char8, Char16};
  35. (* composite structure forms *)
  36. Basic = 1; Array = 2; DynArr = 3; Record = 4;
  37. (*function number*)
  38. haltfn = 0; newfn = 1; incfn = 13; sysnewfn = 30;
  39. (* nodes classes *)
  40. Nvar = 0; Nvarpar = 1; Nfield = 2; Nderef = 3; Nindex = 4; Nguard = 5; Neguard = 6;
  41. Nconst = 7; Ntype = 8; Nproc = 9; Nupto = 10; Nmop = 11; Ndop = 12; Ncall = 13;
  42. Ninittd = 14; Nif = 15; Ncaselse = 16; Ncasedo = 17; Nenter = 18; Nassign = 19;
  43. Nifelse = 20; Ncase = 21; Nwhile = 22; Nrepeat = 23; Nloop = 24; Nexit = 25;
  44. Nreturn = 26; Nwith = 27; Ntrap = 28; Ncomp = 30;
  45. (* node subclasses *)
  46. super = 1;
  47. (* module visibility of objects *)
  48. internal = 0; external = 1; externalR = 2; inPar = 3; outPar = 4;
  49. (* procedure flags (conval.setval) *)
  50. hasBody = 1; isRedef = 2; slNeeded = 3; imVar = 4;
  51. (* attribute flags (attr.adr, struct.attribute, proc.conval.setval)*)
  52. newAttr = 16; absAttr = 17; limAttr = 18; empAttr = 19; extAttr = 20;
  53. (* case statement flags (conval.setval) *)
  54. useTable = 1; useTree = 2;
  55. (* sysflags *)
  56. nilBit = 1; inBit = 2; outBit = 4; newBit = 8; iidBit = 16; interface = 10; som = 20; jstr = -13;
  57. TYPE
  58. Elem = POINTER TO RECORD
  59. next: Elem;
  60. struct: DevCPT.Struct;
  61. obj, base: DevCPT.Object;
  62. pos: INTEGER;
  63. name: DevCPT.String
  64. END;
  65. VAR
  66. sym, level: BYTE;
  67. LoopLevel: SHORTINT;
  68. TDinit, lastTDinit: DevCPT.Node;
  69. userList: Elem;
  70. recList: Elem;
  71. hasReturn: BOOLEAN;
  72. numUsafeVarPar, numFuncVarPar: INTEGER;
  73. PROCEDURE^ Type(VAR typ: DevCPT.Struct; VAR name: DevCPT.String);
  74. PROCEDURE^ Expression(VAR x: DevCPT.Node);
  75. PROCEDURE^ Block(VAR procdec, statseq: DevCPT.Node);
  76. (* forward type handling *)
  77. PROCEDURE IncompleteType (typ: DevCPT.Struct): BOOLEAN;
  78. BEGIN
  79. IF typ.form = Pointer THEN typ := typ.BaseTyp END;
  80. RETURN (typ = DevCPT.undftyp) OR (typ.comp = Record) & (typ.BaseTyp = DevCPT.undftyp)
  81. END IncompleteType;
  82. PROCEDURE SetType (struct: DevCPT.Struct; obj: DevCPT.Object; typ: DevCPT.Struct; name: DevCPT.String);
  83. VAR u: Elem;
  84. BEGIN
  85. IF obj # NIL THEN obj.typ := typ ELSE struct.BaseTyp := typ END;
  86. IF name # NIL THEN
  87. NEW(u); u.struct := struct; u.obj := obj; u.pos := DevCPM.errpos; u.name := name;
  88. u.next := userList; userList := u
  89. END
  90. END SetType;
  91. PROCEDURE CheckAlloc (VAR typ: DevCPT.Struct; dynAllowed: BOOLEAN; pos: INTEGER);
  92. BEGIN
  93. typ.pvused := TRUE;
  94. IF typ.comp = DynArr THEN
  95. IF ~dynAllowed THEN DevCPM.Mark(88, pos); typ := DevCPT.undftyp END
  96. ELSIF typ.comp = Record THEN
  97. IF (typ.attribute = absAttr) OR (typ.attribute = limAttr) & (typ.mno # 0) THEN
  98. DevCPM.Mark(193, pos); typ := DevCPT.undftyp
  99. END
  100. END
  101. END CheckAlloc;
  102. PROCEDURE CheckRecursiveType (outer, inner: DevCPT.Struct; pos: INTEGER);
  103. VAR fld: DevCPT.Object;
  104. BEGIN
  105. IF outer = inner THEN DevCPM.Mark(58, pos)
  106. ELSIF inner.comp IN {Array, DynArr} THEN CheckRecursiveType(outer, inner.BaseTyp, pos)
  107. ELSIF inner.comp = Record THEN
  108. fld := inner.link;
  109. WHILE (fld # NIL) & (fld.mode = Fld) DO
  110. CheckRecursiveType(outer, fld.typ, pos);
  111. fld := fld.link
  112. END;
  113. IF inner.BaseTyp # NIL THEN CheckRecursiveType(outer, inner.BaseTyp, pos) END
  114. END
  115. END CheckRecursiveType;
  116. PROCEDURE FixType (struct: DevCPT.Struct; obj: DevCPT.Object; typ: DevCPT.Struct; pos: INTEGER);
  117. (* fix forward reference *)
  118. VAR t: DevCPT.Struct; f, bf: DevCPT.Object; i: SHORTINT;
  119. BEGIN
  120. IF obj # NIL THEN
  121. IF obj.mode = Var THEN (* variable type *)
  122. IF struct # NIL THEN (* receiver type *)
  123. IF (typ.form # Pointer) OR (typ.BaseTyp # struct) THEN DevCPM.Mark(180, pos) END;
  124. ELSE CheckAlloc(typ, obj.mnolev > level, pos) (* TRUE for parameters *)
  125. END
  126. ELSIF obj.mode = VarPar THEN (* varpar type *)
  127. IF struct # NIL THEN (* varpar receiver type *)
  128. IF typ # struct THEN DevCPM.Mark(180, pos) END
  129. END
  130. ELSIF obj.mode = Fld THEN (* field type *)
  131. CheckAlloc(typ, FALSE, pos);
  132. CheckRecursiveType(struct, typ, pos)
  133. ELSIF obj.mode = TProc THEN (* proc return type *)
  134. IF typ.form = Comp THEN typ := DevCPT.undftyp; DevCPM.Mark(54, pos) END
  135. ELSIF obj.mode = Typ THEN (* alias type *)
  136. IF typ.form IN {Byte..Set, Char16, Int64} THEN (* make alias structure *)
  137. t := DevCPT.NewStr(typ.form, Basic); i := t.ref;
  138. t^ := typ^; t.ref := i; t.strobj := obj; t.mno := 0;
  139. t.BaseTyp := typ; typ := t
  140. END;
  141. IF obj.vis # internal THEN
  142. IF typ.comp = Record THEN typ.exp := TRUE
  143. ELSIF typ.form = Pointer THEN typ.BaseTyp.exp := TRUE
  144. END
  145. END
  146. ELSE HALT(100)
  147. END;
  148. obj.typ := typ
  149. ELSE
  150. IF struct.form = Pointer THEN (* pointer base type *)
  151. IF typ.comp = Record THEN DevCPM.PropagateRecPtrSysFlag(typ.sysflag, struct.sysflag)
  152. ELSIF typ.comp IN {Array, DynArr} THEN DevCPM.PropagateArrPtrSysFlag(typ.sysflag, struct.sysflag)
  153. ELSE typ := DevCPT.undftyp; DevCPM.Mark(57, pos)
  154. END;
  155. struct.untagged := struct.sysflag > 0;
  156. IF (struct.strobj # NIL) & (struct.strobj.vis # internal) THEN typ.exp := TRUE END;
  157. ELSIF struct.comp = Array THEN (* array base type *)
  158. CheckAlloc(typ, FALSE, pos);
  159. CheckRecursiveType(struct, typ, pos)
  160. ELSIF struct.comp = DynArr THEN (* array base type *)
  161. CheckAlloc(typ, TRUE, pos);
  162. CheckRecursiveType(struct, typ, pos)
  163. ELSIF struct.comp = Record THEN (* record base type *)
  164. IF typ.form = Pointer THEN typ := typ.BaseTyp END;
  165. typ.pvused := TRUE; struct.extlev := SHORT(SHORT(typ.extlev + 1));
  166. DevCPM.PropagateRecordSysFlag(typ.sysflag, struct.sysflag);
  167. IF (typ.attribute = 0) OR (typ.attribute = limAttr) & (typ.mno # 0) THEN DevCPM.Mark(181, pos)
  168. ELSIF (struct.attribute = absAttr) & (typ.attribute # absAttr) THEN DevCPM.Mark(191, pos)
  169. ELSIF (typ.attribute = limAttr) & (struct.attribute # limAttr) THEN DevCPM.Mark(197, pos)
  170. END;
  171. f := struct.link;
  172. WHILE f # NIL DO (* check for field name conflicts *)
  173. DevCPT.FindField(f.name, typ, bf);
  174. IF bf # NIL THEN DevCPM.Mark(1, pos) END;
  175. f := f.link
  176. END;
  177. CheckRecursiveType(struct, typ, pos);
  178. struct.untagged := struct.sysflag > 0;
  179. ELSIF struct.form = ProcTyp THEN (* proc type return type *)
  180. IF typ.form = Comp THEN typ := DevCPT.undftyp; DevCPM.Mark(54, pos) END;
  181. ELSE HALT(100)
  182. END;
  183. struct.BaseTyp := typ
  184. END
  185. END FixType;
  186. PROCEDURE CheckForwardTypes;
  187. VAR u, next: Elem; progress: BOOLEAN;
  188. BEGIN
  189. u := userList; userList := NIL;
  190. WHILE u # NIL DO
  191. next := u.next; DevCPS.name := u.name^$; DevCPT.Find(DevCPS.name, u.base);
  192. IF u.base = NIL THEN DevCPM.Mark(0, u.pos)
  193. ELSIF u.base.mode # Typ THEN DevCPM.Mark(72, u.pos)
  194. ELSE u.next := userList; userList := u (* reinsert *)
  195. END;
  196. u := next
  197. END;
  198. REPEAT (* iteration for multy level alias *)
  199. u := userList; userList := NIL; progress := FALSE;
  200. WHILE u # NIL DO
  201. next := u.next;
  202. IF IncompleteType(u.base.typ) THEN
  203. u.next := userList; userList := u (* reinsert *)
  204. ELSE
  205. progress := TRUE;
  206. FixType(u.struct, u.obj, u.base.typ, u.pos)
  207. END;
  208. u := next
  209. END
  210. UNTIL (userList = NIL) OR ~progress;
  211. u := userList; (* remaining type relations are cyclic *)
  212. WHILE u # NIL DO
  213. IF (u.obj = NIL) OR (u.obj.mode = Typ) THEN DevCPM.Mark(58, u.pos) END;
  214. u := u.next
  215. END;
  216. END CheckForwardTypes;
  217. PROCEDURE CheckUnimpl (m: DevCPT.Object; typ: DevCPT.Struct; pos: INTEGER);
  218. VAR obj: DevCPT.Object;
  219. BEGIN
  220. IF m # NIL THEN
  221. IF (m.mode = TProc) & (absAttr IN m.conval.setval) THEN
  222. DevCPT.FindField(m.name^, typ, obj);
  223. IF (obj = NIL) OR (obj.mode # TProc) OR (absAttr IN obj.conval.setval) THEN
  224. DevCPM.Mark(192, pos);
  225. DevCPM.LogWLn; DevCPM.LogWStr(" "); DevCPM.LogWStr(m.name^);
  226. DevCPM.LogWStr(" not implemented");
  227. IF typ.strobj # NIL THEN
  228. DevCPM.LogWStr(" in "); DevCPM.LogWStr(typ.strobj.name^)
  229. END
  230. END
  231. END;
  232. CheckUnimpl(m.left, typ, pos);
  233. CheckUnimpl(m.right, typ, pos)
  234. END
  235. END CheckUnimpl;
  236. PROCEDURE CheckRecords (rec: Elem);
  237. VAR b: DevCPT.Struct;
  238. BEGIN
  239. WHILE rec # NIL DO (* check for unimplemented methods in base type *)
  240. b := rec.struct.BaseTyp;
  241. WHILE (b # NIL) & (b # DevCPT.undftyp) DO
  242. CheckUnimpl(b.link, rec.struct, rec.pos);
  243. b := b.BaseTyp
  244. END;
  245. rec := rec.next
  246. END
  247. END CheckRecords;
  248. PROCEDURE err(n: SHORTINT);
  249. BEGIN DevCPM.err(n)
  250. END err;
  251. PROCEDURE CheckSym(s: SHORTINT);
  252. BEGIN
  253. IF sym = s THEN DevCPS.Get(sym) ELSE DevCPM.err(s) END
  254. END CheckSym;
  255. PROCEDURE qualident(VAR id: DevCPT.Object);
  256. VAR obj: DevCPT.Object; lev: BYTE;
  257. BEGIN (*sym = ident*)
  258. DevCPT.Find(DevCPS.name, obj); DevCPS.Get(sym);
  259. IF (sym = period) & (obj # NIL) & (obj.mode = Mod) THEN
  260. DevCPS.Get(sym);
  261. IF sym = ident THEN
  262. DevCPT.FindImport(DevCPS.name, obj, obj); DevCPS.Get(sym)
  263. ELSE err(ident); obj := NIL
  264. END
  265. END ;
  266. IF obj = NIL THEN err(0);
  267. obj := DevCPT.NewObj(); obj.mode := Var; obj.typ := DevCPT.undftyp; obj.adr := 0
  268. ELSE lev := obj.mnolev;
  269. IF (obj.mode IN {Var, VarPar}) & (lev # level) THEN
  270. obj.leaf := FALSE;
  271. IF lev > 0 THEN DevCPB.StaticLink(SHORT(SHORT(level-lev)), TRUE) END (* !!! *)
  272. END
  273. END ;
  274. id := obj
  275. END qualident;
  276. PROCEDURE ConstExpression(VAR x: DevCPT.Node);
  277. BEGIN Expression(x);
  278. IF x.class # Nconst THEN
  279. err(50); x := DevCPB.NewIntConst(1)
  280. END
  281. END ConstExpression;
  282. PROCEDURE CheckMark(obj: DevCPT.Object); (* !!! *)
  283. VAR n: INTEGER; mod: ARRAY 256 OF DevCPT.String;
  284. BEGIN DevCPS.Get(sym);
  285. IF (sym = times) OR (sym = minus) THEN
  286. IF (level > 0) OR ~(obj.mode IN {Var, Fld, TProc}) & (sym = minus) THEN err(41) END ;
  287. IF sym = times THEN obj.vis := external ELSE obj.vis := externalR END ;
  288. DevCPS.Get(sym)
  289. ELSE obj.vis := internal
  290. END;
  291. IF (obj.mode IN {TProc, LProc, XProc, CProc, Var, Typ, Con, Fld}) & (sym = lbrak) THEN
  292. DevCPS.Get(sym);
  293. IF (sym = number) & (DevCPS.numtyp = char) THEN
  294. NEW(DevCPS.str, 2); DevCPS.str[0] := SHORT(CHR(DevCPS.intval)); DevCPS.str[1] := 0X; sym := string
  295. END;
  296. IF sym = string THEN
  297. IF DevCPS.str^ # "" THEN obj.entry := DevCPS.str END;
  298. DevCPS.Get(sym); n := 0;
  299. IF (sym = comma) & (obj.mode IN {LProc, XProc, CProc, Var, Con}) THEN
  300. DevCPS.Get(sym);
  301. IF (sym = number) & (DevCPS.numtyp = char) THEN
  302. NEW(DevCPS.str, 2); DevCPS.str[0] := SHORT(CHR(DevCPS.intval)); DevCPS.str[1] := 0X; sym := string
  303. END;
  304. IF sym = string THEN
  305. obj.library := obj.entry; obj.entry := NIL;
  306. IF DevCPS.str^ # "" THEN obj.entry := DevCPS.str END;
  307. DevCPS.Get(sym);
  308. ELSE err(string)
  309. END
  310. END;
  311. WHILE sym = comma DO
  312. DevCPS.Get(sym);
  313. IF (sym = number) & (DevCPS.numtyp = char) THEN
  314. NEW(DevCPS.str, 2); DevCPS.str[0] := SHORT(CHR(DevCPS.intval)); DevCPS.str[1] := 0X; sym := string
  315. END;
  316. IF sym = string THEN
  317. IF n < LEN(mod) THEN mod[n] := DevCPS.str; INC(n)
  318. ELSE err(235)
  319. END;
  320. DevCPS.Get(sym)
  321. ELSE err(string)
  322. END
  323. END;
  324. IF n > 0 THEN
  325. NEW(obj.modifiers, n);
  326. WHILE n > 0 DO DEC(n); obj.modifiers[n] := mod[n] END
  327. END
  328. ELSE err(string)
  329. END;
  330. CheckSym(rbrak);
  331. IF DevCPM.options * {DevCPM.interface, DevCPM.java} = {} THEN err(225) END
  332. END
  333. END CheckMark;
  334. PROCEDURE CheckSysFlag (VAR sysflag: SHORTINT;
  335. GetSF: PROCEDURE(id: ARRAY OF SHORTCHAR; num: SHORTINT; VAR flag: SHORTINT));
  336. VAR x: DevCPT.Object; i: SHORTINT;
  337. BEGIN
  338. sysflag := 0;
  339. IF sym = lbrak THEN
  340. DevCPS.Get(sym);
  341. WHILE (sym = number) OR (sym = ident) OR (sym = string) DO
  342. IF sym = number THEN
  343. IF DevCPS.numtyp = integer THEN
  344. i := SHORT(DevCPS.intval); GetSF("", i, sysflag)
  345. ELSE err(225)
  346. END
  347. ELSIF sym = ident THEN
  348. DevCPT.Find(DevCPS.name, x);
  349. IF (x # NIL) & (x.mode = Con) & (x.typ.form IN {Int8, Int16, Int32}) THEN
  350. i := SHORT(x.conval.intval); GetSF("", i, sysflag)
  351. ELSE
  352. GetSF(DevCPS.name, 0, sysflag)
  353. END
  354. ELSE
  355. GetSF(DevCPS.str^, 0, sysflag)
  356. END;
  357. DevCPS.Get(sym);
  358. IF (sym = comma) OR (sym = plus) THEN DevCPS.Get(sym) END
  359. END;
  360. CheckSym(rbrak)
  361. END
  362. END CheckSysFlag;
  363. PROCEDURE Receiver(VAR mode, vis: BYTE; VAR name: DevCPT.Name; VAR typ, rec: DevCPT.Struct);
  364. VAR obj: DevCPT.Object; tname: DevCPT.String;
  365. BEGIN typ := DevCPT.undftyp; rec := NIL; vis := 0;
  366. IF sym = var THEN DevCPS.Get(sym); mode := VarPar;
  367. ELSIF sym = in THEN DevCPS.Get(sym); mode := VarPar; vis := inPar (* ??? *)
  368. ELSE mode := Var
  369. END ;
  370. name := DevCPS.name; CheckSym(ident); CheckSym(colon);
  371. IF sym # ident THEN err(ident) END;
  372. Type(typ, tname);
  373. IF tname = NIL THEN
  374. IF typ.form = Pointer THEN rec := typ.BaseTyp ELSE rec := typ END;
  375. IF ~((mode = Var) & (typ.form = Pointer) & (rec.comp = Record) OR
  376. (mode = VarPar) & (typ.comp = Record)) THEN err(70); rec := NIL END;
  377. IF (rec # NIL) & (rec.mno # level) THEN err(72); rec := NIL END
  378. ELSE err(0)
  379. END;
  380. CheckSym(rparen);
  381. IF rec = NIL THEN rec := DevCPT.NewStr(Comp, Record); rec.BaseTyp := NIL END
  382. END Receiver;
  383. PROCEDURE FormalParameters(
  384. VAR firstPar: DevCPT.Object; VAR resTyp: DevCPT.Struct; VAR name: DevCPT.String
  385. );
  386. VAR mode, vis: BYTE; sys: SHORTINT;
  387. par, first, last, res, newPar, iidPar: DevCPT.Object; typ: DevCPT.Struct;
  388. BEGIN
  389. first := NIL; last := firstPar;
  390. newPar := NIL; iidPar := NIL;
  391. IF (sym = ident) OR (sym = var) OR (sym = in) OR (sym = out) THEN
  392. LOOP
  393. sys := 0; vis := 0;
  394. IF sym = var THEN DevCPS.Get(sym); mode := VarPar
  395. ELSIF sym = in THEN DevCPS.Get(sym); mode := VarPar; vis := inPar
  396. ELSIF sym = out THEN DevCPS.Get(sym); mode := VarPar; vis := outPar
  397. ELSE mode := Var
  398. END ;
  399. IF mode = VarPar THEN CheckSysFlag(sys, DevCPM.GetVarParSysFlag) END;
  400. IF ODD(sys DIV inBit) THEN vis := inPar
  401. ELSIF ODD(sys DIV outBit) THEN vis := outPar
  402. END;
  403. IF ODD(sys DIV newBit) & (vis # outPar) THEN err(225)
  404. ELSIF ODD(sys DIV iidBit) & (vis # inPar) THEN err(225)
  405. END;
  406. LOOP
  407. IF sym = ident THEN
  408. DevCPT.Insert(DevCPS.name, par); DevCPS.Get(sym);
  409. par.mode := mode; par.link := NIL; par.vis := vis; par.sysflag := SHORT(sys);
  410. IF first = NIL THEN first := par END ;
  411. IF firstPar = NIL THEN firstPar := par ELSE last.link := par END ;
  412. last := par
  413. ELSE err(ident)
  414. END;
  415. IF sym = comma THEN DevCPS.Get(sym)
  416. ELSIF sym = ident THEN err(comma)
  417. ELSIF sym = var THEN err(comma); DevCPS.Get(sym)
  418. ELSE EXIT
  419. END
  420. END ;
  421. CheckSym(colon); Type(typ, name);
  422. IF mode # VarPar THEN CheckAlloc(typ, TRUE, DevCPM.errpos) END;
  423. IF (mode = VarPar) & (vis = inPar) & (typ.form # Undef) & (typ.form # Comp) & (typ.sysflag = 0) THEN err(177)
  424. END;
  425. (* typ.pbused is set when parameter type name is parsed *)
  426. WHILE first # NIL DO
  427. SetType (NIL, first, typ, name);
  428. IF DevCPM.com IN DevCPM.options THEN
  429. IF ODD(sys DIV newBit) THEN
  430. IF (newPar # NIL) OR (typ.form # Pointer) OR (typ.sysflag # interface) THEN err(168) END;
  431. newPar := first
  432. ELSIF ODD(sys DIV iidBit) THEN
  433. IF (iidPar # NIL) OR (typ # DevCPT.guidtyp) THEN err(168) END;
  434. iidPar := first
  435. END
  436. END;
  437. first := first.link
  438. END;
  439. IF sym = semicolon THEN DevCPS.Get(sym)
  440. ELSIF sym = ident THEN err(semicolon)
  441. ELSE EXIT
  442. END
  443. END
  444. END;
  445. CheckSym(rparen);
  446. IF (newPar = NIL) # (iidPar = NIL) THEN err(168) END;
  447. name := NIL;
  448. IF sym = colon THEN
  449. DevCPS.Get(sym);
  450. Type(resTyp, name);
  451. IF resTyp.form = Comp THEN resTyp := DevCPT.undftyp; err(54) END
  452. ELSE resTyp := DevCPT.notyp
  453. END
  454. END FormalParameters;
  455. PROCEDURE CheckOverwrite (proc, base: DevCPT.Object; rec: DevCPT.Struct);
  456. VAR o, bo: DevCPT.Object;
  457. BEGIN
  458. IF base # NIL THEN
  459. IF base.conval.setval * {absAttr, empAttr, extAttr} = {} THEN err(182) END;
  460. IF (proc.link.mode # base.link.mode) OR (proc.link.vis # base.link.vis)
  461. OR ~DevCPT.Extends(proc.link.typ, base.link.typ) THEN err(115) END;
  462. o := proc.link; bo := base.link;
  463. WHILE (o # NIL) & (bo # NIL) DO
  464. IF (bo.sysflag # 0) & (o.sysflag = 0) THEN (* propagate sysflags *)
  465. o.sysflag := bo.sysflag
  466. END;
  467. o := o.link; bo := bo.link
  468. END;
  469. DevCPB.CheckParameters(proc.link.link, base.link.link, FALSE);
  470. IF ~DevCPT.Extends(proc.typ, base.typ) THEN err(117) END;
  471. IF (base.vis # proc.vis) & ((proc.vis # internal) OR rec.exp) THEN err(183) END;
  472. INCL(proc.conval.setval, isRedef)
  473. END;
  474. END CheckOverwrite;
  475. PROCEDURE GetAttributes (proc, base: DevCPT.Object; owner: DevCPT.Struct); (* read method attributes *)
  476. VAR attr, battr: SET; o: DevCPT.Object;
  477. BEGIN
  478. attr := {};
  479. IF sym = comma THEN (* read attributes *)
  480. DevCPS.Get(sym);
  481. IF sym = ident THEN
  482. DevCPT.Find(DevCPS.name, o);
  483. IF (o # NIL) & (o.mode = SProc) & (o.adr = newfn) THEN
  484. IF ~(DevCPM.oberon IN DevCPM.options) THEN INCL(attr, newAttr) ELSE err(178) END;
  485. DevCPS.Get(sym);
  486. IF sym = comma THEN
  487. DevCPS.Get(sym);
  488. IF sym = ident THEN DevCPT.Find(DevCPS.name, o) ELSE o := NIL; err(ident) END
  489. ELSE o := NIL
  490. END
  491. END;
  492. IF o # NIL THEN
  493. IF (o.mode # Attr) OR (o.adr = limAttr) OR (DevCPM.oberon IN DevCPM.options) THEN err(178)
  494. ELSE INCL(attr, o.adr)
  495. END;
  496. DevCPS.Get(sym)
  497. END
  498. ELSE err(ident)
  499. END
  500. END;
  501. IF (base = NIL) & ~(newAttr IN attr) THEN err(185); INCL(attr, newAttr)
  502. ELSIF (base # NIL) & (newAttr IN attr) THEN err(186)
  503. END;
  504. IF absAttr IN attr THEN
  505. IF owner.attribute # absAttr THEN err(190) END;
  506. IF (proc.vis = internal) & owner.exp THEN err(179) END
  507. END;
  508. IF (owner.attribute = 0) OR (owner.attribute = limAttr) THEN
  509. IF (empAttr IN attr) & (newAttr IN attr) THEN err(187)
  510. (*
  511. ELSIF extAttr IN attr THEN err(188)
  512. *)
  513. END
  514. END;
  515. IF base # NIL THEN
  516. battr := base.conval.setval;
  517. IF empAttr IN battr THEN
  518. IF absAttr IN attr THEN err(189) END
  519. ELSIF ~(absAttr IN battr) THEN
  520. IF (absAttr IN attr) OR (empAttr IN attr) THEN err(189) END
  521. END
  522. END;
  523. IF empAttr IN attr THEN
  524. IF proc.typ # DevCPT.notyp THEN err(195)
  525. ELSE
  526. o := proc.link; WHILE (o # NIL) & (o.vis # outPar) DO o := o.link END;
  527. IF o # NIL THEN err(195) END
  528. END
  529. END;
  530. IF (owner.sysflag = interface) & ~(absAttr IN attr) THEN err(162) END;
  531. proc.conval.setval := attr
  532. END GetAttributes;
  533. PROCEDURE RecordType(VAR typ: DevCPT.Struct; attr: DevCPT.Object);
  534. VAR fld, first, last, base: DevCPT.Object; r: Elem; ftyp: DevCPT.Struct; name: DevCPT.String;
  535. BEGIN typ := DevCPT.NewStr(Comp, Record); typ.BaseTyp := NIL;
  536. CheckSysFlag(typ.sysflag, DevCPM.GetRecordSysFlag);
  537. IF attr # NIL THEN
  538. IF ~(DevCPM.oberon IN DevCPM.options) & (attr.adr # empAttr) THEN typ.attribute := SHORT(SHORT(attr.adr))
  539. ELSE err(178)
  540. END
  541. END;
  542. IF typ.sysflag = interface THEN
  543. IF (DevCPS.str # NIL) & (DevCPS.str[0] = "{") THEN typ.ext := DevCPS.str END;
  544. IF typ.attribute # absAttr THEN err(163) END;
  545. IF sym # lparen THEN err(160) END
  546. END;
  547. IF sym = lparen THEN
  548. DevCPS.Get(sym); (*record extension*)
  549. IF sym = ident THEN
  550. Type(ftyp, name);
  551. IF ftyp.form = Pointer THEN ftyp := ftyp.BaseTyp END;
  552. SetType(typ, NIL, ftyp, name);
  553. IF (ftyp.comp = Record) & (ftyp # DevCPT.anytyp) THEN
  554. ftyp.pvused := TRUE; typ.extlev := SHORT(SHORT(ftyp.extlev + 1));
  555. DevCPM.PropagateRecordSysFlag(ftyp.sysflag, typ.sysflag);
  556. IF (ftyp.attribute = 0) OR (ftyp.attribute = limAttr) & (ftyp.mno # 0) THEN err(181)
  557. ELSIF (typ.attribute = absAttr) & (ftyp.attribute # absAttr) & ~(DevCPM.java IN DevCPM.options) THEN err(191)
  558. ELSIF (ftyp.attribute = limAttr) & (typ.attribute # limAttr) THEN err(197)
  559. END
  560. ELSIF ftyp # DevCPT.undftyp THEN err(53)
  561. END
  562. ELSE err(ident)
  563. END ;
  564. IF typ.attribute # absAttr THEN (* save typ for unimplemented method check *)
  565. NEW(r); r.struct := typ; r.pos := DevCPM.errpos; r.next := recList; recList := r
  566. END;
  567. CheckSym(rparen)
  568. END;
  569. (*
  570. DevCPT.OpenScope(0, NIL);
  571. *)
  572. first := NIL; last := NIL;
  573. LOOP
  574. IF sym = ident THEN
  575. LOOP
  576. IF sym = ident THEN
  577. IF (typ.BaseTyp # NIL) & (typ.BaseTyp # DevCPT.undftyp) THEN
  578. DevCPT.FindBaseField(DevCPS.name, typ, fld);
  579. IF fld # NIL THEN err(1) END
  580. END ;
  581. DevCPT.InsertField(DevCPS.name, typ, fld);
  582. fld.mode := Fld; fld.link := NIL; fld.typ := DevCPT.undftyp;
  583. CheckMark(fld);
  584. IF first = NIL THEN first := fld END ;
  585. IF last = NIL THEN typ.link := fld ELSE last.link := fld END ;
  586. last := fld
  587. ELSE err(ident)
  588. END ;
  589. IF sym = comma THEN DevCPS.Get(sym)
  590. ELSIF sym = ident THEN err(comma)
  591. ELSE EXIT
  592. END
  593. END ;
  594. CheckSym(colon); Type(ftyp, name);
  595. CheckAlloc(ftyp, FALSE, DevCPM.errpos);
  596. WHILE first # NIL DO
  597. SetType(typ, first, ftyp, name); first := first.link
  598. END;
  599. IF typ.sysflag = interface THEN err(161) END
  600. END;
  601. IF sym = semicolon THEN DevCPS.Get(sym)
  602. ELSIF sym = ident THEN err(semicolon)
  603. ELSE EXIT
  604. END
  605. END;
  606. (*
  607. IF typ.link # NIL THEN ASSERT(typ.link = DevCPT.topScope.right) END;
  608. typ.link := DevCPT.topScope.right; DevCPT.CloseScope;
  609. *)
  610. typ.untagged := typ.sysflag > 0;
  611. DevCPB.Inittd(TDinit, lastTDinit, typ); CheckSym(end)
  612. END RecordType;
  613. PROCEDURE ArrayType(VAR typ: DevCPT.Struct);
  614. VAR x: DevCPT.Node; n: INTEGER; sysflag: SHORTINT; name: DevCPT.String;
  615. BEGIN CheckSysFlag(sysflag, DevCPM.GetArraySysFlag);
  616. IF sym = of THEN (*dynamic array*)
  617. typ := DevCPT.NewStr(Comp, DynArr); typ.mno := 0; typ.sysflag := sysflag;
  618. DevCPS.Get(sym); Type(typ.BaseTyp, name); SetType(typ, NIL, typ.BaseTyp, name);
  619. CheckAlloc(typ.BaseTyp, TRUE, DevCPM.errpos);
  620. IF typ.BaseTyp.comp = DynArr THEN typ.n := typ.BaseTyp.n + 1 ELSE typ.n := 0 END
  621. ELSE
  622. typ := DevCPT.NewStr(Comp, Array); typ.sysflag := sysflag; ConstExpression(x);
  623. IF x.typ.form IN {Int8, Int16, Int32} THEN n := x.conval.intval;
  624. IF (n <= 0) OR (n > DevCPM.MaxIndex) THEN err(63); n := 1 END
  625. ELSE err(42); n := 1
  626. END ;
  627. typ.n := n;
  628. IF sym = of THEN
  629. DevCPS.Get(sym); Type(typ.BaseTyp, name); SetType(typ, NIL, typ.BaseTyp, name);
  630. CheckAlloc(typ.BaseTyp, FALSE, DevCPM.errpos)
  631. ELSIF sym = comma THEN
  632. DevCPS.Get(sym);
  633. IF sym # of THEN ArrayType(typ.BaseTyp) END
  634. ELSE err(35)
  635. END
  636. END;
  637. typ.untagged := typ.sysflag > 0
  638. END ArrayType;
  639. PROCEDURE PointerType(VAR typ: DevCPT.Struct);
  640. VAR id: DevCPT.Object; name: DevCPT.String;
  641. BEGIN typ := DevCPT.NewStr(Pointer, Basic); CheckSysFlag(typ.sysflag, DevCPM.GetPointerSysFlag);
  642. CheckSym(to);
  643. Type(typ.BaseTyp, name);
  644. SetType(typ, NIL, typ.BaseTyp, name);
  645. IF (typ.BaseTyp # DevCPT.undftyp) & (typ.BaseTyp.comp = Basic) THEN
  646. typ.BaseTyp := DevCPT.undftyp; err(57)
  647. END;
  648. IF typ.BaseTyp.comp = Record THEN DevCPM.PropagateRecPtrSysFlag(typ.BaseTyp.sysflag, typ.sysflag)
  649. ELSIF typ.BaseTyp.comp IN {Array, DynArr} THEN DevCPM.PropagateArrPtrSysFlag(typ.BaseTyp.sysflag, typ.sysflag)
  650. END;
  651. typ.untagged := typ.sysflag > 0
  652. END PointerType;
  653. PROCEDURE Type (VAR typ: DevCPT.Struct; VAR name: DevCPT.String); (* name # NIL => forward reference *)
  654. VAR id: DevCPT.Object; tname: DevCPT.String;
  655. BEGIN
  656. typ := DevCPT.undftyp; name := NIL;
  657. IF sym < lparen THEN err(12);
  658. REPEAT DevCPS.Get(sym) UNTIL sym >= lparen
  659. END ;
  660. IF sym = ident THEN
  661. DevCPT.Find(DevCPS.name, id);
  662. IF (id = NIL) OR (id.mode = -1) OR (id.mode = Typ) & IncompleteType(id.typ) THEN (* forward type definition *)
  663. name := DevCPT.NewName(DevCPS.name); DevCPS.Get(sym);
  664. IF (id = NIL) & (sym = period) THEN (* missing module *)
  665. err(0); DevCPS.Get(sym); name := NIL;
  666. IF sym = ident THEN DevCPS.Get(sym) END
  667. ELSIF sym = record THEN (* wrong attribute *)
  668. err(178); DevCPS.Get(sym); name := NIL; RecordType(typ, NIL)
  669. END
  670. ELSE
  671. qualident(id);
  672. IF id.mode = Typ THEN
  673. IF ~(DevCPM.oberon IN DevCPM.options)
  674. & ((id.typ = DevCPT.lreal64typ) OR (id.typ = DevCPT.lint64typ) OR (id.typ = DevCPT.lchar16typ)) THEN
  675. err(198)
  676. END;
  677. typ := id.typ
  678. ELSIF id.mode = Attr THEN
  679. IF sym = record THEN
  680. DevCPS.Get(sym); RecordType(typ, id)
  681. ELSE err(12)
  682. END
  683. ELSE err(52)
  684. END
  685. END
  686. ELSIF sym = array THEN
  687. DevCPS.Get(sym); ArrayType(typ)
  688. ELSIF sym = record THEN
  689. DevCPS.Get(sym); RecordType(typ, NIL)
  690. ELSIF sym = pointer THEN
  691. DevCPS.Get(sym); PointerType(typ)
  692. ELSIF sym = procedure THEN
  693. DevCPS.Get(sym); typ := DevCPT.NewStr(ProcTyp, Basic);
  694. CheckSysFlag(typ.sysflag, DevCPM.GetProcTypSysFlag);
  695. typ.untagged := typ.sysflag > 0;
  696. IF sym = lparen THEN
  697. DevCPS.Get(sym); DevCPT.OpenScope(level, NIL);
  698. FormalParameters(typ.link, typ.BaseTyp, tname); SetType(typ, NIL, typ.BaseTyp, tname); DevCPT.CloseScope
  699. ELSE typ.BaseTyp := DevCPT.notyp; typ.link := NIL
  700. END
  701. ELSE err(12)
  702. END ;
  703. LOOP
  704. IF (sym >= semicolon) & (sym <= else) OR (sym = rparen) OR (sym = eof)
  705. OR (sym = number) OR (sym = comma) OR (sym = string) THEN EXIT END;
  706. err(15); IF sym = ident THEN EXIT END;
  707. DevCPS.Get(sym)
  708. END
  709. END Type;
  710. PROCEDURE ActualParameters(VAR aparlist: DevCPT.Node; fpar: DevCPT.Object; VAR pre, lastp: DevCPT.Node);
  711. VAR apar, last, newPar, iidPar, n: DevCPT.Node;
  712. BEGIN
  713. aparlist := NIL; last := NIL;
  714. IF sym # rparen THEN
  715. newPar := NIL; iidPar := NIL;
  716. LOOP Expression(apar);
  717. IF fpar # NIL THEN
  718. IF (apar.typ.form = Pointer) & (fpar.typ.form = Comp) THEN DevCPB.DeRef(apar) END;
  719. DevCPB.Param(apar, fpar);
  720. IF (fpar.mode = Var) OR (fpar.vis = inPar) THEN DevCPB.CheckBuffering(apar, NIL, fpar, pre, lastp) END;
  721. DevCPB.Link(aparlist, last, apar);
  722. IF ODD(fpar.sysflag DIV newBit) THEN newPar := apar
  723. ELSIF ODD(fpar.sysflag DIV iidBit) THEN iidPar := apar
  724. END;
  725. IF (newPar # NIL) & (iidPar # NIL) THEN DevCPB.CheckNewParamPair(newPar, iidPar) END;
  726. IF anchorVarPar & (fpar.mode = VarPar) & ~(DevCPM.java IN DevCPM.options)
  727. OR (DevCPM.allSysVal IN DevCPM.options) (* source output: avoid double evaluation *)
  728. & ((fpar.mode = VarPar) & (fpar.typ.comp = Record) & ~fpar.typ.untagged
  729. OR (fpar.typ.comp = DynArr) & ~fpar.typ.untagged) THEN
  730. n := apar;
  731. WHILE n.class IN {Nfield, Nindex, Nguard} DO n := n.left END;
  732. IF (n.class = Nderef) & (n.subcl = 0) THEN
  733. IF n.left.class = Nguard THEN n := n.left END;
  734. DevCPB.CheckVarParBuffering(n.left, pre, lastp)
  735. END
  736. END;
  737. fpar := fpar.link
  738. ELSE err(64)
  739. END;
  740. IF sym = comma THEN DevCPS.Get(sym)
  741. ELSIF (lparen <= sym) & (sym <= ident) THEN err(comma)
  742. ELSE EXIT
  743. END
  744. END
  745. END;
  746. IF fpar # NIL THEN err(65) END
  747. END ActualParameters;
  748. PROCEDURE selector(VAR x: DevCPT.Node);
  749. VAR obj, proc, p, fpar: DevCPT.Object; y, apar, pre, lastp: DevCPT.Node; typ: DevCPT.Struct; name: DevCPT.Name;
  750. BEGIN
  751. LOOP
  752. IF sym = lbrak THEN DevCPS.Get(sym);
  753. LOOP
  754. IF (x.typ # NIL) & (x.typ.form = Pointer) THEN DevCPB.DeRef(x) END ;
  755. Expression(y); DevCPB.Index(x, y);
  756. IF sym = comma THEN DevCPS.Get(sym) ELSE EXIT END
  757. END ;
  758. CheckSym(rbrak)
  759. ELSIF sym = period THEN DevCPS.Get(sym);
  760. IF sym = ident THEN name := DevCPS.name; DevCPS.Get(sym);
  761. IF x.typ # NIL THEN
  762. IF x.typ.form = Pointer THEN DevCPB.DeRef(x) END ;
  763. IF x.typ.comp = Record THEN
  764. typ := x.typ; DevCPT.FindField(name, typ, obj); DevCPB.Field(x, obj);
  765. IF (obj # NIL) & (obj.mode = TProc) THEN
  766. IF sym = arrow THEN (* super call *) DevCPS.Get(sym);
  767. y := x.left;
  768. IF y.class = Nderef THEN y := y.left END ; (* y = record variable *)
  769. IF y.obj # NIL THEN
  770. proc := DevCPT.topScope; (* find innermost scope which owner is a TProc *)
  771. WHILE (proc.link # NIL) & (proc.link.mode # TProc) DO proc := proc.left END ;
  772. IF (proc.link = NIL) OR (proc.link.link # y.obj) (* OR (proc.link.name^ # name) *) THEN err(75)
  773. END ;
  774. typ := y.obj.typ;
  775. IF typ.form = Pointer THEN typ := typ.BaseTyp END ;
  776. DevCPT.FindBaseField(x.obj.name^, typ, p);
  777. IF p # NIL THEN
  778. x.subcl := super; x.typ := p.typ; (* correct result type *)
  779. IF p.conval.setval * {absAttr, empAttr} # {} THEN err(194) END;
  780. IF (p.vis = externalR) & (p.mnolev < 0) & (proc.link.name^ # name) THEN err(196) END;
  781. ELSE err(74)
  782. END
  783. ELSE err(75)
  784. END
  785. ELSE
  786. proc := obj;
  787. WHILE (proc.mnolev >= 0) & ~(newAttr IN proc.conval.setval) & (typ.BaseTyp # NIL) DO
  788. (* find base method *)
  789. typ := typ.BaseTyp; DevCPT.FindField(name, typ, proc);
  790. END;
  791. IF (proc.vis = externalR) & (proc.mnolev < 0) THEN err(196) END;
  792. END ;
  793. IF (obj.typ # DevCPT.notyp) & (sym # lparen) THEN err(lparen) END
  794. END
  795. ELSE err(53)
  796. END
  797. ELSE err(52)
  798. END
  799. ELSE err(ident)
  800. END
  801. ELSIF sym = arrow THEN DevCPS.Get(sym); DevCPB.DeRef(x)
  802. ELSIF sym = dollar THEN
  803. IF x.typ.form = Pointer THEN DevCPB.DeRef(x) END;
  804. DevCPS.Get(sym); DevCPB.StrDeref(x)
  805. ELSIF sym = lparen THEN
  806. IF (x.obj # NIL) & (x.obj.mode IN {XProc, LProc, CProc, TProc}) THEN typ := x.obj.typ
  807. ELSIF x.typ.form = ProcTyp THEN typ := x.typ.BaseTyp
  808. ELSIF x.class = Nproc THEN EXIT (* standard procedure *)
  809. ELSE typ := NIL
  810. END;
  811. IF typ # DevCPT.notyp THEN
  812. DevCPS.Get(sym);
  813. IF typ = NIL THEN (* type guard *)
  814. IF sym = ident THEN
  815. qualident(obj);
  816. IF obj.mode = Typ THEN DevCPB.TypTest(x, obj, TRUE)
  817. ELSE err(52)
  818. END
  819. ELSE err(ident)
  820. END
  821. ELSE (* function call *)
  822. pre := NIL; lastp := NIL;
  823. DevCPB.PrepCall(x, fpar);
  824. IF (x.obj # NIL) & (x.obj.mode = TProc) THEN DevCPB.CheckBuffering(x.left, NIL, x.obj.link, pre, lastp)
  825. END;
  826. ActualParameters(apar, fpar, pre, lastp);
  827. DevCPB.Call(x, apar, fpar);
  828. IF pre # NIL THEN DevCPB.Construct(Ncomp, pre, x); pre.typ := x.typ; x := pre END;
  829. IF level > 0 THEN DevCPT.topScope.link.leaf := FALSE END
  830. END;
  831. CheckSym(rparen)
  832. ELSE EXIT
  833. END
  834. (*
  835. ELSIF (sym = lparen) & (x.class # Nproc) & (x.typ.form # ProcTyp) &
  836. ((x.obj = NIL) OR (x.obj.mode # TProc)) THEN
  837. DevCPS.Get(sym);
  838. IF sym = ident THEN
  839. qualident(obj);
  840. IF obj.mode = Typ THEN DevCPB.TypTest(x, obj, TRUE)
  841. ELSE err(52)
  842. END
  843. ELSE err(ident)
  844. END ;
  845. CheckSym(rparen)
  846. *)
  847. ELSE EXIT
  848. END
  849. END
  850. END selector;
  851. PROCEDURE StandProcCall(VAR x: DevCPT.Node);
  852. VAR y: DevCPT.Node; m: BYTE; n: SHORTINT;
  853. BEGIN m := SHORT(SHORT(x.obj.adr)); n := 0;
  854. IF sym = lparen THEN DevCPS.Get(sym);
  855. IF sym # rparen THEN
  856. LOOP
  857. IF n = 0 THEN Expression(x); DevCPB.StPar0(x, m); n := 1
  858. ELSIF n = 1 THEN Expression(y); DevCPB.StPar1(x, y, m); n := 2
  859. ELSE Expression(y); DevCPB.StParN(x, y, m, n); INC(n)
  860. END ;
  861. IF sym = comma THEN DevCPS.Get(sym)
  862. ELSIF (lparen <= sym) & (sym <= ident) THEN err(comma)
  863. ELSE EXIT
  864. END
  865. END ;
  866. CheckSym(rparen)
  867. ELSE DevCPS.Get(sym)
  868. END ;
  869. DevCPB.StFct(x, m, n)
  870. ELSE err(lparen)
  871. END ;
  872. IF (level > 0) & ((m = newfn) OR (m = sysnewfn)) THEN DevCPT.topScope.link.leaf := FALSE END
  873. END StandProcCall;
  874. PROCEDURE Element(VAR x: DevCPT.Node);
  875. VAR y: DevCPT.Node;
  876. BEGIN Expression(x);
  877. IF sym = upto THEN
  878. DevCPS.Get(sym); Expression(y); DevCPB.SetRange(x, y)
  879. ELSE DevCPB.SetElem(x)
  880. END
  881. END Element;
  882. PROCEDURE Sets(VAR x: DevCPT.Node);
  883. VAR y: DevCPT.Node;
  884. BEGIN
  885. IF sym # rbrace THEN
  886. Element(x);
  887. LOOP
  888. IF sym = comma THEN DevCPS.Get(sym)
  889. ELSIF (lparen <= sym) & (sym <= ident) THEN err(comma)
  890. ELSE EXIT
  891. END ;
  892. Element(y); DevCPB.Op(plus, x, y)
  893. END
  894. ELSE x := DevCPB.EmptySet()
  895. END ;
  896. CheckSym(rbrace)
  897. END Sets;
  898. PROCEDURE Factor(VAR x: DevCPT.Node);
  899. VAR fpar, id: DevCPT.Object; apar: DevCPT.Node;
  900. BEGIN
  901. IF sym < not THEN err(13);
  902. REPEAT DevCPS.Get(sym) UNTIL sym >= lparen
  903. END ;
  904. IF sym = ident THEN
  905. qualident(id); x := DevCPB.NewLeaf(id); selector(x);
  906. IF (x.class = Nproc) & (x.obj.mode = SProc) THEN StandProcCall(x) (* x may be NIL *)
  907. (*
  908. ELSIF sym = lparen THEN
  909. DevCPS.Get(sym); DevCPB.PrepCall(x, fpar);
  910. ActualParameters(apar, fpar);
  911. DevCPB.Call(x, apar, fpar);
  912. CheckSym(rparen);
  913. IF level > 0 THEN DevCPT.topScope.link.leaf := FALSE END
  914. *)
  915. END
  916. ELSIF sym = number THEN
  917. CASE DevCPS.numtyp OF
  918. char:
  919. x := DevCPB.NewIntConst(DevCPS.intval); x.typ := DevCPT.char8typ;
  920. IF DevCPS.intval > 255 THEN x.typ := DevCPT.char16typ END
  921. | integer: x := DevCPB.NewIntConst(DevCPS.intval)
  922. | int64: x := DevCPB.NewLargeIntConst(DevCPS.intval, DevCPS.realval)
  923. | real: x := DevCPB.NewRealConst(DevCPS.realval, NIL)
  924. | real32: x := DevCPB.NewRealConst(DevCPS.realval, DevCPT.real32typ)
  925. | real64: x := DevCPB.NewRealConst(DevCPS.realval, DevCPT.real64typ)
  926. END ;
  927. DevCPS.Get(sym)
  928. ELSIF sym = string THEN
  929. x := DevCPB.NewString(DevCPS.str, DevCPS.lstr, DevCPS.intval);
  930. DevCPS.Get(sym)
  931. ELSIF sym = nil THEN
  932. x := DevCPB.Nil(); DevCPS.Get(sym)
  933. ELSIF sym = lparen THEN
  934. DevCPS.Get(sym); Expression(x); CheckSym(rparen)
  935. ELSIF sym = lbrak THEN
  936. DevCPS.Get(sym); err(lparen); Expression(x); CheckSym(rparen)
  937. ELSIF sym = lbrace THEN DevCPS.Get(sym); Sets(x)
  938. ELSIF sym = not THEN
  939. DevCPS.Get(sym); Factor(x); DevCPB.MOp(not, x)
  940. ELSE err(13); DevCPS.Get(sym); x := NIL
  941. END ;
  942. IF x = NIL THEN x := DevCPB.NewIntConst(1); x.typ := DevCPT.undftyp END
  943. END Factor;
  944. PROCEDURE Term(VAR x: DevCPT.Node);
  945. VAR y: DevCPT.Node; mulop: BYTE;
  946. BEGIN Factor(x);
  947. WHILE (times <= sym) & (sym <= and) DO
  948. mulop := sym; DevCPS.Get(sym);
  949. Factor(y); DevCPB.Op(mulop, x, y)
  950. END
  951. END Term;
  952. PROCEDURE SimpleExpression(VAR x: DevCPT.Node);
  953. VAR y: DevCPT.Node; addop: BYTE;
  954. BEGIN
  955. IF sym = minus THEN DevCPS.Get(sym); Term(x); DevCPB.MOp(minus, x)
  956. ELSIF sym = plus THEN DevCPS.Get(sym); Term(x); DevCPB.MOp(plus, x)
  957. ELSE Term(x)
  958. END ;
  959. WHILE (plus <= sym) & (sym <= or) DO
  960. addop := sym; DevCPS.Get(sym); Term(y);
  961. IF x.typ.form = Pointer THEN DevCPB.DeRef(x) END;
  962. IF (x.typ.comp IN {Array, DynArr}) & (x.typ.BaseTyp.form IN charSet) (* OR (x.typ.sysflag = jstr) *) THEN
  963. DevCPB.StrDeref(x)
  964. END;
  965. IF y.typ.form = Pointer THEN DevCPB.DeRef(y) END;
  966. IF (y.typ.comp IN {Array, DynArr}) & (y.typ.BaseTyp.form IN charSet) (* OR (y.typ.sysflag = jstr) *) THEN
  967. DevCPB.StrDeref(y)
  968. END;
  969. DevCPB.Op(addop, x, y)
  970. END
  971. END SimpleExpression;
  972. PROCEDURE Expression(VAR x: DevCPT.Node);
  973. VAR y, pre, last: DevCPT.Node; obj: DevCPT.Object; relation: BYTE;
  974. BEGIN SimpleExpression(x);
  975. IF (eql <= sym) & (sym <= geq) THEN
  976. relation := sym; DevCPS.Get(sym); SimpleExpression(y);
  977. pre := NIL; last := NIL;
  978. IF (x.typ.comp IN {Array, DynArr}) & (x.typ.BaseTyp.form IN charSet) THEN
  979. DevCPB.StrDeref(x)
  980. END;
  981. IF (y.typ.comp IN {Array, DynArr}) & (y.typ.BaseTyp.form IN charSet) THEN
  982. DevCPB.StrDeref(y)
  983. END;
  984. DevCPB.CheckBuffering(x, NIL, NIL, pre, last);
  985. DevCPB.CheckBuffering(y, NIL, NIL, pre, last);
  986. DevCPB.Op(relation, x, y);
  987. IF pre # NIL THEN DevCPB.Construct(Ncomp, pre, x); pre.typ := x.typ; x := pre END
  988. ELSIF sym = in THEN
  989. DevCPS.Get(sym); SimpleExpression(y); DevCPB.In(x, y)
  990. ELSIF sym = is THEN
  991. DevCPS.Get(sym);
  992. IF sym = ident THEN
  993. qualident(obj);
  994. IF obj.mode = Typ THEN DevCPB.TypTest(x, obj, FALSE)
  995. ELSE err(52)
  996. END
  997. ELSE err(ident)
  998. END
  999. END
  1000. END Expression;
  1001. PROCEDURE ProcedureDeclaration(VAR x: DevCPT.Node);
  1002. VAR proc, fwd: DevCPT.Object;
  1003. name: DevCPT.Name;
  1004. mode: BYTE;
  1005. forward: BOOLEAN;
  1006. sys: SHORTINT;
  1007. PROCEDURE GetCode;
  1008. VAR ext: DevCPT.ConstExt; i, n, c: INTEGER; s: ARRAY 256 OF SHORTCHAR;
  1009. BEGIN
  1010. n := 0;
  1011. IF sym = string THEN
  1012. NEW(ext, DevCPS.intval);
  1013. WHILE DevCPS.str[n] # 0X DO ext[n+1] := DevCPS.str[n]; INC(n) END ;
  1014. ext^[0] := SHORT(CHR(n)); DevCPS.Get(sym);
  1015. ELSE
  1016. LOOP
  1017. IF sym = number THEN c := DevCPS.intval; INC(n);
  1018. IF (c < 0) OR (c > 255) OR (n = 255) THEN
  1019. err(64); c := 1; n := 1
  1020. END ;
  1021. DevCPS.Get(sym); s[n] := SHORT(CHR(c))
  1022. END ;
  1023. IF sym = comma THEN DevCPS.Get(sym)
  1024. ELSIF sym = number THEN err(comma)
  1025. ELSE s[0] := SHORT(CHR(n)); EXIT
  1026. END
  1027. END;
  1028. NEW(ext, n + 1); i := 0;
  1029. WHILE i <= n DO ext[i] := s[i]; INC(i) END;
  1030. END;
  1031. proc.conval.ext := ext;
  1032. INCL(proc.conval.setval, hasBody)
  1033. END GetCode;
  1034. PROCEDURE GetParams;
  1035. VAR name: DevCPT.String;
  1036. BEGIN
  1037. proc.mode := mode; proc.typ := DevCPT.notyp;
  1038. proc.sysflag := SHORT(sys);
  1039. proc.conval.setval := {};
  1040. IF sym = lparen THEN
  1041. DevCPS.Get(sym); FormalParameters(proc.link, proc.typ, name);
  1042. IF name # NIL THEN err(0) END
  1043. END;
  1044. CheckForwardTypes; userList := NIL;
  1045. IF fwd # NIL THEN
  1046. DevCPB.CheckParameters(proc.link, fwd.link, TRUE);
  1047. IF ~DevCPT.EqualType(proc.typ, fwd.typ) THEN err(117) END ;
  1048. proc := fwd; DevCPT.topScope := proc.scope;
  1049. IF mode = IProc THEN proc.mode := IProc END
  1050. END
  1051. END GetParams;
  1052. PROCEDURE Body;
  1053. VAR procdec, statseq: DevCPT.Node; c: INTEGER;
  1054. BEGIN
  1055. c := DevCPM.errpos;
  1056. INCL(proc.conval.setval, hasBody);
  1057. CheckSym(semicolon); Block(procdec, statseq);
  1058. DevCPB.Enter(procdec, statseq, proc); x := procdec;
  1059. x.conval := DevCPT.NewConst(); x.conval.intval := c; x.conval.intval2 := DevCPM.startpos;
  1060. CheckSym(end);
  1061. IF sym = ident THEN
  1062. IF DevCPS.name # proc.name^ THEN err(4) END ;
  1063. DevCPS.Get(sym)
  1064. ELSE err(ident)
  1065. END
  1066. END Body;
  1067. PROCEDURE TProcDecl;
  1068. VAR baseProc, o, bo: DevCPT.Object;
  1069. objTyp, recTyp: DevCPT.Struct;
  1070. objMode, objVis: BYTE;
  1071. objName: DevCPT.Name;
  1072. pnode: DevCPT.Node;
  1073. fwdAttr: SET;
  1074. BEGIN
  1075. DevCPS.Get(sym); mode := TProc;
  1076. IF level > 0 THEN err(73) END;
  1077. Receiver(objMode, objVis, objName, objTyp, recTyp);
  1078. IF sym = ident THEN
  1079. name := DevCPS.name;
  1080. DevCPT.FindField(name, recTyp, fwd);
  1081. DevCPT.FindBaseField(name, recTyp, baseProc);
  1082. IF (baseProc # NIL) & (baseProc.mode # TProc) THEN baseProc := NIL; err(1) END ;
  1083. IF fwd = baseProc THEN fwd := NIL END ;
  1084. IF (fwd # NIL) & (fwd.mnolev # level) THEN fwd := NIL END ;
  1085. IF (fwd # NIL) & (fwd.mode = TProc) & (fwd.conval.setval * {hasBody, absAttr, empAttr} = {}) THEN
  1086. (* there exists a corresponding forward declaration *)
  1087. proc := DevCPT.NewObj(); proc.leaf := TRUE;
  1088. proc.mode := TProc; proc.conval := DevCPT.NewConst();
  1089. CheckMark(proc);
  1090. IF fwd.vis # proc.vis THEN err(118) END;
  1091. fwdAttr := fwd.conval.setval
  1092. ELSE
  1093. IF fwd # NIL THEN err(1); fwd := NIL END ;
  1094. DevCPT.InsertField(name, recTyp, proc);
  1095. proc.mode := TProc; proc.conval := DevCPT.NewConst();
  1096. CheckMark(proc);
  1097. IF recTyp.strobj # NIL THEN (* preserve declaration order *)
  1098. o := recTyp.strobj.link;
  1099. IF o = NIL THEN recTyp.strobj.link := proc
  1100. ELSE
  1101. WHILE o.nlink # NIL DO o := o.nlink END;
  1102. o.nlink := proc
  1103. END
  1104. END
  1105. END;
  1106. INC(level); DevCPT.OpenScope(level, proc);
  1107. DevCPT.Insert(objName, proc.link); proc.link.mode := objMode; proc.link.vis := objVis; proc.link.typ := objTyp;
  1108. ASSERT(DevCPT.topScope # NIL);
  1109. GetParams; (* may change proc := fwd !!! *)
  1110. ASSERT(DevCPT.topScope # NIL);
  1111. GetAttributes(proc, baseProc, recTyp);
  1112. IF (fwd # NIL) & (fwdAttr / proc.conval.setval * {absAttr, empAttr, extAttr} # {}) THEN err(184) END;
  1113. CheckOverwrite(proc, baseProc, recTyp);
  1114. IF ~forward THEN
  1115. IF empAttr IN proc.conval.setval THEN (* insert empty procedure *)
  1116. pnode := NIL; DevCPB.Enter(pnode, NIL, proc);
  1117. pnode.conval := DevCPT.NewConst();
  1118. pnode.conval.intval := DevCPM.errpos;
  1119. pnode.conval.intval2 := DevCPM.errpos;
  1120. x := pnode;
  1121. ELSIF DevCPM.noCode IN DevCPM.options THEN INCL(proc.conval.setval, hasBody)
  1122. ELSIF ~(absAttr IN proc.conval.setval) THEN Body
  1123. END;
  1124. proc.adr := 0
  1125. ELSE
  1126. proc.adr := DevCPM.errpos;
  1127. IF proc.conval.setval * {empAttr, absAttr} # {} THEN err(184) END
  1128. END;
  1129. DEC(level); DevCPT.CloseScope;
  1130. ELSE err(ident)
  1131. END;
  1132. END TProcDecl;
  1133. BEGIN proc := NIL; forward := FALSE; x := NIL; mode := LProc; sys := 0;
  1134. IF (sym # ident) & (sym # lparen) THEN
  1135. CheckSysFlag(sys, DevCPM.GetProcSysFlag);
  1136. IF sys # 0 THEN
  1137. IF ODD(sys DIV DevCPM.CProcFlag) THEN mode := CProc END
  1138. ELSE
  1139. IF sym = times THEN (* mode set later in DevCPB.CheckAssign *)
  1140. ELSIF sym = arrow THEN forward := TRUE
  1141. ELSE err(ident)
  1142. END;
  1143. DevCPS.Get(sym)
  1144. END
  1145. END ;
  1146. IF sym = lparen THEN TProcDecl
  1147. ELSIF sym = ident THEN DevCPT.Find(DevCPS.name, fwd);
  1148. name := DevCPS.name;
  1149. IF (fwd # NIL) & ((fwd.mnolev # level) OR (fwd.mode = SProc)) THEN fwd := NIL END ;
  1150. IF (fwd # NIL) & (fwd.mode IN {LProc, XProc}) & ~(hasBody IN fwd.conval.setval) THEN
  1151. (* there exists a corresponding forward declaration *)
  1152. proc := DevCPT.NewObj(); proc.leaf := TRUE;
  1153. proc.mode := mode; proc.conval := DevCPT.NewConst();
  1154. CheckMark(proc);
  1155. IF fwd.vis # proc.vis THEN err(118) END
  1156. ELSE
  1157. IF fwd # NIL THEN err(1); fwd := NIL END ;
  1158. DevCPT.Insert(name, proc);
  1159. proc.mode := mode; proc.conval := DevCPT.NewConst();
  1160. CheckMark(proc);
  1161. END ;
  1162. IF (proc.vis # internal) & (mode = LProc) THEN mode := XProc END ;
  1163. IF (mode # LProc) & (level > 0) THEN err(73) END ;
  1164. INC(level); DevCPT.OpenScope(level, proc);
  1165. proc.link := NIL; GetParams; (* may change proc := fwd !!! *)
  1166. IF mode = CProc THEN GetCode
  1167. ELSIF DevCPM.noCode IN DevCPM.options THEN INCL(proc.conval.setval, hasBody)
  1168. ELSIF ~forward THEN Body; proc.adr := 0
  1169. ELSE proc.adr := DevCPM.errpos
  1170. END ;
  1171. DEC(level); DevCPT.CloseScope
  1172. ELSE err(ident)
  1173. END
  1174. END ProcedureDeclaration;
  1175. PROCEDURE CaseLabelList(VAR lab, root: DevCPT.Node; LabelForm: SHORTINT; VAR min, max: INTEGER);
  1176. VAR x, y, lastlab: DevCPT.Node; i, f: SHORTINT; xval, yval: INTEGER;
  1177. PROCEDURE Insert(VAR n: DevCPT.Node); (* build binary tree of label ranges *) (* !!! *)
  1178. BEGIN
  1179. IF n = NIL THEN
  1180. IF x.hint # 1 THEN n := x END
  1181. ELSIF yval < n.conval.intval THEN Insert(n.left)
  1182. ELSIF xval > n.conval.intval2 THEN Insert(n.right)
  1183. ELSE err(63)
  1184. END
  1185. END Insert;
  1186. BEGIN lab := NIL; lastlab := NIL;
  1187. LOOP ConstExpression(x); f := x.typ.form;
  1188. IF f IN {Int8..Int32} + charSet THEN xval := x.conval.intval
  1189. ELSE err(61); xval := 1
  1190. END ;
  1191. IF (f IN {Int8..Int32}) # (LabelForm IN {Int8..Int32}) THEN err(60) END;
  1192. IF sym = upto THEN
  1193. DevCPS.Get(sym); ConstExpression(y); yval := y.conval.intval;
  1194. IF (y.typ.form IN {Int8..Int32}) # (LabelForm IN {Int8..Int32}) THEN err(60) END;
  1195. IF yval < xval THEN err(63); yval := xval END
  1196. ELSE yval := xval
  1197. END ;
  1198. x.conval.intval2 := yval;
  1199. IF xval < min THEN min := xval END;
  1200. IF yval > max THEN max := yval END;
  1201. IF lab = NIL THEN lab := x; Insert(root)
  1202. ELSIF yval < lab.conval.intval - 1 THEN x.link := lab; lab := x; Insert(root)
  1203. ELSIF yval = lab.conval.intval - 1 THEN x.hint := 1; Insert(root); lab.conval.intval := xval
  1204. ELSIF xval = lab.conval.intval2 + 1 THEN x.hint := 1; Insert(root); lab.conval.intval2 := yval
  1205. ELSE
  1206. y := lab;
  1207. WHILE (y.link # NIL) & (xval > y.link.conval.intval2 + 1) DO y := y.link END;
  1208. IF y.link = NIL THEN y.link := x; Insert(root)
  1209. ELSIF yval < y.link.conval.intval - 1 THEN x.link := y.link; y.link := x; Insert(root)
  1210. ELSIF yval = y.link.conval.intval - 1 THEN x.hint := 1; Insert(root); y.link.conval.intval := xval
  1211. ELSIF xval = y.link.conval.intval2 + 1 THEN x.hint := 1; Insert(root); y.link.conval.intval2 := yval
  1212. END
  1213. END;
  1214. IF sym = comma THEN DevCPS.Get(sym)
  1215. ELSIF (sym = number) OR (sym = ident) THEN err(comma)
  1216. ELSE EXIT
  1217. END
  1218. END
  1219. END CaseLabelList;
  1220. PROCEDURE StatSeq(VAR stat: DevCPT.Node);
  1221. VAR fpar, id, t, obj: DevCPT.Object; idtyp: DevCPT.Struct; e: BOOLEAN;
  1222. s, x, y, z, apar, last, lastif, pre, lastp: DevCPT.Node; pos, p: INTEGER; name: DevCPT.Name;
  1223. PROCEDURE CasePart(VAR x: DevCPT.Node);
  1224. VAR low, high: INTEGER; e: BOOLEAN; cases, lab, y, lastcase, root: DevCPT.Node;
  1225. BEGIN
  1226. Expression(x);
  1227. IF (x.class = Ntype) OR (x.class = Nproc) THEN err(126)
  1228. ELSIF x.typ.form = Int64 THEN err(260)
  1229. ELSIF ~(x.typ.form IN {Int8..Int32} + charSet) THEN err(125)
  1230. END ;
  1231. CheckSym(of); cases := NIL; lastcase := NIL; root := NIL;
  1232. low := MAX(INTEGER); high := MIN(INTEGER);
  1233. LOOP
  1234. IF sym < bar THEN
  1235. CaseLabelList(lab, root, x.typ.form, low, high);
  1236. CheckSym(colon); StatSeq(y);
  1237. DevCPB.Construct(Ncasedo, lab, y); DevCPB.Link(cases, lastcase, lab)
  1238. END ;
  1239. IF sym = bar THEN DevCPS.Get(sym) ELSE EXIT END
  1240. END;
  1241. e := sym = else;
  1242. IF e THEN DevCPS.Get(sym); StatSeq(y) ELSE y := NIL END ;
  1243. DevCPB.Construct(Ncaselse, cases, y); DevCPB.Construct(Ncase, x, cases);
  1244. cases.conval := DevCPT.NewConst();
  1245. cases.conval.intval := low; cases.conval.intval2 := high;
  1246. IF e THEN cases.conval.setval := {1} ELSE cases.conval.setval := {} END;
  1247. DevCPB.OptimizeCase(root); cases.link := root (* !!! *)
  1248. END CasePart;
  1249. PROCEDURE SetPos(x: DevCPT.Node);
  1250. BEGIN
  1251. x.conval := DevCPT.NewConst(); x.conval.intval := pos
  1252. END SetPos;
  1253. PROCEDURE CheckBool(VAR x: DevCPT.Node);
  1254. BEGIN
  1255. IF (x.class = Ntype) OR (x.class = Nproc) THEN err(126); x := DevCPB.NewBoolConst(FALSE)
  1256. ELSIF x.typ.form # Bool THEN err(120); x := DevCPB.NewBoolConst(FALSE)
  1257. END
  1258. END CheckBool;
  1259. BEGIN stat := NIL; last := NIL;
  1260. LOOP x := NIL;
  1261. IF sym < ident THEN err(14);
  1262. REPEAT DevCPS.Get(sym) UNTIL sym >= ident
  1263. END ;
  1264. pos := DevCPM.startpos;
  1265. IF sym = ident THEN
  1266. qualident(id); x := DevCPB.NewLeaf(id); selector(x);
  1267. IF sym = becomes THEN
  1268. DevCPS.Get(sym); Expression(y);
  1269. IF (y.typ.form = Pointer) & (x.typ.form = Comp) THEN DevCPB.DeRef(y) END;
  1270. pre := NIL; lastp := NIL;
  1271. DevCPB.CheckBuffering(y, x, NIL, pre, lastp);
  1272. DevCPB.Assign(x, y);
  1273. IF pre # NIL THEN SetPos(x); DevCPB.Construct(Ncomp, pre, x); x := pre END;
  1274. ELSIF sym = eql THEN
  1275. err(becomes); DevCPS.Get(sym); Expression(y); DevCPB.Assign(x, y)
  1276. ELSIF (x.class = Nproc) & (x.obj.mode = SProc) THEN
  1277. StandProcCall(x);
  1278. IF (x # NIL) & (x.typ # DevCPT.notyp) THEN err(55) END;
  1279. IF (x # NIL) & (x.class = Nifelse) THEN (* error pos for ASSERT *)
  1280. SetPos(x.left); SetPos(x.left.right)
  1281. END
  1282. ELSIF x.class = Ncall THEN err(55)
  1283. ELSE
  1284. pre := NIL; lastp := NIL;
  1285. DevCPB.PrepCall(x, fpar);
  1286. IF (x.obj # NIL) & (x.obj.mode = TProc) THEN DevCPB.CheckBuffering(x.left, NIL, x.obj.link, pre, lastp) END;
  1287. IF sym = lparen THEN
  1288. DevCPS.Get(sym); ActualParameters(apar, fpar, pre, lastp); CheckSym(rparen)
  1289. ELSE apar := NIL;
  1290. IF fpar # NIL THEN err(65) END
  1291. END ;
  1292. DevCPB.Call(x, apar, fpar);
  1293. IF x.typ # DevCPT.notyp THEN err(55) END;
  1294. IF pre # NIL THEN SetPos(x); DevCPB.Construct(Ncomp, pre, x); x := pre END;
  1295. IF level > 0 THEN DevCPT.topScope.link.leaf := FALSE END
  1296. END
  1297. ELSIF sym = if THEN
  1298. DevCPS.Get(sym); pos := DevCPM.startpos; Expression(x); CheckBool(x); CheckSym(then); StatSeq(y);
  1299. DevCPB.Construct(Nif, x, y); SetPos(x); lastif := x;
  1300. WHILE sym = elsif DO
  1301. DevCPS.Get(sym); pos := DevCPM.startpos; Expression(y); CheckBool(y); CheckSym(then); StatSeq(z);
  1302. DevCPB.Construct(Nif, y, z); SetPos(y); DevCPB.Link(x, lastif, y)
  1303. END ;
  1304. pos := DevCPM.startpos;
  1305. IF sym = else THEN DevCPS.Get(sym); StatSeq(y) ELSE y := NIL END ;
  1306. DevCPB.Construct(Nifelse, x, y); CheckSym(end); DevCPB.OptIf(x);
  1307. ELSIF sym = case THEN
  1308. DevCPS.Get(sym); pos := DevCPM.startpos; CasePart(x); CheckSym(end)
  1309. ELSIF sym = while THEN
  1310. DevCPS.Get(sym); pos := DevCPM.startpos; Expression(x); CheckBool(x); CheckSym(do); StatSeq(y);
  1311. DevCPB.Construct(Nwhile, x, y); CheckSym(end)
  1312. ELSIF sym = repeat THEN
  1313. DevCPS.Get(sym); StatSeq(x);
  1314. IF sym = until THEN DevCPS.Get(sym); pos := DevCPM.startpos; Expression(y); CheckBool(y)
  1315. ELSE err(43)
  1316. END ;
  1317. DevCPB.Construct(Nrepeat, x, y)
  1318. ELSIF sym = for THEN
  1319. DevCPS.Get(sym); pos := DevCPM.startpos;
  1320. IF sym = ident THEN qualident(id);
  1321. IF ~(id.typ.form IN intSet) THEN err(68) END ;
  1322. CheckSym(becomes); Expression(y);
  1323. x := DevCPB.NewLeaf(id); DevCPB.Assign(x, y); SetPos(x);
  1324. CheckSym(to); pos := DevCPM.startpos; Expression(y);
  1325. IF y.class # Nconst THEN
  1326. DevCPB.GetTempVar("@for", x.left.typ, t);
  1327. z := DevCPB.NewLeaf(t); DevCPB.Assign(z, y); SetPos(z); DevCPB.Link(stat, last, z);
  1328. y := DevCPB.NewLeaf(t)
  1329. ELSE
  1330. DevCPB.CheckAssign(x.left.typ, y)
  1331. END ;
  1332. DevCPB.Link(stat, last, x);
  1333. p := DevCPM.startpos;
  1334. IF sym = by THEN DevCPS.Get(sym); ConstExpression(z) ELSE z := DevCPB.NewIntConst(1) END ;
  1335. x := DevCPB.NewLeaf(id);
  1336. IF z.conval.intval > 0 THEN DevCPB.Op(leq, x, y)
  1337. ELSIF z.conval.intval < 0 THEN DevCPB.Op(geq, x, y)
  1338. ELSE err(63); DevCPB.Op(geq, x, y)
  1339. END ;
  1340. CheckSym(do); StatSeq(s);
  1341. y := DevCPB.NewLeaf(id); DevCPB.StPar1(y, z, incfn); pos := DevCPM.startpos; SetPos(y);
  1342. IF s = NIL THEN s := y
  1343. ELSE z := s;
  1344. WHILE z.link # NIL DO z := z.link END ;
  1345. z.link := y
  1346. END ;
  1347. CheckSym(end); DevCPB.Construct(Nwhile, x, s); pos := p
  1348. ELSE err(ident)
  1349. END
  1350. ELSIF sym = loop THEN
  1351. DevCPS.Get(sym); INC(LoopLevel); StatSeq(x); DEC(LoopLevel);
  1352. DevCPB.Construct(Nloop, x, NIL); CheckSym(end)
  1353. ELSIF sym = with THEN
  1354. DevCPS.Get(sym); idtyp := NIL; x := NIL;
  1355. LOOP
  1356. IF sym < bar THEN
  1357. pos := DevCPM.startpos;
  1358. IF sym = ident THEN
  1359. qualident(id); y := DevCPB.NewLeaf(id);
  1360. IF (id # NIL) & (id.typ.form = Pointer) & ((id.mode = VarPar) OR ~id.leaf) THEN
  1361. err(-302) (* warning 302 *)
  1362. END ;
  1363. CheckSym(colon);
  1364. IF sym = ident THEN qualident(t);
  1365. IF t.mode = Typ THEN
  1366. IF id # NIL THEN
  1367. idtyp := id.typ; DevCPB.TypTest(y, t, FALSE); id.typ := t.typ;
  1368. IF id.ptyp = NIL THEN id.ptyp := idtyp END
  1369. ELSE err(130)
  1370. END
  1371. ELSE err(52)
  1372. END
  1373. ELSE err(ident)
  1374. END
  1375. ELSE err(ident)
  1376. END ;
  1377. CheckSym(do); StatSeq(s); DevCPB.Construct(Nif, y, s); SetPos(y);
  1378. IF idtyp # NIL THEN
  1379. IF id.ptyp = idtyp THEN id.ptyp := NIL END;
  1380. id.typ := idtyp; idtyp := NIL
  1381. END ;
  1382. IF x = NIL THEN x := y; lastif := x ELSE DevCPB.Link(x, lastif, y) END
  1383. END;
  1384. IF sym = bar THEN DevCPS.Get(sym) ELSE EXIT END
  1385. END;
  1386. e := sym = else; pos := DevCPM.startpos;
  1387. IF e THEN DevCPS.Get(sym); StatSeq(s) ELSE s := NIL END ;
  1388. DevCPB.Construct(Nwith, x, s); CheckSym(end);
  1389. IF e THEN x.subcl := 1 END
  1390. ELSIF sym = exit THEN
  1391. DevCPS.Get(sym);
  1392. IF LoopLevel = 0 THEN err(46) END ;
  1393. DevCPB.Construct(Nexit, x, NIL)
  1394. ELSIF sym = return THEN DevCPS.Get(sym);
  1395. IF sym < semicolon THEN Expression(x) END ;
  1396. IF level > 0 THEN DevCPB.Return(x, DevCPT.topScope.link)
  1397. ELSE (* not standard Oberon *) DevCPB.Return(x, NIL)
  1398. END;
  1399. hasReturn := TRUE
  1400. END ;
  1401. IF x # NIL THEN SetPos(x); DevCPB.Link(stat, last, x) END ;
  1402. IF sym = semicolon THEN DevCPS.Get(sym)
  1403. ELSIF (sym <= ident) OR (if <= sym) & (sym <= return) THEN err(semicolon)
  1404. ELSE EXIT
  1405. END
  1406. END
  1407. END StatSeq;
  1408. PROCEDURE Block(VAR procdec, statseq: DevCPT.Node);
  1409. VAR typ: DevCPT.Struct;
  1410. obj, first, last, o: DevCPT.Object;
  1411. x, lastdec: DevCPT.Node;
  1412. i: SHORTINT;
  1413. rname: DevCPT.Name;
  1414. name: DevCPT.String;
  1415. rec: Elem;
  1416. BEGIN
  1417. IF ((sym < begin) OR (sym > var)) & (sym # procedure) & (sym # end) & (sym # close) THEN err(36) END;
  1418. first := NIL; last := NIL; userList := NIL; recList := NIL;
  1419. LOOP
  1420. IF sym = const THEN
  1421. DevCPS.Get(sym);
  1422. WHILE sym = ident DO
  1423. DevCPT.Insert(DevCPS.name, obj);
  1424. obj.mode := Con; CheckMark(obj);
  1425. obj.typ := DevCPT.int8typ; obj.mode := Var; (* Var to avoid recursive definition *)
  1426. IF sym = eql THEN
  1427. DevCPS.Get(sym); ConstExpression(x)
  1428. ELSIF sym = becomes THEN
  1429. err(eql); DevCPS.Get(sym); ConstExpression(x)
  1430. ELSE err(eql); x := DevCPB.NewIntConst(1)
  1431. END ;
  1432. obj.mode := Con; obj.typ := x.typ; obj.conval := x.conval; (* ConstDesc ist not copied *)
  1433. CheckSym(semicolon)
  1434. END
  1435. END ;
  1436. IF sym = type THEN
  1437. DevCPS.Get(sym);
  1438. WHILE sym = ident DO
  1439. DevCPT.Insert(DevCPS.name, obj); obj.mode := Typ; obj.typ := DevCPT.undftyp;
  1440. CheckMark(obj); obj.mode := -1;
  1441. IF sym # eql THEN err(eql) END;
  1442. IF (sym = eql) OR (sym = becomes) OR (sym = colon) THEN
  1443. DevCPS.Get(sym); Type(obj.typ, name); SetType(NIL, obj, obj.typ, name);
  1444. END;
  1445. obj.mode := Typ;
  1446. IF obj.typ.form IN {Byte..Set, Char16, Int64} THEN (* make alias structure *)
  1447. typ := DevCPT.NewStr(obj.typ.form, Basic); i := typ.ref;
  1448. typ^ := obj.typ^; typ.ref := i; typ.strobj := NIL; typ.mno := 0; typ.txtpos := DevCPM.errpos;
  1449. typ.BaseTyp := obj.typ; obj.typ := typ;
  1450. END;
  1451. IF obj.typ.strobj = NIL THEN obj.typ.strobj := obj END ;
  1452. IF obj.typ.form = Pointer THEN (* !!! *)
  1453. typ := obj.typ.BaseTyp;
  1454. IF (typ # NIL) & (typ.comp = Record) & (typ.strobj = NIL) THEN
  1455. (* pointer to unnamed record: name record as "pointerName^" *)
  1456. rname := obj.name^$; i := 0;
  1457. WHILE rname[i] # 0X DO INC(i) END;
  1458. rname[i] := "^"; rname[i+1] := 0X;
  1459. DevCPT.Insert(rname, o); o.mode := Typ; o.typ := typ; typ.strobj := o
  1460. END
  1461. END;
  1462. IF obj.vis # internal THEN
  1463. typ := obj.typ;
  1464. IF typ.form = Pointer THEN typ := typ.BaseTyp END;
  1465. IF typ.comp = Record THEN typ.exp := TRUE END
  1466. END;
  1467. CheckSym(semicolon)
  1468. END
  1469. END ;
  1470. IF sym = var THEN
  1471. DevCPS.Get(sym);
  1472. WHILE sym = ident DO
  1473. LOOP
  1474. IF sym = ident THEN
  1475. DevCPT.Insert(DevCPS.name, obj);
  1476. obj.mode := Var; obj.link := NIL; obj.leaf := obj.vis = internal; obj.typ := DevCPT.undftyp;
  1477. CheckMark(obj);
  1478. IF first = NIL THEN first := obj END ;
  1479. IF last = NIL THEN DevCPT.topScope.scope := obj ELSE last.link := obj END ;
  1480. last := obj
  1481. ELSE err(ident)
  1482. END ;
  1483. IF sym = comma THEN DevCPS.Get(sym)
  1484. ELSIF sym = ident THEN err(comma)
  1485. ELSE EXIT
  1486. END
  1487. END ;
  1488. CheckSym(colon); Type(typ, name);
  1489. CheckAlloc(typ, FALSE, DevCPM.errpos);
  1490. WHILE first # NIL DO SetType(NIL, first, typ, name); first := first.link END ;
  1491. CheckSym(semicolon)
  1492. END
  1493. END ;
  1494. IF (sym < const) OR (sym > var) THEN EXIT END ;
  1495. END ;
  1496. CheckForwardTypes;
  1497. userList := NIL; rec := recList; recList := NIL;
  1498. DevCPT.topScope.adr := DevCPM.errpos;
  1499. procdec := NIL; lastdec := NIL;
  1500. IF (sym # procedure) & (sym # begin) & (sym # end) & (sym # close) THEN err(37) END;
  1501. WHILE sym = procedure DO
  1502. DevCPS.Get(sym); ProcedureDeclaration(x);
  1503. IF x # NIL THEN
  1504. IF lastdec = NIL THEN procdec := x ELSE lastdec.link := x END ;
  1505. lastdec := x
  1506. END ;
  1507. CheckSym(semicolon)
  1508. END ;
  1509. IF DevCPM.noerr & ~(DevCPM.oberon IN DevCPM.options) THEN CheckRecords(rec) END;
  1510. hasReturn := FALSE;
  1511. IF (sym # begin) & (sym # end) & (sym # close) THEN err(38) END;
  1512. IF sym = begin THEN DevCPS.Get(sym); StatSeq(statseq)
  1513. ELSE statseq := NIL
  1514. END ;
  1515. IF (DevCPT.topScope.link # NIL) & (DevCPT.topScope.link.typ # DevCPT.notyp)
  1516. & ~hasReturn & (DevCPT.topScope.link.sysflag = 0) THEN err(133) END;
  1517. IF (level = 0) & (TDinit # NIL) THEN
  1518. lastTDinit.link := statseq; statseq := TDinit
  1519. END
  1520. END Block;
  1521. PROCEDURE Module*(VAR prog: DevCPT.Node);
  1522. VAR impName, aliasName: DevCPT.Name;
  1523. procdec, statseq: DevCPT.Node;
  1524. c, sf: INTEGER; done: BOOLEAN;
  1525. BEGIN
  1526. DevCPS.Init; LoopLevel := 0; level := 0; DevCPS.Get(sym);
  1527. IF sym = module THEN DevCPS.Get(sym) ELSE err(16) END ;
  1528. IF sym = ident THEN
  1529. DevCPT.Open(DevCPS.name); DevCPS.Get(sym);
  1530. DevCPT.libName := "";
  1531. IF sym = lbrak THEN
  1532. INCL(DevCPM.options, DevCPM.interface); DevCPS.Get(sym);
  1533. IF sym = eql THEN DevCPS.Get(sym)
  1534. ELSE INCL(DevCPM.options, DevCPM.noCode)
  1535. END;
  1536. IF sym = string THEN DevCPT.libName := DevCPS.str^$; DevCPS.Get(sym)
  1537. ELSE err(string)
  1538. END;
  1539. CheckSym(rbrak)
  1540. END;
  1541. CheckSym(semicolon);
  1542. IF sym = import THEN DevCPS.Get(sym);
  1543. LOOP
  1544. IF sym = ident THEN
  1545. aliasName := DevCPS.name$; impName := aliasName$; DevCPS.Get(sym);
  1546. IF sym = becomes THEN DevCPS.Get(sym);
  1547. IF sym = ident THEN impName := DevCPS.name$; DevCPS.Get(sym) ELSE err(ident) END
  1548. END ;
  1549. DevCPT.Import(aliasName, impName, done)
  1550. ELSE err(ident)
  1551. END ;
  1552. IF sym = comma THEN DevCPS.Get(sym)
  1553. ELSIF sym = ident THEN err(comma)
  1554. ELSE EXIT
  1555. END
  1556. END ;
  1557. CheckSym(semicolon)
  1558. END ;
  1559. IF DevCPM.noerr THEN TDinit := NIL; lastTDinit := NIL; c := DevCPM.errpos;
  1560. Block(procdec, statseq); DevCPB.Enter(procdec, statseq, NIL); prog := procdec;
  1561. prog.conval := DevCPT.NewConst(); prog.conval.intval := c; prog.conval.intval2 := DevCPM.startpos;
  1562. IF sym = close THEN DevCPS.Get(sym); StatSeq(prog.link) END;
  1563. prog.conval.realval := DevCPM.startpos;
  1564. CheckSym(end);
  1565. IF sym = ident THEN
  1566. IF DevCPS.name # DevCPT.SelfName THEN err(4) END ;
  1567. DevCPS.Get(sym)
  1568. ELSE err(ident)
  1569. END;
  1570. IF sym # period THEN err(period) END
  1571. END
  1572. ELSE err(ident)
  1573. END ;
  1574. TDinit := NIL; lastTDinit := NIL;
  1575. DevCPS.str := NIL
  1576. END Module;
  1577. END DevCPP.