FoxParser.Mod 84 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266
  1. MODULE FoxParser; (** AUTHOR "fof & fn"; PURPOSE "Oberon Compiler: Parser"; **)
  2. (* (c) fof ETH Zurich, 2009 *)
  3. IMPORT Basic := FoxBasic, Scanner := FoxScanner, D := Debugging, SyntaxTree := FoxSyntaxTree, Global := FoxGlobal, Diagnostics;
  4. CONST
  5. Trace = FALSE;
  6. CascadedWithSupport = TRUE;
  7. (** the parser reflects the following EBNF:
  8. Module = ('module' | 'cellnet' [Flags]) Identifier ['in' Identifier]';'
  9. [ImportList] DeclarationSequence [Body]
  10. 'end' Identifier '.'.
  11. ImportList = 'import' Import { ',' Import } ';'.
  12. Import = Identifier [':=' Identifier] ['in' Identifier].
  13. DeclarationSequence = {
  14. 'const' [ConstDeclaration] {';' [ConstDeclaration]}
  15. |'type' [TypeDeclaration] {';' [TypeDeclaration]}
  16. |'var' [VariableDeclaration] {';' [VariableDeclaration]}
  17. | ProcedureDeclaration
  18. | OperatorDeclaration
  19. | ';'
  20. }
  21. Declaration = IdentifierDefinition '=' Expression.
  22. TypeDeclaration = IdentifierDefinition '=' Type.
  23. VariableDeclaration = VariableNameList ':' Type.
  24. VariableNameList = IdentifierDefinition [Flags] [':=' Expression | 'extern' String] {',' IdentifierDefinition [Flags] [':=' Expression | 'extern' String] }.
  25. OperatorDeclaration = 'operator' [Flags] ['-'] String ['*'|'-'] FormalParameters ';'
  26. DeclarationSequence [Body] 'end' String.
  27. ProcedureDeclaration = 'procedure' ['^'|'&'|'~'|'-'|Flags ['-']] IdentifierDefinition [FormalParameters]';'
  28. DeclarationSequence [Body] 'end' Identifier.
  29. Flags = '{' [Identifier ['(' Expression ')'|'=' Expression] {',' Identifier ['(' Expression ')' | '=' Expression ] } ] '}'.
  30. FormalParameters = '(' [ParameterDeclaration {';' ParameterDeclaration}] ')' [':' [Flags] Type].
  31. ParameterDeclaration = ['var'|'const'] Identifier [Flags] ['= Expression] {',' Identifier [Flags] ['= Expression]}':' Type.
  32. PortList = '(' [PortDeclaration {';' PortDeclaration}] ')'.
  33. PortDeclaration = Identifier [Flags] {',' Identifier [Flags]}':' Type.
  34. Type = ArrayType | RecordType | PointerType | ObjectType | CellType | CellnetType | PortType
  35. | ProcedureType | EnumerationType | QualifiedIdentifier.
  36. PortType = 'port' ('in'|'out') ['(' Expression ')']
  37. EnumerationType = 'enum' ['('QualifiedIdentifier')'] IdentifierDefinition ['=' Expression]
  38. {',' IdentifierDefinition ['=' Expression]} 'end'.
  39. ArrayType = 'array' 'of' Type | 'array' Expression {',' Expression} 'of' Type
  40. | 'array' '[' MathArraySize {',' MathArraySize} ']' ['of' Type].
  41. MathArraySize = Expression | '*' | '?'.
  42. RecordType = 'record' [Flags] ['(' QualifiedIdentifier ')'] [VariableDeclaration {';' VariableDeclaration}] 'end'.
  43. PointerType = 'pointer' [Flags] 'to' Type.
  44. CellType = 'cell' [Flags] [PortList] [';'] DeclarationSequence [Body] 'end' [Identifier].
  45. ObjectType = 'object' | 'object' [Flags] ['(' (QualifiedIdentifier | ArrayType) ')'] DeclarationSequence [Body] 'end' [Identifier] .
  46. ProcedureType = 'procedure' [Flags] [FormalParameters].
  47. Body = 'begin' [Flags] StatementSequence ['finally' StatementSequence]
  48. | 'code' Code.
  49. Code = { any \ 'end' \ 'with' } ['with' {('in'|'out') StatementSequence}] .
  50. StatementBlock = [Flags] StatementSequence.
  51. StatementSequence = Statement {';' Statement}.
  52. Statement =
  53. [
  54. Designator [':=' Expression |'!' Expression | '?' Expression | '<<' Expresssion | '>>' Expression]
  55. | 'if' Expression 'then' StatementSequence
  56. {'elsif' Expression 'then' StatementSequence}
  57. ['else' StatementSequence]
  58. 'end'
  59. | 'with' Identifier ':' QualifiedIdentifier 'do' StatementSequence
  60. {'|' Identifier ':' QualifiedIdentifier 'do' StatementSequence}
  61. [else StatementSequence]
  62. 'end'
  63. | 'case' Expression 'of' ['|'] Case {'|' Case} ['else' StatementSequence] 'end'
  64. | 'while' Expression 'do' StatementSequence 'end'
  65. | 'repeat' StatementSequence 'until' Expression
  66. | 'for' Identifier ':=' Expression 'to' Expression ['by' Expression] 'do'
  67. StatementSequence 'end'
  68. | 'loop' StatementSequence 'end'
  69. | 'exit'
  70. | 'return' [Expression]
  71. | 'await' Expression
  72. | 'begin' StatementBlock 'end'
  73. | 'code' {any} 'end'
  74. ].
  75. Case = RangeExpression {',' RangeExpression} ':' StatementSequence.
  76. Expression = RangeExpression [RelationOp RangeExpression].
  77. RelationOp = '=' | '.=' | '#' | '.#'
  78. | '<' | '.<' | '<=' | '.<=' | '>' | '.>' | '>=' | '.>='
  79. | '??' | '!!' | '<<?' | '>>?'
  80. | 'in' | 'is'
  81. SimpleExpression = ['+'|'-'] Term {AddOp Term}.
  82. AddOp = '+' | '-' | 'or'.
  83. Term = Factor {MulOp Factor}.
  84. MulOp = '*' | '**' | '.*' | '+*' | '/' | '\' | './' | 'div' | 'mod' | '&'.
  85. Factor = Number | Character | String | 'nil' | 'imag' | 'true' | 'false' | Set
  86. | '(' Expression ')' | '~' Factor | Factor '`' | Designator | MathArray.
  87. | 'SIZE' 'OF' Designator | 'ADDRESS' 'OF' Designator
  88. MathArray = '[' Expression {',' Expression} ']'.
  89. Set = '{' [ RangeExpression {',' RangeExpression} ] '}'.
  90. Designator = ('self' | 'result' | Identifier)
  91. {'.' Identifier | '[' IndexList ']' | '(' [ExpressionList] ')' | '^'} [Flags].
  92. RangeExpression = SimpleExpression | [SimpleExpression] '..' [SimpleExpression] ['by' SimpleExpression] | '*'.
  93. IndexList = '?' [',' ExpressionList ] | ExpressionList [',' '?'].
  94. ExpressionList = Expression { ',' Expression }.
  95. IdentifierDefinition = Identifier [ '*' | '-' ].
  96. QualifiedIdentifier = Identifier ['.' Identifier].
  97. Identifier = Letter {Letter | Digit | '_'}.
  98. Letter = 'A' | 'B' | .. | 'Z' | 'a' | 'b' | .. | 'z'.
  99. String = '"' {Character} '"' | "'" {Character} "'".
  100. Number = Integer | Real.
  101. Integer = Digit {Digit} | '0' 'x' {HexDigit} | Digit {HexDigit} 'H' .
  102. Real = Digit {Digit} '.' {Digit} [ScaleFactor].
  103. ScaleFactor = ('E' | 'D') ['+' | '-'] digit {digit}.
  104. HexDigit = Digit | 'A' | 'B' | 'C' | 'D' | 'E' | 'F'
  105. Digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9' .
  106. **)
  107. TYPE
  108. Position*=Scanner.Position;
  109. Parser* = OBJECT
  110. VAR scanner-: Scanner.Scanner;
  111. symbol-: Scanner.Symbol;
  112. diagnostics: Diagnostics.Diagnostics;
  113. currentScope: SyntaxTree.Scope;
  114. recentCommentItem: ANY; recentLine: LONGINT;
  115. recentComment: SyntaxTree.Comment;
  116. moduleScope: SyntaxTree.ModuleScope;
  117. error-: BOOLEAN;
  118. Lax-: BOOLEAN;
  119. indent: LONGINT; (* for debugging purposes only *)
  120. hasSymbol: BOOLEAN;
  121. PROCEDURE S( CONST s: ARRAY OF CHAR ); (* for debugging purposes only *)
  122. VAR i: LONGINT;
  123. BEGIN
  124. D.Ln; INC( indent ); D.Int( indent,1 );
  125. FOR i := 1 TO indent DO D.Str( " " ); END;
  126. D.Str( "start: " ); D.Str( s ); D.Str( " at pos " ); D.Int( symbol.position.start,1 );
  127. END S;
  128. PROCEDURE E( CONST s: ARRAY OF CHAR ); (* for debugging purposes only *)
  129. VAR i: LONGINT;
  130. BEGIN
  131. D.Ln; D.Int( indent,1 );
  132. FOR i := 1 TO indent DO D.Str( " " ); END;
  133. D.Str( "end : " ); D.Str( s ); D.Str( " at pos " ); D.Int( symbol.position.start,1 );
  134. DEC(indent);
  135. END E;
  136. PROCEDURE EE( CONST s, t: ARRAY OF CHAR ); (* for debugging purposes only *)
  137. VAR i: LONGINT;
  138. BEGIN
  139. D.Ln; D.Int( indent,1 );
  140. FOR i := 1 TO indent DO D.Str( " " ); END;
  141. D.Str( "end : " ); D.Str( s ); D.Str( " (" ); D.Str( t ); D.Str( ") at pos " );
  142. DEC(indent);
  143. END EE;
  144. (** constructor, init parser with scanner providing input and with diagnostics for error output *)
  145. PROCEDURE & Init*( scanner: Scanner.Scanner; diagnostics: Diagnostics.Diagnostics );
  146. BEGIN
  147. SELF.scanner := scanner;
  148. SELF.diagnostics := diagnostics;
  149. error := ~scanner.GetNextSymbol(symbol);
  150. hasSymbol := TRUE;
  151. IF error THEN Basic.Error(diagnostics, scanner.source^, Basic.invalidPosition, "no input stream") END;
  152. recentCommentItem := NIL; recentComment := NIL;
  153. (* debugging *)
  154. indent := 0;
  155. Lax := FALSE;
  156. END Init;
  157. PROCEDURE Reset*;
  158. BEGIN
  159. error := FALSE;
  160. END Reset;
  161. PROCEDURE SetLax*;
  162. BEGIN
  163. Lax := TRUE;
  164. END SetLax;
  165. (** output error message and / or given code *)
  166. PROCEDURE Error(position: Position; code: LONGINT; CONST message: ARRAY OF CHAR);
  167. BEGIN
  168. Basic.ErrorC(diagnostics, scanner.source^, position, code, message);
  169. error := TRUE
  170. END Error;
  171. (** helper procedures interfacing to the scanner **)
  172. PROCEDURE SkipComments(b: BOOLEAN);
  173. VAR comment: SyntaxTree.Comment;
  174. BEGIN
  175. WHILE ~error & (b & (TokenB()= Scanner.Comment) OR ~b & (Token() = Scanner.Comment)) DO
  176. comment := SyntaxTree.NewComment(symbol.position, currentScope, symbol.string^,symbol.stringLength);
  177. IF moduleScope # NIL THEN
  178. moduleScope.AddComment(comment);
  179. END;
  180. IF recentComment = NIL THEN
  181. recentComment := comment;
  182. IF symbol.position.line = recentLine THEN
  183. IF recentCommentItem # NIL THEN
  184. IF (recentCommentItem IS SyntaxTree.Symbol) THEN
  185. IF recentCommentItem(SyntaxTree.Symbol).comment = NIL THEN
  186. recentCommentItem(SyntaxTree.Symbol).SetComment(comment)
  187. END;
  188. ELSIF (recentCommentItem IS SyntaxTree.Statement) THEN
  189. IF recentCommentItem(SyntaxTree.Statement).comment = NIL THEN
  190. recentCommentItem(SyntaxTree.Statement).SetComment(comment)
  191. END;
  192. ELSIF (recentCommentItem IS SyntaxTree.IfPart) THEN
  193. IF recentCommentItem(SyntaxTree.IfPart).comment = NIL THEN
  194. recentCommentItem(SyntaxTree.IfPart).SetComment(comment)
  195. END;
  196. ELSIF (recentCommentItem IS SyntaxTree.CasePart) THEN
  197. IF recentCommentItem(SyntaxTree.CasePart).comment = NIL THEN
  198. recentCommentItem(SyntaxTree.CasePart).SetComment(comment)
  199. END;
  200. ELSIF (recentCommentItem IS SyntaxTree.WithPart) THEN
  201. IF recentCommentItem(SyntaxTree.WithPart).comment = NIL THEN
  202. recentCommentItem(SyntaxTree.WithPart).SetComment(comment)
  203. END;
  204. END;
  205. comment.SetItem(recentCommentItem,TRUE);
  206. recentComment := NIL;
  207. recentCommentItem := NIL
  208. END;
  209. END;
  210. END;
  211. NextSymbol;
  212. (*error := ~scanner.GetNextSymbol(symbol);*)
  213. END;
  214. END SkipComments;
  215. (** Get next symbol from scanner and store it in object-local variable 'symbol' *)
  216. PROCEDURE NextSymbol*;
  217. VAR comment: SyntaxTree.Comment;
  218. BEGIN
  219. (*
  220. error := ~scanner.GetNextSymbol(symbol) OR error;
  221. hasSymbol := TRUE;
  222. SkipComments();
  223. *)
  224. hasSymbol := FALSE;
  225. END NextSymbol;
  226. PROCEDURE Token*(): LONGINT;
  227. BEGIN
  228. IF ~hasSymbol OR (symbol.token = Scanner.Escape) THEN
  229. error := ~scanner.GetNextSymbol(symbol) OR error;
  230. IF symbol.token = Scanner.Escape THEN
  231. error := ~scanner.GetNextSymbol(symbol) OR error;
  232. END;
  233. hasSymbol := TRUE;
  234. SkipComments(FALSE);
  235. END;
  236. RETURN symbol.token;
  237. END Token;
  238. (* stop on escape token *)
  239. PROCEDURE TokenB*(): LONGINT;
  240. BEGIN
  241. IF ~hasSymbol THEN
  242. error := ~scanner.GetNextSymbol(symbol) OR error;
  243. hasSymbol := TRUE;
  244. SkipComments(TRUE);
  245. END;
  246. RETURN symbol.token;
  247. END TokenB;
  248. (** Check if current symbol equals sym. If yes then return true, return false otherwise *)
  249. PROCEDURE PeekB*(token: Scanner.Token): BOOLEAN;
  250. VAR comment: SyntaxTree.Comment;
  251. BEGIN
  252. RETURN TokenB() = token
  253. END PeekB;
  254. (** Check if current symbol equals sym. If yes then return true, return false otherwise *)
  255. PROCEDURE Peek*(token: Scanner.Token): BOOLEAN;
  256. VAR comment: SyntaxTree.Comment;
  257. BEGIN
  258. SkipComments(FALSE);
  259. RETURN Token() = token
  260. END Peek;
  261. (** Check if the current symbol equals sym.If yes then read next symbol, report error otherwise. returns success value *)
  262. PROCEDURE Mandatory*( token: Scanner.Token): BOOLEAN;
  263. BEGIN
  264. ASSERT( token # Scanner.Identifier ); ASSERT( token # Scanner.String ); ASSERT( token # Scanner.Number ); (* because of NextSymbol ! *)
  265. IF ~Peek(token) THEN
  266. Error( symbol.position, token, "" );
  267. RETURN FALSE
  268. ELSE
  269. NextSymbol;
  270. RETURN TRUE
  271. END
  272. END Mandatory;
  273. (** Check if the current symbol equals sym. If yes then read next symbol, report error otherwise *)
  274. PROCEDURE Check( token: Scanner.Token );
  275. VAR b: BOOLEAN;
  276. BEGIN
  277. b := Mandatory( token );
  278. END Check;
  279. (** Check if current symbol is an identifier. If yes then copy identifier to name and get next symbol,
  280. report error otherwise and set name to empty name. returns success value *)
  281. PROCEDURE MandatoryIdentifier( VAR name: SyntaxTree.Identifier): BOOLEAN;
  282. BEGIN
  283. IF Peek(Scanner.Identifier) THEN
  284. name := symbol.identifier;
  285. NextSymbol;
  286. RETURN TRUE
  287. ELSE
  288. Error( symbol.position, Scanner.Identifier, "" );
  289. name := SyntaxTree.invalidIdentifier;
  290. RETURN FALSE
  291. END
  292. END MandatoryIdentifier;
  293. (** Expect an identifier (using MandatoryIdentifier) and return identifier object **)
  294. PROCEDURE Identifier(VAR position: Position): SyntaxTree.Identifier;
  295. VAR name: SyntaxTree.Identifier; identifier: SyntaxTree.Identifier;
  296. BEGIN
  297. position := symbol.position;
  298. IF MandatoryIdentifier(name) THEN
  299. identifier := name;
  300. ELSE
  301. identifier := SyntaxTree.invalidIdentifier;
  302. END;
  303. RETURN identifier
  304. END Identifier;
  305. (** Check if current symbol is a string (or string-like character). If yes then copy identifier to name and get next symbol,
  306. report error otherwise and set name to empty name. returns success value*)
  307. PROCEDURE MandatoryString*( VAR name: Scanner.StringType ): BOOLEAN;
  308. BEGIN
  309. IF Peek( Scanner.String) THEN
  310. name := symbol.string;
  311. NextSymbol;
  312. RETURN TRUE
  313. ELSIF Peek( Scanner.Character) THEN (* for compatibility with release: characters treated as strings *)
  314. name := symbol.string;
  315. NextSymbol;
  316. RETURN TRUE
  317. ELSE
  318. Error( symbol.position, Scanner.String, "" );
  319. NEW(name,1); name^ := "";
  320. RETURN FALSE
  321. END
  322. END MandatoryString;
  323. (** Check if current symbol is an identifier and if the name matches. If yes then get next symbol, report error otherwise. returns success value*)
  324. PROCEDURE ExpectThisIdentifier( name: SyntaxTree.Identifier ): BOOLEAN;
  325. VAR string: ARRAY 64 OF CHAR;
  326. BEGIN
  327. IF name = SyntaxTree.invalidIdentifier THEN (* nothing to be expected *)
  328. RETURN TRUE
  329. ELSIF (Token() # Scanner.Identifier) OR (symbol.identifier # name) THEN
  330. Basic.GetString(name,string);
  331. Error( symbol.position, Scanner.Identifier, string );
  332. RETURN FALSE
  333. ELSE
  334. NextSymbol;
  335. RETURN TRUE
  336. END
  337. END ExpectThisIdentifier;
  338. (** Check if current symbol is an identifier and if the name matches. If yes then get next symbol, report error otherwise. returns success value*)
  339. PROCEDURE ExpectThisString( CONST name: ARRAY OF CHAR ): BOOLEAN;
  340. BEGIN
  341. IF Peek(Scanner.String) & (symbol.string^ = name) THEN
  342. NextSymbol;
  343. RETURN TRUE
  344. ELSE
  345. Error( symbol.position, Scanner.String, name );
  346. RETURN FALSE
  347. END
  348. END ExpectThisString;
  349. (** Check if current symbol equals sym. If yes then get next symbol, return false otherwise *)
  350. PROCEDURE Optional*( token: Scanner.Token ): BOOLEAN;
  351. BEGIN
  352. (* do not use for Identifier, String or Number, if the result is needed ! *)
  353. IF Peek(token) THEN
  354. NextSymbol;
  355. RETURN TRUE
  356. ELSE
  357. RETURN FALSE
  358. END
  359. END Optional;
  360. PROCEDURE OptionalB*( token: Scanner.Token ): BOOLEAN;
  361. BEGIN
  362. (* do not use for Identifier, String or Number, if the result is needed ! *)
  363. IF PeekB(token) THEN
  364. NextSymbol;
  365. RETURN TRUE
  366. ELSE
  367. RETURN FALSE
  368. END
  369. END OptionalB;
  370. (* ignore one ore more symbols of type token *)
  371. PROCEDURE Ignore(token: Scanner.Token);
  372. BEGIN WHILE Optional(token) DO END;
  373. END Ignore;
  374. (** Parsing according to the EBNF **)
  375. (** QualifiedIdentifier = Identifier ['.' Identifier]. **)
  376. PROCEDURE QualifiedIdentifier*( ): SyntaxTree.QualifiedIdentifier;
  377. VAR prefix,suffix: SyntaxTree.Identifier; qualifiedIdentifier: SyntaxTree.QualifiedIdentifier; position0,position1: Position;
  378. BEGIN
  379. IF Trace THEN S( "QualifiedIdentifier" ) END;
  380. prefix := Identifier(position0);
  381. IF prefix # SyntaxTree.invalidIdentifier THEN
  382. IF ~Optional( Scanner.Period )THEN
  383. suffix := prefix; prefix := SyntaxTree.invalidIdentifier; (* empty *)
  384. ELSE
  385. suffix := Identifier(position1);
  386. END;
  387. qualifiedIdentifier := SyntaxTree.NewQualifiedIdentifier( position0, prefix,suffix);
  388. ELSE
  389. qualifiedIdentifier := SyntaxTree.invalidQualifiedIdentifier;
  390. END;
  391. IF Trace THEN E( "QualifiedIdentifier" ) END;
  392. RETURN qualifiedIdentifier
  393. END QualifiedIdentifier;
  394. (** IdentifierDefinition = Identifier [ '*' | '-' ]. **)
  395. PROCEDURE IdentifierDefinition( VAR name: SyntaxTree.Identifier; VAR access: SET; allowedReadOnly: BOOLEAN);
  396. VAR position: Position;
  397. BEGIN
  398. IF Trace THEN S( "IdentifierDefinition" ) END;
  399. name := Identifier(position);
  400. IF Optional( Scanner.Times ) THEN
  401. access := SyntaxTree.Public + SyntaxTree.Protected + SyntaxTree.Internal;
  402. ELSIF Optional( Scanner.Minus ) THEN
  403. IF ~allowedReadOnly THEN
  404. Error( symbol.position, Diagnostics.Invalid, "may not be defined read-only" )
  405. ELSE
  406. access := SyntaxTree.ReadOnly + {SyntaxTree.InternalWrite};
  407. END;
  408. ELSE
  409. access := SyntaxTree.Internal;
  410. END;
  411. IF Trace THEN E( "IdentifierDefinition") END;
  412. END IdentifierDefinition;
  413. (** ExpressionList = Expression { ',' Expression }. **)
  414. PROCEDURE ExpressionList( expressionList: SyntaxTree.ExpressionList );
  415. VAR expression: SyntaxTree.Expression;
  416. BEGIN
  417. IF Trace THEN S( "ExpressionList" ) END;
  418. REPEAT
  419. expression := Expression();
  420. expressionList.AddExpression( expression )
  421. UNTIL ~Optional( Scanner.Comma );
  422. IF Trace THEN E( "ExpressionList" ) END;
  423. END ExpressionList;
  424. (** IndexList = '?' [',' ExpressionList ] | ExpressionList [',' '?']. **)
  425. PROCEDURE IndexList(expressionList: SyntaxTree.ExpressionList);
  426. VAR
  427. position: Position;
  428. done: BOOLEAN;
  429. BEGIN
  430. IF Trace THEN S( "IndexList" ) END;
  431. position := symbol.position;
  432. IF Optional(Scanner.Questionmark) THEN
  433. expressionList.AddExpression(SyntaxTree.NewTensorRangeExpression(position));
  434. IF Optional(Scanner.Comma) THEN
  435. ExpressionList(expressionList);
  436. END
  437. ELSE
  438. expressionList.AddExpression(Expression());
  439. done := FALSE;
  440. WHILE ~done DO
  441. IF Optional(Scanner.Comma) THEN
  442. IF Optional(Scanner.Questionmark) THEN
  443. expressionList.AddExpression(SyntaxTree.NewTensorRangeExpression(position));
  444. done := TRUE;
  445. ELSE
  446. expressionList.AddExpression(Expression())
  447. END
  448. ELSE
  449. done := TRUE
  450. END
  451. END
  452. END;
  453. IF Trace THEN E( "IndexList" ) END;
  454. END IndexList;
  455. (** RangeExpression = SimpleExpression | [SimpleExpression] '..' [SimpleExpression] ['by' SimpleExpression] | '*'.
  456. i.e., a RangeExpression can have one of the following forms:
  457. '*'
  458. '..' [delimiter]
  459. '..' 'by' SimpleExpression
  460. '..' SimpleExpression
  461. '..' SimpleExpression 'by' SimpleExpression
  462. SimpleExpression
  463. SimpleExpression '..' [delimiter]
  464. SimpleExpression '..' 'by' SimpleExpression
  465. SimpleExpression '..' SimpleExpression
  466. SimpleExpression '..' SimpleExpression 'by' SimpleExpression
  467. a RangeExpression is always delimited by any of the following tokens: ",", ";", ":", "]", ")", "}", "=", "#", "END".
  468. **)
  469. PROCEDURE RangeExpression(): SyntaxTree.Expression;
  470. VAR
  471. expression, first, last, step: SyntaxTree.Expression;
  472. position: Position;
  473. PROCEDURE HasDelimiter(): BOOLEAN;
  474. BEGIN RETURN
  475. Peek(Scanner.Comma) OR Peek(Scanner.Semicolon) OR Peek(Scanner.Colon) OR
  476. Peek(Scanner.RightBracket) OR Peek(Scanner.RightParenthesis) OR Peek(Scanner.RightBrace) OR
  477. Peek(Scanner.Equal) OR Peek(Scanner.Unequal) OR Peek(Scanner.End)
  478. END HasDelimiter;
  479. BEGIN
  480. IF Trace THEN S( "RangeExpression" ) END;
  481. position := symbol.position;
  482. IF Optional(Scanner.Times) THEN
  483. expression := SyntaxTree.NewRangeExpression(position, NIL, NIL, NIL)
  484. ELSIF Optional(Scanner.Upto) THEN
  485. (* is range expression *)
  486. first := NIL;
  487. IF HasDelimiter() THEN
  488. last := NIL;
  489. step := NIL
  490. ELSIF Optional(Scanner.By) THEN
  491. last := NIL;
  492. step := SimpleExpression()
  493. ELSE
  494. last := SimpleExpression();
  495. IF Optional(Scanner.By) THEN
  496. step := SimpleExpression()
  497. ELSE
  498. step := NIL
  499. END
  500. END;
  501. expression := SyntaxTree.NewRangeExpression(position, first, last, step)
  502. ELSE
  503. expression := SimpleExpression();
  504. IF OptionalB(Scanner.Upto) THEN
  505. (* is range expression *)
  506. first := expression;
  507. IF HasDelimiter() THEN
  508. last := NIL;
  509. step := NIL
  510. ELSIF Optional(Scanner.By) THEN
  511. last := NIL;
  512. step := SimpleExpression()
  513. ELSE
  514. last := SimpleExpression();
  515. IF Optional(Scanner.By) THEN
  516. step := SimpleExpression()
  517. ELSE
  518. step := NIL
  519. END
  520. END;
  521. expression := SyntaxTree.NewRangeExpression(position, first, last, step)
  522. END;
  523. END;
  524. IF Trace THEN E( "RangeExpression" ) END;
  525. RETURN expression
  526. END RangeExpression;
  527. (** Designator = ('self' | 'result' | Identifier)
  528. {'.' Identifier | '[' IndexList ']' | '(' [ExpressionList] ')' | '^'} [Flags].
  529. **)
  530. PROCEDURE Designator( ): SyntaxTree.Designator;
  531. VAR
  532. designator: SyntaxTree.Designator; expressionList: SyntaxTree.ExpressionList;
  533. identifier: SyntaxTree.Identifier; position: Position;
  534. qualifiedIdentifier: SyntaxTree.QualifiedIdentifier;
  535. qualifiedType : SyntaxTree.QualifiedType;
  536. BEGIN
  537. IF Trace THEN S( "Designator" ) END;
  538. position := symbol.position;
  539. IF Optional(Scanner.Self) THEN
  540. designator := SyntaxTree.NewSelfDesignator(position);
  541. ELSIF Optional(Scanner.Result) THEN
  542. designator := SyntaxTree.NewResultDesignator(position);
  543. ELSIF (Token() = Scanner.Address) OR (Token()=Scanner.Size) OR (Token() = Scanner.Alias) THEN
  544. identifier := symbol.identifier;
  545. designator := SyntaxTree.NewIdentifierDesignator(position,identifier);
  546. NextSymbol;
  547. ELSIF (Token() = Scanner.New) THEN
  548. identifier := symbol.identifier;
  549. designator := SyntaxTree.NewIdentifierDesignator(position,identifier);
  550. NextSymbol;
  551. IF Token() # Scanner.LeftParenthesis THEN (* NEW Type () *)
  552. qualifiedIdentifier := QualifiedIdentifier();
  553. qualifiedType := SyntaxTree.NewQualifiedType(qualifiedIdentifier.position, currentScope, qualifiedIdentifier);
  554. IF Mandatory( Scanner.LeftParenthesis ) THEN
  555. expressionList := SyntaxTree.NewExpressionList();
  556. IF ~Optional(Scanner.RightParenthesis) THEN
  557. ExpressionList( expressionList );
  558. Check( Scanner.RightParenthesis )
  559. END;
  560. END;
  561. designator := SyntaxTree.NewBuiltinCallDesignator(position,Global.New, NIL, expressionList);
  562. designator(SyntaxTree.BuiltinCallDesignator).SetReturnType(qualifiedType);
  563. (* special case: NEW Type() *)
  564. END;
  565. ELSE
  566. identifier := Identifier(position);
  567. designator := SyntaxTree.NewIdentifierDesignator(position,identifier);
  568. END;
  569. LOOP
  570. position := symbol.position;
  571. IF OptionalB( Scanner.LeftParenthesis ) THEN
  572. expressionList := SyntaxTree.NewExpressionList();
  573. IF ~Optional( Scanner.RightParenthesis ) THEN
  574. ExpressionList( expressionList );
  575. Check( Scanner.RightParenthesis )
  576. END;
  577. designator := SyntaxTree.NewParameterDesignator( position,designator,expressionList);
  578. ELSIF OptionalB( Scanner.Period ) THEN
  579. IF ~Optional(Scanner.Identifier) THEN (* make sure symbol is read *) END;
  580. position := symbol.position;
  581. CASE symbol.identifierString[0] OF
  582. "a".."z", "A" .. "Z":
  583. (*IF Peek(Scanner.Size) (* special rule: support for SYSTEM.SIZE *) THEN*)
  584. identifier := symbol.identifier;
  585. NextSymbol;
  586. ELSE
  587. identifier := Identifier(position);
  588. END;
  589. designator := SyntaxTree.NewSelectorDesignator(position,designator,identifier);
  590. ELSIF OptionalB( Scanner.LeftBracket ) THEN
  591. expressionList := SyntaxTree.NewExpressionList();
  592. IndexList( expressionList );
  593. Check( Scanner.RightBracket );
  594. designator:= SyntaxTree.NewBracketDesignator( position,designator,expressionList );
  595. ELSIF OptionalB( Scanner.Arrow ) THEN
  596. designator:= SyntaxTree.NewArrowDesignator( position,designator );
  597. ELSE EXIT
  598. END;
  599. END;
  600. IF OptionalB(Scanner.LeftBrace) THEN
  601. designator.SetModifiers(Flags());
  602. END;
  603. (*IF OptionalB(Scanner.Escape) THEN END; (* skip breaking signal *)*)
  604. IF Trace THEN E( "Designator" ) END;
  605. RETURN designator
  606. END Designator;
  607. (** Set = '{' [ RangeExpression {',' RangeExpression} ] '}'. **)
  608. PROCEDURE Set( ): SyntaxTree.Expression;
  609. VAR
  610. set: SyntaxTree.Set;
  611. BEGIN
  612. IF Trace THEN S( "Set" ) END;
  613. set := SyntaxTree.NewSet(symbol.position);
  614. Check(Scanner.LeftBrace);
  615. IF ~Optional(Scanner.RightBrace) THEN
  616. REPEAT
  617. set.elements.AddExpression(RangeExpression())
  618. UNTIL ~Optional(Scanner.Comma);
  619. Check(Scanner.RightBrace);
  620. END;
  621. set.End(symbol.position.end);
  622. IF Trace THEN E( "Set" ) END;
  623. RETURN set
  624. END Set;
  625. (* MathArray = '[' Expression {',' Expression} ']'. *)
  626. PROCEDURE MathArray(): SyntaxTree.Expression;
  627. VAR array: SyntaxTree.MathArrayExpression; element: SyntaxTree.Expression;
  628. BEGIN
  629. array := SyntaxTree.NewMathArrayExpression(symbol.position);
  630. IF ~Optional(Scanner.RightBracket) THEN
  631. REPEAT
  632. element := Expression();
  633. array.elements.AddExpression(element);
  634. UNTIL ~Optional(Scanner.Comma);
  635. Check(Scanner.RightBracket);
  636. END;
  637. RETURN array
  638. END MathArray;
  639. (** Factor = Number | Character | String | 'nil' | 'imag' | 'true' | 'false' | Set
  640. | '(' Expression ')' | '~' Factor | Factor '`' | Designator | MathArray.
  641. | 'SIZE' 'OF' Designator | 'ADDRESS' 'OF' Designator
  642. **)
  643. PROCEDURE Factor( ): SyntaxTree.Expression;
  644. VAR factor: SyntaxTree.Expression; position: Position; operator: LONGINT;
  645. BEGIN
  646. IF Trace THEN S( "Factor" ) END;
  647. position := symbol.position;
  648. CASE Token() OF
  649. | Scanner.Number:
  650. IF (symbol.numberType = Scanner.Integer) THEN
  651. factor := SyntaxTree.NewIntegerValue( position, symbol.integer);
  652. ELSIF (symbol.numberType = Scanner.Hugeint) THEN
  653. factor := SyntaxTree.NewIntegerValue(position, symbol.hugeint);
  654. ELSIF (symbol.numberType = Scanner.Real) OR (symbol.numberType = Scanner.Longreal) THEN
  655. factor := SyntaxTree.NewRealValue( position, symbol.real);
  656. factor(SyntaxTree.RealValue).SetSubtype(symbol.numberType);
  657. ELSE HALT( 100 )
  658. END;
  659. NextSymbol;
  660. | Scanner.Character:
  661. factor := SyntaxTree.NewCharacterValue(position,symbol.character);
  662. NextSymbol;
  663. | Scanner.String:
  664. factor := SyntaxTree.NewStringValue( position, symbol.string );
  665. NextSymbol;
  666. WHILE (Token() = Scanner.String) OR (Token() = Scanner.Character) DO
  667. IF Token() = Scanner.Character THEN
  668. factor(SyntaxTree.StringValue).AppendChar(symbol.character);
  669. ELSE
  670. factor(SyntaxTree.StringValue).Append(symbol.string);
  671. END;
  672. factor.End(symbol.position.end);
  673. NextSymbol;
  674. END;
  675. | Scanner.Nil:
  676. factor := SyntaxTree.NewNilValue( position );
  677. NextSymbol;
  678. | Scanner.Imag:
  679. factor := SyntaxTree.NewComplexValue(position, 0, 1);
  680. factor(SyntaxTree.ComplexValue).SetSubtype(Scanner.Real);
  681. NextSymbol;
  682. | Scanner.True:
  683. factor := SyntaxTree.NewBooleanValue( position, TRUE );
  684. NextSymbol;
  685. | Scanner.False:
  686. factor := SyntaxTree.NewBooleanValue( position, FALSE );
  687. NextSymbol;
  688. | Scanner.LeftBrace:
  689. factor := Set();
  690. | Scanner.LeftParenthesis:
  691. NextSymbol;
  692. factor := Expression();
  693. Check( Scanner.RightParenthesis );
  694. factor.End( symbol.position.end );
  695. | Scanner.Not:
  696. NextSymbol;
  697. factor := Factor();
  698. factor := SyntaxTree.NewUnaryExpression( position, factor, Scanner.Not );
  699. factor.End( symbol.position.end );
  700. | Scanner.Address, Scanner.Size, Scanner.Alias:
  701. operator := Token();
  702. factor := Designator();
  703. IF Optional(Scanner.Of) THEN
  704. factor := Designator();
  705. factor := SyntaxTree.NewUnaryExpression( position, factor, operator );
  706. END;
  707. factor.End (symbol.position.end)
  708. | Scanner.Self, Scanner.Result, Scanner.Identifier, Scanner.New:
  709. factor := Designator();
  710. factor.End( symbol.position.end );
  711. | Scanner.LeftBracket:
  712. NextSymbol;
  713. factor := MathArray();
  714. factor.End(symbol.position.end);
  715. ELSE
  716. Error( position, Basic.ValueStartIncorrectSymbol, "" );
  717. NextSymbol; factor := SyntaxTree.invalidExpression;
  718. END;
  719. (* suffix *)
  720. IF OptionalB(Scanner.Transpose) THEN
  721. IF (factor IS SyntaxTree.UnaryExpression) & (factor(SyntaxTree.UnaryExpression).operator = Scanner.Transpose) THEN
  722. (* transpose operator has higher precedence than not, reevaluate expression: *)
  723. factor := factor(SyntaxTree.UnaryExpression).left;
  724. factor := SyntaxTree.NewUnaryExpression(position,factor,Scanner.Transpose);
  725. factor := SyntaxTree.NewUnaryExpression(position,factor,Scanner.Not);
  726. ELSE
  727. factor := SyntaxTree.NewUnaryExpression(position,factor,Scanner.Transpose);
  728. END;
  729. END;
  730. IF Trace THEN E( "Factor" ) END;
  731. RETURN factor
  732. END Factor;
  733. (** Term = Factor {MulOp Factor}.
  734. MulOp = '*' | '**' | '.*' | '+*' | '/' | '\' | './' | 'div' | 'mod' | '&'.
  735. **)
  736. PROCEDURE Term( ): SyntaxTree.Expression;
  737. VAR term, factor: SyntaxTree.Expression; operator: LONGINT; position: Position;
  738. BEGIN
  739. IF Trace THEN S( "Term" ) END;
  740. position := symbol.position;
  741. term := Factor();
  742. WHILE (TokenB() >= Scanner.Times) & (TokenB() <= Scanner.And) DO
  743. operator := Token();
  744. NextSymbol;
  745. factor := Factor();
  746. term := SyntaxTree.NewBinaryExpression( position, term, factor, operator );
  747. END;
  748. term.End( symbol.position.end );
  749. IF Trace THEN E( "Term" ) END;
  750. RETURN term
  751. END Term;
  752. (** SimpleExpression = ['+'|'-'] Term {AddOp Term}.
  753. AddOp = '+' | '-' | 'or'.
  754. **)
  755. PROCEDURE SimpleExpression( ): SyntaxTree.Expression;
  756. VAR operator: LONGINT; term, expression: SyntaxTree.Expression; position: Position;
  757. BEGIN
  758. IF Trace THEN S( "SimpleExpression" ) END;
  759. position := symbol.position;
  760. IF Peek(Scanner.Plus) OR Peek(Scanner.Minus) THEN (* sign should be part of the factor *)
  761. operator := Token();
  762. NextSymbol;
  763. term := Term();
  764. expression := SyntaxTree.NewUnaryExpression( position, term, operator );
  765. ELSE expression := Term();
  766. END;
  767. WHILE (TokenB() >= Scanner.Or) & (TokenB() <= Scanner.Minus) DO
  768. operator := Token();
  769. NextSymbol;
  770. term := Term();
  771. expression := SyntaxTree.NewBinaryExpression( position, expression, term, operator );
  772. END;
  773. IF Trace THEN E( "SimpleExpression" ) END;
  774. RETURN expression
  775. END SimpleExpression;
  776. (**
  777. Expression = RangeExpression [RelationOp RangeExpression].
  778. RelationOp = '=' | '.=' | '#' | '.#'
  779. | '<' | '.<' | '<=' | '.<=' | '>' | '.>' | '>=' | '.>='
  780. | '??' | '!!' | '<<?' | '>>?'
  781. | 'in' | 'is'
  782. **)
  783. PROCEDURE Expression*( ): SyntaxTree.Expression;
  784. VAR expression, rightExpression: SyntaxTree.Expression; operator: LONGINT; position: Position;
  785. BEGIN
  786. IF Trace THEN S( "Expression" ) END;
  787. position := symbol.position;
  788. expression := RangeExpression();
  789. IF (TokenB() >= Scanner.Equal) & (TokenB() <= Scanner.Is) THEN
  790. operator := Token();
  791. NextSymbol;
  792. rightExpression := RangeExpression();
  793. expression := SyntaxTree.NewBinaryExpression(position, expression, rightExpression, operator );
  794. END;
  795. (*IF OptionalB(Scanner.Escape) THEN END; (* skip breaking escape *)*)
  796. IF Trace THEN E( "Expression" ) END;
  797. RETURN expression
  798. END Expression;
  799. (** Case = RangeExpression {',' RangeExpression} ':' StatementSequence. **)
  800. PROCEDURE Case( caseStatement: SyntaxTree.CaseStatement );
  801. VAR
  802. casePart: SyntaxTree.CasePart;
  803. statements: SyntaxTree.StatementSequence;
  804. element: SyntaxTree.Expression;
  805. BEGIN
  806. IF Trace THEN S( "Case" ) END;
  807. casePart := SyntaxTree.NewCasePart();
  808. CommentCasePart(casePart);
  809. REPEAT
  810. element := RangeExpression();
  811. casePart.elements.AddExpression( element );
  812. UNTIL ~Optional( Scanner.Comma );
  813. Check( Scanner.Colon );
  814. statements := StatementSequence(caseStatement);
  815. casePart.SetStatements( statements );
  816. caseStatement.AddCasePart( casePart );
  817. IF Trace THEN E( "Case" ) END;
  818. END Case;
  819. (** Statement =
  820. [
  821. Designator [':=' Expression |'!' Expression | '?' Expression | '<<' Expresssion | '>>' Expression]
  822. | 'if' Expression 'then' StatementSequence
  823. {'elsif' Expression 'then' StatementSequence}
  824. ['else' StatementSequence]
  825. 'end'
  826. | 'with' Identifier ':' QualifiedIdentifier 'do' StatementSequence
  827. {'|' Identifier ':' QualifiedIdentifier 'do' StatementSequence}
  828. [else StatementSequence]
  829. 'end'
  830. | 'case' Expression 'of' ['|'] Case {'|' Case} ['else' StatementSequence] 'end'
  831. | 'while' Expression 'do' StatementSequence 'end'
  832. | 'repeat' StatementSequence 'until' Expression
  833. | 'for' Identifier ':=' Expression 'to' Expression ['by' Expression] 'do'
  834. StatementSequence 'end'
  835. | 'loop' StatementSequence 'end'
  836. | 'exit'
  837. | 'return' [Expression]
  838. | 'await' Expression
  839. | 'begin' StatementBlock 'end'
  840. | 'code' {any} 'end'
  841. ].
  842. **)
  843. PROCEDURE Statement*( statements: SyntaxTree.StatementSequence; outer: SyntaxTree.Statement): BOOLEAN;
  844. VAR qualifiedIdentifier: SyntaxTree.QualifiedIdentifier; expression: SyntaxTree.Expression; designator: SyntaxTree.Designator; statement: SyntaxTree.Statement;
  845. ifStatement: SyntaxTree.IfStatement; elsifPart: SyntaxTree.IfPart; statementSequence: SyntaxTree.StatementSequence; withStatement: SyntaxTree.WithStatement;
  846. withPart: SyntaxTree.WithPart; caller: SyntaxTree.ProcedureCallStatement;
  847. caseStatement: SyntaxTree.CaseStatement; whileStatement: SyntaxTree.WhileStatement; repeatStatement: SyntaxTree.RepeatStatement; forStatement: SyntaxTree.ForStatement;
  848. identifier: SyntaxTree.Identifier; loopStatement: SyntaxTree.LoopStatement; returnStatement: SyntaxTree.ReturnStatement; awaitStatement: SyntaxTree.AwaitStatement;
  849. qualifiedType: SyntaxTree.QualifiedType; code : SyntaxTree.Code; position: Position; result: BOOLEAN;
  850. commToken: Scanner.Token;
  851. BEGIN
  852. IF Trace THEN S( "Statement" ) END;
  853. CASE Token() OF
  854. | Scanner.Identifier, Scanner.Self, Scanner.Result, Scanner.New:
  855. designator := Designator();
  856. position := symbol.position;
  857. IF OptionalB( Scanner.Becomes ) THEN
  858. expression := Expression();
  859. statement := SyntaxTree.NewAssignment( position, designator, expression,outer );
  860. CommentStatement(statement);
  861. ELSIF PeekB(Scanner.ExclamationMark) OR PeekB(Scanner.Questionmark) OR PeekB(Scanner.LessLess) OR PeekB(Scanner.GreaterGreater) THEN
  862. commToken := Token();
  863. NextSymbol;
  864. expression := Expression();
  865. statement := SyntaxTree.NewCommunicationStatement(position, commToken, designator, expression, outer);
  866. CommentStatement(statement);
  867. ELSE
  868. caller := SyntaxTree.NewProcedureCallStatement(designator.position, designator,outer);
  869. statement := caller;
  870. CommentStatement(statement);
  871. END;
  872. statements.AddStatement( statement );
  873. (*IF OptionalB(Scanner.Escape) THEN END;*)
  874. result := TRUE
  875. | Scanner.If:
  876. NextSymbol;
  877. ifStatement := SyntaxTree.NewIfStatement( symbol.position ,outer);
  878. CommentStatement(ifStatement);
  879. expression := Expression();
  880. ifStatement.ifPart.SetCondition( expression );
  881. Check( Scanner.Then );
  882. statementSequence := StatementSequence(ifStatement);
  883. ifStatement.ifPart.SetStatements( statementSequence );
  884. WHILE Optional( Scanner.Elsif ) DO
  885. elsifPart := SyntaxTree.NewIfPart();
  886. CommentIfPart(elsifPart);
  887. ifStatement.AddElsifPart( elsifPart);
  888. expression := Expression();
  889. elsifPart.SetCondition( expression );
  890. Check( Scanner.Then );
  891. statementSequence := StatementSequence(ifStatement);
  892. elsifPart.SetStatements( statementSequence );
  893. END;
  894. IF Optional( Scanner.Else ) THEN
  895. statementSequence := StatementSequence(ifStatement);
  896. ifStatement.SetElsePart( statementSequence );
  897. END;
  898. Check( Scanner.End ); statements.AddStatement( ifStatement );
  899. result := TRUE
  900. | Scanner.With:
  901. withStatement := SyntaxTree.NewWithStatement( symbol.position ,outer);
  902. CommentStatement(withStatement);
  903. NextSymbol;
  904. REPEAT
  905. identifier := Identifier(position);
  906. IF Optional(Scanner.Period) & Optional(Scanner.Identifier) THEN
  907. Error(position,Diagnostics.Invalid,"forbidden qualified identifier in with statement");
  908. END;
  909. withPart := SyntaxTree.NewWithPart();
  910. CommentWithPart(withPart);
  911. withStatement.AddWithPart(withPart);
  912. designator := SyntaxTree.NewIdentifierDesignator(position,identifier);
  913. withPart.SetVariable( designator );
  914. Check( Scanner.Colon );
  915. qualifiedIdentifier := QualifiedIdentifier();
  916. qualifiedType := SyntaxTree.NewQualifiedType(qualifiedIdentifier.position, currentScope, qualifiedIdentifier);
  917. withPart.SetType(qualifiedType);
  918. Check( Scanner.Do );
  919. statementSequence := StatementSequence(withStatement);
  920. withPart.SetStatements( statementSequence );
  921. UNTIL ~Optional(Scanner.Bar) OR ~CascadedWithSupport;
  922. IF CascadedWithSupport & Optional(Scanner.Else) THEN
  923. statementSequence := StatementSequence(withStatement);
  924. withStatement.SetElsePart(statementSequence);
  925. END;
  926. Check( Scanner.End );
  927. statements.AddStatement( withStatement );
  928. result := TRUE
  929. | Scanner.Case:
  930. caseStatement := SyntaxTree.NewCaseStatement( symbol.position,outer );
  931. CommentStatement(caseStatement);
  932. NextSymbol;
  933. expression := Expression();
  934. Check( Scanner.Of );
  935. caseStatement.SetVariable( expression );
  936. IF Optional(Scanner.Bar) THEN END;
  937. REPEAT
  938. Case(caseStatement)
  939. UNTIL ~Optional(Scanner.Bar);
  940. IF Optional( Scanner.Else ) THEN
  941. statementSequence := StatementSequence(caseStatement);
  942. caseStatement.SetElsePart( statementSequence );
  943. END;
  944. Check( Scanner.End );
  945. statements.AddStatement( caseStatement );
  946. result := TRUE
  947. | Scanner.While:
  948. NextSymbol;
  949. whileStatement := SyntaxTree.NewWhileStatement( symbol.position, outer );
  950. CommentStatement(whileStatement);
  951. expression := Expression();
  952. Check( Scanner.Do );
  953. whileStatement.SetCondition( expression );
  954. statementSequence := StatementSequence(whileStatement);
  955. whileStatement.SetStatements( statementSequence );
  956. Check( Scanner.End );
  957. statements.AddStatement( whileStatement );
  958. result := TRUE
  959. | Scanner.Repeat:
  960. NextSymbol;
  961. repeatStatement := SyntaxTree.NewRepeatStatement( symbol.position, outer );
  962. CommentStatement(repeatStatement);
  963. statementSequence := StatementSequence(repeatStatement);
  964. repeatStatement.SetStatements( statementSequence );
  965. Check( Scanner.Until );
  966. expression := Expression();
  967. repeatStatement.SetCondition( expression );
  968. statements.AddStatement( repeatStatement );
  969. result := TRUE
  970. | Scanner.For:
  971. NextSymbol;
  972. forStatement := SyntaxTree.NewForStatement( symbol.position, outer);
  973. CommentStatement(forStatement);
  974. identifier := Identifier(position);
  975. IF Optional(Scanner.Period) & Optional(Scanner.Identifier) THEN
  976. Error(position,Diagnostics.Invalid,"forbidden non-local counter variable");
  977. END;
  978. designator := SyntaxTree.NewIdentifierDesignator(position,identifier);
  979. forStatement.SetVariable( designator );
  980. Check( Scanner.Becomes );
  981. expression := Expression();
  982. forStatement.SetFrom( expression );
  983. Check( Scanner.To );
  984. expression := Expression();
  985. forStatement.SetTo( expression );
  986. IF Optional( Scanner.By ) THEN
  987. expression := Expression();
  988. forStatement.SetBy( expression );
  989. END;
  990. Check( Scanner.Do );
  991. statementSequence := StatementSequence(forStatement);
  992. forStatement.SetStatements( statementSequence );
  993. Check( Scanner.End );
  994. statements.AddStatement( forStatement );
  995. result := TRUE
  996. | Scanner.Loop:
  997. NextSymbol;
  998. loopStatement := SyntaxTree.NewLoopStatement( symbol.position ,outer);
  999. CommentStatement(loopStatement);
  1000. statementSequence := StatementSequence(loopStatement);
  1001. loopStatement.SetStatements( statementSequence );
  1002. Check( Scanner.End );
  1003. statements.AddStatement( loopStatement );
  1004. result := TRUE;
  1005. | Scanner.Exit:
  1006. NextSymbol;
  1007. statement := SyntaxTree.NewExitStatement( symbol.position, outer);
  1008. CommentStatement(statement);
  1009. statements.AddStatement( statement );
  1010. result := TRUE;
  1011. | Scanner.Return:
  1012. NextSymbol;
  1013. returnStatement := SyntaxTree.NewReturnStatement( symbol.position, outer);
  1014. CommentStatement(returnStatement);
  1015. IF (Token() >= Scanner.Plus) & (Token() <= Scanner.Identifier) THEN
  1016. expression := Expression();
  1017. returnStatement.SetReturnValue( expression );
  1018. END;
  1019. statements.AddStatement( returnStatement );
  1020. result := TRUE;
  1021. | Scanner.Begin:
  1022. NextSymbol; statement := StatementBlock(outer); statements.AddStatement( statement ); Check( Scanner.End );
  1023. result := TRUE;
  1024. | Scanner.Await:
  1025. awaitStatement := SyntaxTree.NewAwaitStatement( symbol.position, outer );
  1026. CommentStatement(awaitStatement);
  1027. NextSymbol;
  1028. expression := Expression();
  1029. awaitStatement.SetCondition( expression );
  1030. statements.AddStatement( awaitStatement );
  1031. result := TRUE
  1032. | Scanner.Code:
  1033. (* assemble *)
  1034. code := Code(outer);
  1035. Check(Scanner.End);
  1036. statements.AddStatement( code );
  1037. result := TRUE
  1038. | Scanner.End: result := FALSE (* end of if, with, case, while, for, loop, or statement sequence *)
  1039. | Scanner.Until: result := FALSE (* end of repeat *)
  1040. | Scanner.Else: result := FALSE (* else of if or case *)
  1041. | Scanner.Elsif: result := FALSE (* elsif of if *)
  1042. | Scanner.Bar: result := FALSE (* next case *)
  1043. | Scanner.Finally: result := FALSE (* end block by finally statement *)
  1044. | Scanner.Semicolon: result := FALSE (* allow the empty statement *)
  1045. (* builtin pseudo procedures are resolved by checker *)
  1046. ELSE
  1047. result := FALSE;
  1048. (*
  1049. IF Lax THEN
  1050. expression := Expression();
  1051. statement := SyntaxTree.NewAssignment( position, NIL, expression,outer );
  1052. statements.AddStatement(statement);
  1053. result := ~error;
  1054. ELSE
  1055. result := FALSE;
  1056. END;
  1057. *)
  1058. END;
  1059. IF Trace THEN E( "Statement" ) END;
  1060. RETURN result
  1061. END Statement;
  1062. (** StatementSequence = Statement {';' Statement}. **)
  1063. PROCEDURE StatementSequence*(outer: SyntaxTree.Statement ): SyntaxTree.StatementSequence;
  1064. VAR statements: SyntaxTree.StatementSequence; b: BOOLEAN;
  1065. BEGIN
  1066. IF Trace THEN S( "StatementSequence" ) END;
  1067. statements := SyntaxTree.NewStatementSequence();
  1068. IF Lax THEN
  1069. WHILE ~Peek(Scanner.Return) & Statement(statements,outer) DO Ignore(Scanner.Semicolon) END;
  1070. IF Peek(Scanner.Return) & Statement(statements,outer) THEN Ignore(Scanner.Semicolon) END; (* return bound to end of statement sequence *)
  1071. ELSE
  1072. REPEAT
  1073. b := Statement( statements,outer )
  1074. UNTIL ~Optional( Scanner.Semicolon );
  1075. END;
  1076. IF Trace THEN E( "StatementSequence" ) END;
  1077. RETURN statements
  1078. END StatementSequence;
  1079. (** StatementBlock = [Flags] StatementSequence. **)
  1080. PROCEDURE StatementBlock(outer: SyntaxTree.Statement): SyntaxTree.StatementBlock;
  1081. VAR block: SyntaxTree.StatementBlock; position: Position;
  1082. BEGIN
  1083. IF Trace THEN S( "StatementBlock" ) END;
  1084. position := symbol.position;
  1085. position.start := position.end;
  1086. block := SyntaxTree.NewStatementBlock( position, outer );
  1087. CommentStatement(block);
  1088. IF Optional( Scanner.LeftBrace ) THEN
  1089. block.SetModifier(Flags());
  1090. END;
  1091. block.SetStatementSequence( StatementSequence(block) );
  1092. IF Trace THEN E( "StatementBlock" ) END;
  1093. RETURN block
  1094. END StatementBlock;
  1095. (** Code = { any \ 'end' \ 'with' } ['with' {('in'|'out') StatementSequence}] . **)
  1096. PROCEDURE Code(outer: SyntaxTree.Statement): SyntaxTree.Code;
  1097. VAR startPos: Position; endPos, i ,len: LONGINT; codeString: Scanner.StringType; code: SyntaxTree.Code;
  1098. end: Scanner.Token; in, out: BOOLEAN; left, right: SyntaxTree.Identifier;
  1099. statements, rules: SyntaxTree.StatementSequence;
  1100. BEGIN
  1101. startPos := symbol.position;
  1102. end := scanner.SkipToEndOfCode(startPos.start, endPos, symbol);
  1103. IF (end = Scanner.End) OR (end = Scanner.With) THEN
  1104. codeString := symbol.string;
  1105. code := SyntaxTree.NewCode(startPos,outer);
  1106. i := 0; len := LEN(codeString^);
  1107. code.SetSourceCode(codeString,len);
  1108. IF (end = Scanner.With) & Mandatory(Scanner.With) THEN
  1109. in := Optional(Scanner.In);
  1110. out := Optional(Scanner.Out);
  1111. WHILE in OR out DO
  1112. statements := StatementSequence(code);
  1113. IF in THEN rules := code.inRules ELSE rules := code.outRules END;
  1114. FOR i := 0 TO statements.Length()-1 DO
  1115. rules.AddStatement(statements.GetStatement(i));
  1116. END;
  1117. in := Optional(Scanner.In);
  1118. out := Optional(Scanner.Out);
  1119. END;
  1120. END;
  1121. END;
  1122. RETURN code;
  1123. END Code;
  1124. (** Body = 'begin' [Flags] StatementSequence ['finally' StatementSequence]
  1125. | 'code' Code. **)
  1126. PROCEDURE Body( scope: SyntaxTree.ProcedureScope ): SyntaxTree.Body;
  1127. VAR body: SyntaxTree.Body; code: SyntaxTree.Code; position: Position; previousScope: SyntaxTree.Scope;
  1128. BEGIN
  1129. previousScope := currentScope;
  1130. currentScope := scope;
  1131. IF Trace THEN S( "Body" ) END;
  1132. IF Peek( Scanner.Code ) THEN
  1133. body := SyntaxTree.NewBody(symbol.position,scope); (* empty body for the time being *)
  1134. (* assemble *)
  1135. code := Code(body);
  1136. body.SetCode(code);
  1137. ELSIF Mandatory( Scanner.Begin ) THEN
  1138. body := SyntaxTree.NewBody(symbol.position,scope);
  1139. IF Optional( Scanner.LeftBrace ) THEN
  1140. body.SetModifier(Flags());
  1141. END;
  1142. position := symbol.position;
  1143. body.SetStatementSequence(StatementSequence(body));
  1144. IF Optional( Scanner.Finally ) THEN
  1145. body.SetFinally(StatementSequence(body));
  1146. END;
  1147. END;
  1148. IF Trace THEN E( "Body" ) END;
  1149. currentScope := previousScope;
  1150. RETURN body
  1151. END Body;
  1152. (** wrapper for a body in records and modules *)
  1153. PROCEDURE BodyProcedure(parentScope: SyntaxTree.Scope): SyntaxTree.Procedure;
  1154. VAR procedureScope: SyntaxTree.ProcedureScope; procedure: SyntaxTree.Procedure;
  1155. BEGIN
  1156. procedureScope := SyntaxTree.NewProcedureScope(parentScope);
  1157. IF parentScope IS SyntaxTree.ModuleScope THEN
  1158. procedure := SyntaxTree.NewProcedure( symbol.position, Global.ModuleBodyName,procedureScope);
  1159. procedure.SetAccess(SyntaxTree.Hidden);
  1160. ELSE
  1161. procedure := SyntaxTree.NewProcedure( symbol.position, Global.RecordBodyName,procedureScope);
  1162. (*! todo: make this a hidden symbol. Problematic when used with paco. *)
  1163. procedure.SetAccess(SyntaxTree.Public + SyntaxTree.Protected + SyntaxTree.Internal);
  1164. END;
  1165. parentScope.AddProcedure(procedure);
  1166. procedure.SetType(SyntaxTree.NewProcedureType(SyntaxTree.invalidPosition,parentScope));
  1167. procedure.SetBodyProcedure(TRUE);
  1168. procedureScope.SetBody(Body(procedureScope));
  1169. RETURN procedure
  1170. END BodyProcedure;
  1171. (* ProcedureType = 'procedure' [Flags] [FormalParameters]. *)
  1172. PROCEDURE ProcedureType(position: Position; parentScope: SyntaxTree.Scope): SyntaxTree.ProcedureType;
  1173. VAR procedureType: SyntaxTree.ProcedureType;
  1174. BEGIN
  1175. IF Trace THEN S( "ProcedureType" ) END;
  1176. (* procedure symbol already consumed *)
  1177. procedureType := SyntaxTree.NewProcedureType( position, parentScope);
  1178. IF Optional(Scanner.LeftBrace) THEN
  1179. procedureType.SetModifiers(Flags());
  1180. END;
  1181. IF Optional(Scanner.LeftParenthesis) THEN FormalParameters( procedureType, parentScope) END;
  1182. IF Trace THEN E( "ProcedureType" )
  1183. END;
  1184. RETURN procedureType;
  1185. END ProcedureType;
  1186. (** ObjectType = 'object' | 'object' [Flags] ['(' (QualifiedIdentifier | ArrayType) ')'] DeclarationSequence [Body] 'end' [Identifier] . **)
  1187. PROCEDURE ObjectType(position: Position; name: SyntaxTree.Identifier; parentScope: SyntaxTree.Scope ): SyntaxTree.Type;
  1188. VAR
  1189. objectType: SyntaxTree.RecordType;
  1190. pointerType: SyntaxTree.PointerType;
  1191. recordScope: SyntaxTree.RecordScope;
  1192. qualifiedIdentifier: SyntaxTree.QualifiedIdentifier;
  1193. baseType: SyntaxTree.Type;
  1194. identifier: SyntaxTree.Identifier;
  1195. str: Scanner.StringType;
  1196. type: SyntaxTree.Type;
  1197. modifiers: SyntaxTree.Modifier;
  1198. BEGIN
  1199. IF Trace THEN S( "ObjectType" ) END;
  1200. (* symbol object already consumed *)
  1201. (* generic empty OBJECT type *)
  1202. IF Peek(Scanner.Semicolon) OR Peek(Scanner.RightParenthesis) THEN
  1203. Scanner.GetKeyword(scanner.case,Scanner.Object,identifier);
  1204. qualifiedIdentifier := SyntaxTree.NewQualifiedIdentifier(position,SyntaxTree.invalidIdentifier,identifier);
  1205. type := SyntaxTree.NewQualifiedType( position, parentScope, qualifiedIdentifier );
  1206. RETURN type
  1207. END;
  1208. recordScope := SyntaxTree.NewRecordScope(parentScope);
  1209. pointerType := SyntaxTree.NewPointerType(position,parentScope);
  1210. objectType := SyntaxTree.NewRecordType( position,parentScope,recordScope);
  1211. objectType.IsObject(TRUE);
  1212. objectType.SetPointerType(pointerType);
  1213. pointerType.SetPointerBase(objectType);
  1214. IF Optional(Scanner.LeftBrace) THEN
  1215. modifiers := Flags();
  1216. pointerType.SetModifiers(modifiers);
  1217. END;
  1218. IF Optional( Scanner.LeftParenthesis ) THEN
  1219. IF Optional(Scanner.Array) THEN
  1220. baseType := ArrayType(position, parentScope) (* TODO: correct position? *)
  1221. ELSE
  1222. qualifiedIdentifier := QualifiedIdentifier();
  1223. baseType := SyntaxTree.NewQualifiedType(qualifiedIdentifier.position, parentScope, qualifiedIdentifier)
  1224. END;
  1225. objectType.SetBaseType(baseType);
  1226. Check( Scanner.RightParenthesis )
  1227. END;
  1228. (*
  1229. IF Optional( Scanner.Implements ) THEN
  1230. REPEAT
  1231. qualifiedIdentifier := QualifiedIdentifier()
  1232. UNTIL ~Optional( Scanner.Comma );
  1233. END;
  1234. *)
  1235. IF Optional( Scanner.Semicolon ) THEN
  1236. (*Warning(symbol.position,Diagnostics.Invalid,"no semicolon allowed here");*)
  1237. END;
  1238. DeclarationSequence( recordScope);
  1239. IF Peek(Scanner.Begin) OR Peek(Scanner.Code) THEN
  1240. recordScope.SetBodyProcedure(BodyProcedure(recordScope));
  1241. END;
  1242. Check(Scanner.End);
  1243. IF ExpectThisIdentifier( name ) THEN
  1244. (* check name not always, reflect in EBNF? *)
  1245. END;
  1246. IF Trace THEN E( "ObjectType" ) END;
  1247. RETURN pointerType
  1248. END ObjectType;
  1249. (** CellType = 'cell' [Flags] [PortList] [';'] DeclarationSequence [Body] 'end' [Identifier]
  1250. | 'object'. **)
  1251. PROCEDURE CellType(position: Position; name: SyntaxTree.Identifier; parentScope: SyntaxTree.Scope; isCellNet: BOOLEAN): SyntaxTree.Type;
  1252. VAR
  1253. cellType: SyntaxTree.CellType;
  1254. cellScope: SyntaxTree.CellScope;
  1255. modifiers: SyntaxTree.Modifier;
  1256. qualifiedIdentifier: SyntaxTree.QualifiedIdentifier;
  1257. qualifiedType: SyntaxTree.Type;
  1258. BEGIN
  1259. IF Trace THEN S( "CellType" ) END;
  1260. (* symbol cell already consumed *)
  1261. cellScope := SyntaxTree.NewCellScope(parentScope);
  1262. cellType := SyntaxTree.NewCellType( position, parentScope,cellScope);
  1263. cellType.IsCellNet(isCellNet);
  1264. cellScope.SetOwnerCell(cellType);
  1265. IF Optional(Scanner.Colon) THEN
  1266. qualifiedIdentifier := QualifiedIdentifier();
  1267. qualifiedType := SyntaxTree.NewQualifiedType(qualifiedIdentifier.position, parentScope, qualifiedIdentifier );
  1268. cellType.SetBaseType( qualifiedType );
  1269. END;
  1270. IF Optional(Scanner.LeftBrace) THEN
  1271. modifiers := Flags();
  1272. cellType.SetModifiers(modifiers);
  1273. END;
  1274. IF Optional( Scanner.LeftParenthesis ) THEN
  1275. PortList(cellType,cellScope);
  1276. END;
  1277. IF Optional( Scanner.Semicolon ) THEN END;
  1278. IF Optional(Scanner.Import) THEN ImportList(cellScope) END;
  1279. DeclarationSequence( cellScope);
  1280. IF Peek(Scanner.Begin) OR Peek(Scanner.Code) THEN
  1281. cellScope.SetBodyProcedure(BodyProcedure(cellScope));
  1282. END;
  1283. Check(Scanner.End);
  1284. IF ExpectThisIdentifier( name ) THEN
  1285. (* check name not always, reflect in EBNF? *)
  1286. END;
  1287. IF Trace THEN E( "CellType" ) END;
  1288. RETURN cellType
  1289. END CellType;
  1290. (** PointerType = 'pointer' [Flags] 'to' Type. **)
  1291. PROCEDURE PointerType( position: Position; parentScope: SyntaxTree.Scope ): SyntaxTree.PointerType;
  1292. VAR pointerType: SyntaxTree.PointerType; base: SyntaxTree.Type; modifiers: SyntaxTree.Modifier;
  1293. BEGIN
  1294. IF Trace THEN S( "PointerType" ) END;
  1295. (* pointer symbol already consumed *)
  1296. pointerType := SyntaxTree.NewPointerType( position ,parentScope);
  1297. IF Optional(Scanner.LeftBrace) THEN
  1298. modifiers := Flags();
  1299. pointerType.SetModifiers(modifiers)
  1300. END;
  1301. Check( Scanner.To );
  1302. base := Type(SyntaxTree.invalidIdentifier, parentScope);
  1303. pointerType.SetPointerBase( base );
  1304. IF base IS SyntaxTree.RecordType THEN
  1305. base(SyntaxTree.RecordType).SetPointerType(pointerType);
  1306. END;
  1307. IF Trace THEN E( "PointerType" ) END;
  1308. RETURN pointerType
  1309. END PointerType;
  1310. (**
  1311. RecordType = 'record' [Flags] ['(' QualifiedIdentifier ')'] [VariableDeclaration {';' VariableDeclaration}] 'end'.
  1312. **)
  1313. PROCEDURE RecordType(position: Position; parentScope:SyntaxTree.Scope ): SyntaxTree.RecordType;
  1314. VAR
  1315. recordType: SyntaxTree.RecordType;
  1316. recordScope: SyntaxTree.RecordScope;
  1317. qualifiedIdentifier: SyntaxTree.QualifiedIdentifier; flags: SET; qualifiedType: SyntaxTree.QualifiedType;
  1318. modifier: SyntaxTree.Modifier;
  1319. BEGIN
  1320. IF Trace THEN S( "RecordType" ) END;
  1321. (* record symbol already consumed *)
  1322. flags := {};
  1323. recordScope := SyntaxTree.NewRecordScope(parentScope);
  1324. recordType := SyntaxTree.NewRecordType( position, parentScope, recordScope);
  1325. IF Optional( Scanner.LeftBrace ) THEN
  1326. modifier := Flags();
  1327. recordType.SetModifiers(modifier);
  1328. END;
  1329. IF Optional( Scanner.LeftParenthesis ) THEN
  1330. qualifiedIdentifier := QualifiedIdentifier();
  1331. qualifiedType := SyntaxTree.NewQualifiedType(qualifiedIdentifier.position, parentScope, qualifiedIdentifier );
  1332. recordType.SetBaseType( qualifiedType );
  1333. Check( Scanner.RightParenthesis )
  1334. END;
  1335. IF Lax THEN
  1336. WHILE Peek(Scanner.Identifier) DO VariableDeclaration(recordScope); Ignore(Scanner.Semicolon) END;
  1337. ELSE
  1338. REPEAT
  1339. IF Peek(Scanner.Identifier) THEN VariableDeclaration( recordScope ) END;
  1340. UNTIL ~Optional( Scanner.Semicolon );
  1341. END;
  1342. Check( Scanner.End );
  1343. IF Trace THEN E( "RecordType" ) END;
  1344. RETURN recordType
  1345. END RecordType;
  1346. (** ArrayType = 'array' 'of' Type | 'array' Expression {',' Expression} 'of' Type
  1347. | 'array' '[' MathArraySize {',' MathArraySize} ']' ['of' Type].
  1348. MathArraySize = Expression | '*' | '?'.
  1349. **)
  1350. PROCEDURE ArrayType(position: Position; parentScope: SyntaxTree.Scope ): SyntaxTree.Type;
  1351. VAR
  1352. arrayType: SyntaxTree.ArrayType;
  1353. type: SyntaxTree.Type;
  1354. base: SyntaxTree.Type;
  1355. expression: SyntaxTree.Expression;
  1356. PROCEDURE MathArray(): SyntaxTree.Type;
  1357. VAR mathType: SyntaxTree.MathArrayType; base: SyntaxTree.Type;
  1358. BEGIN
  1359. IF Optional(Scanner.Questionmark) THEN
  1360. mathType := SyntaxTree.NewMathArrayType(position,parentScope, SyntaxTree.Tensor);
  1361. ELSIF Optional(Scanner.Times) THEN (* open array *)
  1362. mathType := SyntaxTree.NewMathArrayType(position,parentScope, SyntaxTree.Open);
  1363. ELSE (* size given *)
  1364. mathType := SyntaxTree.NewMathArrayType(position,parentScope, SyntaxTree.Static);
  1365. expression := Expression();
  1366. mathType.SetLength(expression);
  1367. END;
  1368. IF Optional(Scanner.Comma) THEN
  1369. base := MathArray()
  1370. ELSIF Mandatory(Scanner.RightBracket) THEN
  1371. IF Optional( Scanner.Of ) THEN
  1372. base := Type(SyntaxTree.invalidIdentifier , parentScope ); (* base type *)
  1373. END;
  1374. END;
  1375. mathType.SetArrayBase(base);
  1376. RETURN mathType;
  1377. END MathArray;
  1378. BEGIN
  1379. IF Trace THEN S( "ArrayType" ) END;
  1380. (* array symbol already consumed *)
  1381. IF Optional(Scanner.LeftBracket) THEN (* math array *)
  1382. type := MathArray();
  1383. ELSIF Optional( Scanner.Of ) THEN (* open array *)
  1384. arrayType := SyntaxTree.NewArrayType(position,parentScope, SyntaxTree.Open);
  1385. type := arrayType;
  1386. base := Type( SyntaxTree.invalidIdentifier ,parentScope);
  1387. arrayType.SetArrayBase( base )
  1388. ELSE (* static array *)
  1389. arrayType := SyntaxTree.NewArrayType(position,parentScope, SyntaxTree.Static);
  1390. type := arrayType;
  1391. expression := SimpleExpression();
  1392. arrayType.SetLength( expression );
  1393. position := symbol.position;
  1394. IF Optional( Scanner.Comma ) THEN
  1395. base := ArrayType( position,parentScope);
  1396. arrayType.SetArrayBase( base )
  1397. ELSIF Mandatory( Scanner.Of ) THEN
  1398. base := Type(SyntaxTree.invalidIdentifier , parentScope ); (* base type *)
  1399. arrayType.SetArrayBase( base );
  1400. END;
  1401. END;
  1402. IF Trace THEN E( "ArrayType" ) END;
  1403. RETURN type
  1404. END ArrayType;
  1405. (** EnumerationType = 'enum' ['('QualifiedIdentifier')'] IdentifierDefinition ['=' Expression]
  1406. {',' IdentifierDefinition ['=' Expression]} 'end'. *)
  1407. PROCEDURE EnumerationType(position: Position; parentScope: SyntaxTree.Scope): SyntaxTree.Type;
  1408. VAR type: SyntaxTree.EnumerationType; scope: SyntaxTree.EnumerationScope; identifier: SyntaxTree.Identifier;
  1409. qualifiedIdentifier: SyntaxTree.QualifiedIdentifier; qualifiedType: SyntaxTree.QualifiedType; access: SET;
  1410. constant: SyntaxTree.Constant; expression: SyntaxTree.Expression;
  1411. BEGIN
  1412. (* enum symbol already consumed *)
  1413. scope := SyntaxTree.NewEnumerationScope(parentScope);
  1414. type := SyntaxTree.NewEnumerationType(position,parentScope, scope);
  1415. IF Optional( Scanner.LeftParenthesis ) THEN
  1416. qualifiedIdentifier := QualifiedIdentifier();
  1417. qualifiedType := SyntaxTree.NewQualifiedType(qualifiedIdentifier.position, parentScope, qualifiedIdentifier );
  1418. type.SetEnumerationBase( qualifiedType );
  1419. Check( Scanner.RightParenthesis )
  1420. END;
  1421. REPEAT
  1422. IdentifierDefinition(identifier,access,FALSE);
  1423. position := symbol.position;
  1424. constant := SyntaxTree.NewConstant( position, identifier );
  1425. CommentSymbol(constant);
  1426. constant.SetAccess(access);
  1427. IF Optional(Scanner.Equal) THEN
  1428. expression := Expression();
  1429. constant.SetValue( expression );
  1430. END;
  1431. scope.AddConstant( constant );
  1432. UNTIL ~Optional(Scanner.Comma);
  1433. IF Mandatory(Scanner.End) THEN END;
  1434. RETURN type
  1435. END EnumerationType;
  1436. (** PortType = 'port' ('in'|'out') ['(' Expression ')'] *)
  1437. PROCEDURE PortType(position: Position; parentScope: SyntaxTree.Scope): SyntaxTree.Type;
  1438. VAR type: SyntaxTree.Type; direction: LONGINT; sizeExpression: SyntaxTree.Expression;
  1439. BEGIN
  1440. (* port symbol already consumed *)
  1441. IF Optional(Scanner.In) THEN
  1442. direction := SyntaxTree.InPort
  1443. ELSIF Optional(Scanner.Out) THEN
  1444. direction := SyntaxTree.OutPort
  1445. ELSE
  1446. Error(position,Diagnostics.Invalid,"invalid direction, expected IN or OUT");
  1447. END;
  1448. IF Optional(Scanner.LeftParenthesis) THEN
  1449. sizeExpression := Expression();
  1450. IF Mandatory(Scanner.RightParenthesis )THEN END;
  1451. END;
  1452. type := SyntaxTree.NewPortType(position, direction, sizeExpression, parentScope);
  1453. RETURN type
  1454. END PortType;
  1455. (** Type = ArrayType | RecordType | PointerType | ObjectType | CellType | CellnetType | PortType
  1456. | ProcedureType | EnumerationType | QualifiedIdentifier. *)
  1457. PROCEDURE Type( name: SyntaxTree.Identifier; parentScope: SyntaxTree.Scope ): SyntaxTree.Type;
  1458. VAR type: SyntaxTree.Type; qualifiedIdentifier: SyntaxTree.QualifiedIdentifier; position: Position;
  1459. BEGIN
  1460. IF Trace THEN S( "Type" ) END;
  1461. position := symbol.position;
  1462. IF Optional( Scanner.Array ) THEN type := ArrayType( position,parentScope );
  1463. ELSIF Optional( Scanner.Record ) THEN type := RecordType( position,parentScope );
  1464. ELSIF Optional( Scanner.Pointer ) THEN type := PointerType( position,parentScope );
  1465. ELSIF Optional( Scanner.Object ) THEN type := ObjectType( position,name,parentScope );
  1466. ELSIF Optional( Scanner.Cell) THEN type := CellType( position, name, parentScope,FALSE);
  1467. ELSIF Optional( Scanner.CellNet) THEN type := CellType( position, name, parentScope, TRUE);
  1468. ELSIF Optional( Scanner.Port) THEN type := PortType( position, parentScope)
  1469. ELSIF Optional( Scanner.Procedure ) THEN type := ProcedureType( position,parentScope);
  1470. ELSIF Optional( Scanner.Enum ) THEN type := EnumerationType( position,parentScope);
  1471. ELSIF (Token() = Scanner.Address) OR (Token() = Scanner.Size) THEN
  1472. qualifiedIdentifier := SyntaxTree.NewQualifiedIdentifier(position,SyntaxTree.invalidIdentifier, symbol.identifier);
  1473. type := SyntaxTree.NewQualifiedType( qualifiedIdentifier.position, parentScope, qualifiedIdentifier );
  1474. NextSymbol;
  1475. ELSE qualifiedIdentifier := QualifiedIdentifier();
  1476. type := SyntaxTree.NewQualifiedType( qualifiedIdentifier.position, parentScope, qualifiedIdentifier );
  1477. END;
  1478. IF Trace THEN E( "Type" ) END;
  1479. RETURN type
  1480. END Type;
  1481. (** PortDeclaration = Identifier [Flags] {',' Identifier [Flags]}':' Type. **)
  1482. PROCEDURE PortDeclaration(cell: SyntaxTree.CellType; parentScope: SyntaxTree.Scope);
  1483. VAR
  1484. type: SyntaxTree.Type; name: SyntaxTree.Identifier;
  1485. firstParameter, parameter: SyntaxTree.Parameter;
  1486. position: Position; modifiers: SyntaxTree.Modifier;
  1487. BEGIN
  1488. IF Trace THEN S( "PortDeclaration" ) END;
  1489. firstParameter := cell.lastParameter;
  1490. REPEAT
  1491. name := Identifier(position);
  1492. parameter := SyntaxTree.NewParameter(position,cell,name,SyntaxTree.ValueParameter);
  1493. cell.AddParameter(parameter);
  1494. IF Optional(Scanner.LeftBrace) THEN
  1495. modifiers := Flags();
  1496. parameter.SetModifiers(modifiers);
  1497. END;
  1498. UNTIL ~Optional( Scanner.Comma );
  1499. Check( Scanner.Colon );
  1500. type := Type( SyntaxTree.invalidIdentifier, parentScope);
  1501. ASSERT(type # NIL);
  1502. IF firstParameter # NIL THEN parameter := firstParameter.nextParameter ELSE parameter := cell.firstParameter END;
  1503. WHILE parameter # NIL DO
  1504. parameter.SetType( type );
  1505. parameter := parameter.nextParameter;
  1506. END;
  1507. IF Trace THEN E( "PortDeclaration" )
  1508. END;
  1509. END PortDeclaration;
  1510. (** PortList = '(' [PortDeclaration {';' PortDeclaration}] ')'. **)
  1511. PROCEDURE PortList( cell: SyntaxTree.CellType ; parentScope: SyntaxTree.Scope);
  1512. BEGIN
  1513. IF Trace THEN S( "PortList" ) END;
  1514. (* left parenthesis already consumed *)
  1515. IF ~Optional( Scanner.RightParenthesis ) THEN
  1516. IF Lax THEN
  1517. WHILE Peek(Scanner.Identifier) OR Peek(Scanner.Var) OR Peek(Scanner.Const) DO PortDeclaration( cell, parentScope ); Ignore(Scanner.Semicolon) END;
  1518. ELSE
  1519. REPEAT PortDeclaration( cell, parentScope ); UNTIL ~Optional( Scanner.Semicolon );
  1520. END;
  1521. Check( Scanner.RightParenthesis );
  1522. END;
  1523. IF Trace THEN E( "PortList" ) END;
  1524. END PortList;
  1525. (** ParameterDeclaration = ['var'|'const'] Identifier [Flags] ['= Expression] {',' Identifier [Flags] ['= Expression]}':' Type.**)
  1526. PROCEDURE ParameterDeclaration( procedureType: SyntaxTree.ProcedureType ; parentScope: SyntaxTree.Scope);
  1527. VAR
  1528. type: SyntaxTree.Type; name: SyntaxTree.Identifier;
  1529. firstParameter, parameter: SyntaxTree.Parameter; kind: LONGINT; position: Position;
  1530. BEGIN
  1531. IF Trace THEN S( "ParameterDeclaration" ) END;
  1532. IF Optional( Scanner.Var ) THEN (* var parameter *)
  1533. kind := SyntaxTree.VarParameter
  1534. ELSIF Optional( Scanner.Const ) THEN (* const parameter *)
  1535. kind := SyntaxTree.ConstParameter
  1536. ELSIF Token() # Scanner.Identifier THEN
  1537. Error(symbol.position,Scanner.Identifier,"");
  1538. RETURN
  1539. ELSE kind := SyntaxTree.ValueParameter
  1540. END;
  1541. firstParameter := procedureType.lastParameter;
  1542. REPEAT
  1543. name := Identifier(position);
  1544. parameter := SyntaxTree.NewParameter(position,procedureType,name,kind);
  1545. IF Optional(Scanner.LeftBrace) THEN parameter.SetModifiers(Flags()) END;
  1546. procedureType.AddParameter(parameter);
  1547. IF Optional(Scanner.Equal) THEN
  1548. parameter.SetDefaultValue(Expression());
  1549. END
  1550. UNTIL ~Optional( Scanner.Comma );
  1551. Check( Scanner.Colon );
  1552. type := Type( SyntaxTree.invalidIdentifier, parentScope);
  1553. CommentSymbol(parameter);
  1554. ASSERT(type # NIL);
  1555. IF firstParameter # NIL THEN parameter := firstParameter.nextParameter ELSE parameter := procedureType.firstParameter END;
  1556. WHILE parameter # NIL DO
  1557. parameter.SetType( type );
  1558. parameter := parameter.nextParameter;
  1559. END;
  1560. IF Trace THEN E( "ParameterDeclaration" )
  1561. END;
  1562. END ParameterDeclaration;
  1563. (** FormalParameters = '(' [ParameterDeclaration {';' ParameterDeclaration}] ')' [':' [Flags] Type]. **)
  1564. PROCEDURE FormalParameters( procedureType: SyntaxTree.ProcedureType ; parentScope: SyntaxTree.Scope);
  1565. VAR type: SyntaxTree.Type; position: Position;
  1566. BEGIN
  1567. IF Trace THEN S( "FormalParameters" ) END;
  1568. (* left parenthesis already consumed *)
  1569. IF ~Optional( Scanner.RightParenthesis ) THEN
  1570. IF Lax THEN
  1571. WHILE Peek(Scanner.Identifier) OR Peek(Scanner.Const) OR Peek(Scanner.Var) DO
  1572. ParameterDeclaration(procedureType, parentScope); Ignore(Scanner.Semicolon)
  1573. END;
  1574. ELSE
  1575. REPEAT ParameterDeclaration( procedureType, parentScope ); UNTIL ~Optional( Scanner.Semicolon );
  1576. END;
  1577. Check( Scanner.RightParenthesis );
  1578. END;
  1579. IF Optional( Scanner.Colon ) THEN
  1580. position:= symbol.position;
  1581. IF Optional( Scanner.LeftBrace) THEN
  1582. procedureType.SetReturnTypeModifiers(Flags());
  1583. END;
  1584. type := Type(SyntaxTree.invalidIdentifier,parentScope);
  1585. (* formally, any type is permitted as return type. Actually some of them might simply not be usable *)
  1586. procedureType.SetReturnType(type);
  1587. END;
  1588. IF Trace THEN E( "FormalParameters" ) END;
  1589. END FormalParameters;
  1590. (** Flags = '{' [Identifier ['(' Expression ')'|'=' Expression] {',' Identifier ['(' Expression ')' | '=' Expression ] } ] '}'. **)
  1591. PROCEDURE Flags(): SyntaxTree.Modifier;
  1592. VAR identifier: SyntaxTree.Identifier; modifier,list: SyntaxTree.Modifier; position: Position; expression: SyntaxTree.Expression;
  1593. BEGIN
  1594. IF Trace THEN S( "Flags" ) END;
  1595. (* left brace already consumed *)
  1596. list := NIL;
  1597. IF Peek(Scanner.RightBrace) THEN (* empty flags *)
  1598. ELSE
  1599. REPEAT
  1600. position := symbol.position;
  1601. identifier := Identifier(position);
  1602. IF Optional(Scanner.LeftParenthesis) THEN
  1603. expression := Expression();
  1604. Check(Scanner.RightParenthesis)
  1605. ELSIF Optional(Scanner.Equal) THEN
  1606. expression := Expression();
  1607. ELSE
  1608. expression := NIL
  1609. END;
  1610. modifier := SyntaxTree.NewModifier(position,identifier,expression);
  1611. AppendModifier(list,modifier);
  1612. UNTIL ~Optional( Scanner.Comma ) & ~Optional(Scanner.Semicolon);
  1613. END;
  1614. Check(Scanner.RightBrace);
  1615. IF Trace THEN E( "Flags" ) END;
  1616. RETURN list;
  1617. END Flags;
  1618. PROCEDURE SetNextInComment(c: SyntaxTree.Comment; this: ANY);
  1619. BEGIN
  1620. WHILE c # NIL DO
  1621. c.SetItem(this,FALSE);
  1622. c := c.nextComment
  1623. END;
  1624. END SetNextInComment;
  1625. PROCEDURE CommentSymbol(symbol: SyntaxTree.Symbol);
  1626. BEGIN
  1627. IF (recentComment # NIL) (* & (recentComment.nextSymbol = NIL) *) THEN
  1628. symbol.SetComment(recentComment);
  1629. SetNextInComment(recentComment, symbol);
  1630. recentComment := NIL
  1631. END;
  1632. recentLine := scanner.position.line;
  1633. recentCommentItem := symbol;
  1634. END CommentSymbol;
  1635. PROCEDURE CommentStatement(symbol: SyntaxTree.Statement);
  1636. BEGIN
  1637. IF (recentComment # NIL) (* & (recentComment.nextSymbol = NIL) *) THEN
  1638. symbol.SetComment(recentComment);
  1639. SetNextInComment(recentComment, symbol);
  1640. recentComment := NIL
  1641. END;
  1642. recentLine := scanner.position.line;
  1643. recentCommentItem := symbol
  1644. END CommentStatement;
  1645. PROCEDURE CommentCasePart(symbol: SyntaxTree.CasePart);
  1646. BEGIN
  1647. IF (recentComment # NIL) (* & (recentComment.nextSymbol = NIL) *) THEN
  1648. symbol.SetComment(recentComment);
  1649. SetNextInComment(recentComment, symbol);
  1650. recentComment := NIL
  1651. END;
  1652. recentLine := scanner.position.line;
  1653. recentCommentItem := symbol
  1654. END CommentCasePart;
  1655. PROCEDURE CommentIfPart(symbol: SyntaxTree.IfPart);
  1656. BEGIN
  1657. IF (recentComment # NIL) (* & (recentComment.nextSymbol = NIL) *) THEN
  1658. symbol.SetComment(recentComment);
  1659. SetNextInComment(recentComment, symbol);
  1660. recentComment := NIL
  1661. END;
  1662. recentLine := scanner.position.line;
  1663. recentCommentItem := symbol
  1664. END CommentIfPart;
  1665. PROCEDURE CommentWithPart(symbol: SyntaxTree.WithPart);
  1666. BEGIN
  1667. IF (recentComment # NIL) (* & (recentComment.nextSymbol = NIL) *) THEN
  1668. symbol.SetComment(recentComment);
  1669. SetNextInComment(recentComment, symbol);
  1670. recentComment := NIL
  1671. END;
  1672. recentLine := scanner.position.line;
  1673. recentCommentItem := symbol
  1674. END CommentWithPart;
  1675. (** ProcedureDeclaration = 'procedure' ['^'|'&'|'~'|'-'|Flags ['-']] IdentifierDefinition [FormalParameters]';'
  1676. DeclarationSequence [Body] 'end' Identifier.
  1677. Forward declarations ignored.
  1678. **)
  1679. PROCEDURE ProcedureDeclaration( parentScope: SyntaxTree.Scope);
  1680. VAR name: SyntaxTree.Identifier;
  1681. procedure: SyntaxTree.Procedure;
  1682. procedureType: SyntaxTree.ProcedureType;
  1683. procedureScope : SyntaxTree.ProcedureScope;
  1684. access: SET;
  1685. position: Position;
  1686. isConstructor: BOOLEAN;
  1687. isFinalizer: BOOLEAN;
  1688. isInline: BOOLEAN;
  1689. modifiers: SyntaxTree.Modifier;
  1690. forwardDeclaration: BOOLEAN;
  1691. string: Scanner.StringType;
  1692. BEGIN
  1693. IF Trace THEN S( "Procedure" ) END;
  1694. (* symbol procedure has already been consumed *)
  1695. modifiers := NIL;
  1696. isConstructor := FALSE; isFinalizer := FALSE; isInline := FALSE;
  1697. procedureType := SyntaxTree.NewProcedureType(symbol.position, parentScope);
  1698. IF Optional( Scanner.Arrow) THEN (* ignore forward declarations *)
  1699. forwardDeclaration := TRUE;
  1700. ELSE forwardDeclaration := FALSE;
  1701. END;
  1702. IF Optional( Scanner.And ) THEN (* constructor *)
  1703. isConstructor := TRUE
  1704. ELSIF Optional( Scanner.Not ) THEN (* finalizer *)
  1705. isFinalizer := TRUE
  1706. ELSIF Optional( Scanner.Minus ) THEN (* inline *)
  1707. isInline := TRUE;
  1708. ELSIF Optional( Scanner.LeftBrace) THEN
  1709. modifiers := Flags();
  1710. IF Optional( Scanner.Minus ) THEN (* inline *)
  1711. isInline := TRUE
  1712. END;
  1713. END;
  1714. IF Peek(Scanner.String) OR Peek(Scanner.Character) THEN (* for compatibility with release *)
  1715. OperatorDeclaration( parentScope );
  1716. procedure.SetInline(isInline);
  1717. RETURN
  1718. END;
  1719. position:= symbol.position;
  1720. IdentifierDefinition( name, access,TRUE);
  1721. procedureScope := SyntaxTree.NewProcedureScope(parentScope);
  1722. procedure := SyntaxTree.NewProcedure( position, name, procedureScope);
  1723. procedure.SetConstructor(isConstructor);
  1724. procedure.SetFinalizer(isFinalizer);
  1725. procedure.SetInline(isInline);
  1726. CommentSymbol(procedure);
  1727. procedure.SetAccess(access);
  1728. procedureType.SetModifiers(modifiers);
  1729. procedure.SetType(procedureType);
  1730. IF Optional(Scanner.Extern) & MandatoryString(string) THEN procedure.SetExternalName(string); END;
  1731. IF Optional(Scanner.LeftParenthesis) THEN FormalParameters( procedureType, procedureScope) END;
  1732. IF (procedure.externalName = NIL) & ~forwardDeclaration THEN
  1733. IF Lax THEN Ignore(Scanner.Semicolon) ELSE Check( Scanner.Semicolon ) END;
  1734. DeclarationSequence( procedureScope);
  1735. IF Peek(Scanner.Begin) OR Peek(Scanner.Code) THEN
  1736. procedureScope.SetBody(Body(procedureScope));
  1737. END;
  1738. Check(Scanner.End);
  1739. IF ExpectThisIdentifier( name ) THEN END;
  1740. END;
  1741. parentScope.AddProcedure( procedure );
  1742. IF Trace THEN E( "Procedure") END;
  1743. END ProcedureDeclaration;
  1744. (** OperatorDeclaration = 'operator' [Flags] ['-'] String ['*'|'-'] FormalParameters ';'
  1745. DeclarationSequence [Body] 'end' String.
  1746. **)
  1747. PROCEDURE OperatorDeclaration(parentScope: SyntaxTree.Scope );
  1748. VAR
  1749. string: Scanner.StringType;
  1750. procedureScope: SyntaxTree.ProcedureScope;
  1751. procedureType: SyntaxTree.ProcedureType;
  1752. operator: SyntaxTree.Operator;
  1753. access: SET;
  1754. i: LONGINT; ch: CHAR; position: Position;
  1755. modifiers: SyntaxTree.Modifier; (* nopov *)
  1756. isInline, forward: BOOLEAN;
  1757. BEGIN
  1758. IF Trace THEN S( "Operator" ) END;
  1759. (* symbol operator already consumed *)
  1760. position := symbol.position;
  1761. forward := Optional(Scanner.Arrow);
  1762. isInline := FALSE;
  1763. IF Optional( Scanner.LeftBrace) THEN
  1764. modifiers := Flags();
  1765. END;
  1766. IF Optional( Scanner.Minus ) THEN (* inline *)
  1767. isInline := TRUE;
  1768. END;
  1769. IF MandatoryString( string ) THEN
  1770. (* copy string to name and check for length. LEN(name)>0, LEN(string)>0 can be presumed *)
  1771. i := 0; WHILE (string^[i] # 0X) DO INC(i) END;
  1772. IF i >= Scanner.MaxIdentifierLength THEN (* string too long to act as operator identifier *)
  1773. Error(symbol.position,Basic.StringTooLong,"");
  1774. END
  1775. END;
  1776. IF Optional( Scanner.Times ) THEN access := SyntaxTree.ReadOnly;
  1777. ELSIF Optional( Scanner.Minus ) THEN access := SyntaxTree.ReadOnly;
  1778. ELSE access := SyntaxTree.Internal;
  1779. END;
  1780. procedureScope := SyntaxTree.NewProcedureScope(parentScope);
  1781. operator := SyntaxTree.NewOperator( symbol.position, SyntaxTree.NewIdentifier(string^), procedureScope);
  1782. CommentSymbol(operator);
  1783. operator.SetAccess(access);
  1784. procedureType := SyntaxTree.NewProcedureType(symbol.position,parentScope);
  1785. IF Mandatory(Scanner.LeftParenthesis) THEN FormalParameters( procedureType, procedureScope ) END;
  1786. procedureType.SetModifiers(modifiers); (* nopov *)
  1787. operator.SetType( procedureType );
  1788. operator.SetInline(isInline);
  1789. IF Lax THEN Ignore(Scanner.Semicolon) ELSE Check( Scanner.Semicolon ) END;
  1790. IF ~forward THEN
  1791. DeclarationSequence( procedureScope );
  1792. IF Peek(Scanner.Begin) OR Peek(Scanner.Code) THEN
  1793. procedureScope.SetBody(Body(procedureScope));
  1794. END;
  1795. IF Mandatory(Scanner.End) & ExpectThisString(string^) THEN END;
  1796. END;
  1797. parentScope.AddProcedure(operator);
  1798. IF parentScope IS SyntaxTree.ModuleScope THEN
  1799. parentScope(SyntaxTree.ModuleScope).AddOperator(operator);
  1800. ELSIF parentScope IS SyntaxTree.RecordScope THEN
  1801. parentScope(SyntaxTree.RecordScope).AddOperator(operator);
  1802. ELSE
  1803. Error(position,Diagnostics.Invalid,"Operators only allowed in module or record scope"); (* nopov *)
  1804. END;
  1805. IF Trace THEN EE( "Operator", string^ ) END;
  1806. END OperatorDeclaration;
  1807. (** VariableNameList = IdentifierDefinition [Flags] [':=' Expression | 'extern' String] {',' IdentifierDefinition [Flags] [':=' Expression | 'extern' String] }.**)
  1808. PROCEDURE VariableNameList( scope: SyntaxTree.Scope );
  1809. VAR varname: SyntaxTree.Identifier; position: Position; variable: SyntaxTree.Variable; flags,access: SET; string: Scanner.StringType;
  1810. BEGIN
  1811. IF Trace THEN S( "VariableNameList" ) END;
  1812. REPEAT
  1813. flags := {};
  1814. position := symbol.position;
  1815. IdentifierDefinition( varname, access,TRUE);
  1816. variable := SyntaxTree.NewVariable( position, varname );
  1817. CommentSymbol(variable);
  1818. IF Optional(Scanner.LeftBrace) THEN variable.SetModifiers(Flags()) END;
  1819. IF Optional(Scanner.Becomes) THEN variable.SetInitializer (Expression());
  1820. ELSIF Optional(Scanner.Extern) & MandatoryString(string) THEN variable.SetExternalName(string); END;
  1821. variable.SetAccess(access);
  1822. scope.AddVariable(variable);
  1823. UNTIL ~Optional( Scanner.Comma );
  1824. IF Trace THEN E( "VariableNameList" ) END;
  1825. END VariableNameList;
  1826. (** VariableDeclaration = VariableNameList ':' Type. **)
  1827. PROCEDURE VariableDeclaration(parentScope: SyntaxTree.Scope );
  1828. VAR
  1829. variable, firstVariable: SyntaxTree.Variable; type: SyntaxTree.Type;
  1830. BEGIN
  1831. IF Trace THEN S( "VariableDeclaration" ) END;
  1832. firstVariable := parentScope.lastVariable;
  1833. VariableNameList( parentScope );
  1834. Check( Scanner.Colon );
  1835. type := Type( SyntaxTree.invalidIdentifier, parentScope );
  1836. variable := firstVariable;
  1837. IF firstVariable # NIL THEN variable := firstVariable.nextVariable ELSE variable := parentScope.firstVariable END;
  1838. WHILE variable # NIL DO
  1839. variable.SetType( type );
  1840. variable := variable.nextVariable;
  1841. END;
  1842. IF Trace THEN E( "VariableDeclaration" ) END;
  1843. END VariableDeclaration;
  1844. (** TypeDeclaration = IdentifierDefinition '=' Type.**)
  1845. PROCEDURE TypeDeclaration(parentScope: SyntaxTree.Scope);
  1846. VAR name: SyntaxTree.Identifier; position: Position; type: SyntaxTree.Type; typeDeclaration: SyntaxTree.TypeDeclaration; access: SET;
  1847. BEGIN
  1848. IF Trace THEN S( "TypeDeclaration" ) END;
  1849. position := symbol.position;
  1850. IdentifierDefinition( name, access,FALSE);
  1851. typeDeclaration := SyntaxTree.NewTypeDeclaration( position,name);
  1852. CommentSymbol(typeDeclaration);
  1853. Check( Scanner.Equal );
  1854. type := Type( name , parentScope);
  1855. type.SetTypeDeclaration(typeDeclaration);
  1856. typeDeclaration.SetDeclaredType(type);
  1857. (*
  1858. type.SetName(typeDeclaration.name); (* don't do that: overwrites global names ! *)
  1859. *)
  1860. typeDeclaration.SetAccess(access);
  1861. parentScope.AddTypeDeclaration( typeDeclaration );
  1862. IF Trace THEN E( "TypeDeclaration" ) END;
  1863. END TypeDeclaration;
  1864. (** ConstDeclaration = IdentifierDefinition '=' Expression. **)
  1865. PROCEDURE ConstDeclaration(parentScope: SyntaxTree.Scope );
  1866. VAR name: SyntaxTree.Identifier; position: Position; constant: SyntaxTree.Constant; expression: SyntaxTree.Expression; access: SET;
  1867. BEGIN
  1868. IF Trace THEN S( "ConstDeclaration" ) END;
  1869. IdentifierDefinition( name, access, FALSE);
  1870. position := symbol.position;
  1871. constant := SyntaxTree.NewConstant( position, name );
  1872. CommentSymbol(constant);
  1873. constant.SetAccess(access);
  1874. Check( Scanner.Equal );
  1875. expression := Expression();
  1876. constant.SetValue( expression );
  1877. parentScope.AddConstant( constant );
  1878. IF Trace THEN E( "ConstDeclaration" ) END;
  1879. END ConstDeclaration;
  1880. (** DeclarationSequence = { 'const' [ConstDeclaration] {';' [ConstDeclaration]}
  1881. |'type' [TypeDeclaration] {';' [TypeDeclaration]}
  1882. |'var' [VariableDeclaration] {';' [VariableDeclaration]}
  1883. | ProcedureDeclaration
  1884. | OperatorDeclaration
  1885. | ';'
  1886. }
  1887. **)
  1888. PROCEDURE DeclarationSequence( parentScope: SyntaxTree.Scope);
  1889. VAR previousScope: SyntaxTree.Scope;
  1890. BEGIN
  1891. previousScope := currentScope;
  1892. currentScope := parentScope;
  1893. IF Trace THEN S( "DeclarationSequence" ) END;
  1894. IF Lax THEN
  1895. LOOP
  1896. Ignore(Scanner.Semicolon);
  1897. IF Optional(Scanner.Const) THEN
  1898. WHILE Peek(Scanner.Identifier) DO ConstDeclaration(parentScope); Ignore(Scanner.Semicolon) END;
  1899. ELSIF Optional(Scanner.Type) THEN
  1900. WHILE Peek(Scanner.Identifier) DO TypeDeclaration(parentScope); Ignore(Scanner.Semicolon) END;
  1901. ELSIF Optional(Scanner.Var) THEN
  1902. WHILE Peek(Scanner.Identifier) DO VariableDeclaration(parentScope); Ignore(Scanner.Semicolon); END;
  1903. ELSIF Optional(Scanner.Procedure) THEN
  1904. ProcedureDeclaration(parentScope); Ignore(Scanner.Semicolon)
  1905. ELSIF Optional(Scanner.Operator) THEN
  1906. OperatorDeclaration(parentScope); Ignore(Scanner.Semicolon);
  1907. ELSE
  1908. EXIT
  1909. END;
  1910. END;
  1911. ELSE
  1912. LOOP
  1913. IF Optional( Scanner.Const ) THEN
  1914. REPEAT
  1915. IF Peek(Scanner.Identifier) THEN ConstDeclaration( parentScope ) END
  1916. UNTIL ~Optional( Scanner.Semicolon )
  1917. ELSIF Optional( Scanner.Type ) THEN
  1918. REPEAT
  1919. IF Peek(Scanner.Identifier) THEN TypeDeclaration( parentScope) END
  1920. UNTIL ~Optional( Scanner.Semicolon )
  1921. ELSIF Optional( Scanner.Var ) THEN
  1922. REPEAT
  1923. IF Peek(Scanner.Identifier) THEN VariableDeclaration( parentScope ) END
  1924. UNTIL ~Optional( Scanner.Semicolon )
  1925. ELSIF Optional(Scanner.Operator) THEN
  1926. OperatorDeclaration( parentScope);
  1927. ELSIF Optional( Scanner.Procedure ) THEN
  1928. ProcedureDeclaration( parentScope );
  1929. ELSE EXIT
  1930. END;
  1931. Ignore(Scanner.Semicolon)
  1932. END;
  1933. END;
  1934. currentScope := previousScope;
  1935. IF Trace THEN E( "DeclarationSequence" ) END;
  1936. END DeclarationSequence;
  1937. (**
  1938. ImportList = 'import' Import { ',' Import } ';'.
  1939. Import = Identifier [':=' Identifier] ['in' Identifier].
  1940. **)
  1941. PROCEDURE ImportList( scope: SyntaxTree.Scope );
  1942. VAR alias, name, context: SyntaxTree.Identifier; import: SyntaxTree.Import; position, idPosition: Position;
  1943. BEGIN
  1944. IF Trace THEN S( "ImportList" ) END;
  1945. (* import symbol already consumed *)
  1946. REPEAT
  1947. alias := Identifier(idPosition);
  1948. position := symbol.position;
  1949. IF alias # SyntaxTree.invalidIdentifier THEN
  1950. IF Optional( Scanner.Becomes ) THEN name := Identifier(idPosition) ELSE name := alias; END;
  1951. import := SyntaxTree.NewImport( position, alias, name, TRUE );
  1952. CommentSymbol(import);
  1953. IF Optional(Scanner.In) THEN
  1954. position := symbol.position;
  1955. context := Identifier(idPosition);
  1956. IF context # SyntaxTree.invalidIdentifier THEN import.SetContext(context) END;
  1957. END;
  1958. WITH scope: SyntaxTree.ModuleScope DO
  1959. scope.AddImport( import );
  1960. | scope: SyntaxTree.CellScope DO
  1961. scope.AddImport( import );
  1962. END;
  1963. END;
  1964. UNTIL ~Optional( Scanner.Comma );
  1965. IF Lax THEN Ignore(Scanner.Semicolon) ELSE Check( Scanner.Semicolon ) END;
  1966. IF Trace THEN E( "ImportList" ); END;
  1967. END ImportList;
  1968. (** Module = ('module' | 'cellnet' [Flags]) Identifier ['in' Identifier]';'
  1969. [ImportList] DeclarationSequence [Body]
  1970. 'end' Identifier '.'.
  1971. **)
  1972. PROCEDURE Module*(): SyntaxTree.Module;
  1973. VAR moduleName, context: SyntaxTree.Identifier; module: SyntaxTree.Module; position: Position; isCellNet: BOOLEAN;
  1974. scannerDiagnostics: Diagnostics.Diagnostics; modifiers: SyntaxTree.Modifier; c: SyntaxTree.Comment;
  1975. BEGIN
  1976. IF Trace THEN S( "Module" ) END;
  1977. position := symbol.position;
  1978. moduleScope := SyntaxTree.NewModuleScope(); (* needed to feed in comment already before module starts *)
  1979. currentScope := moduleScope;
  1980. isCellNet := Optional(Scanner.CellNet);
  1981. IF isCellNet OR Mandatory( Scanner.Module ) THEN
  1982. (*c := recentComment; recentComment := NIL;*)
  1983. IF isCellNet & Optional(Scanner.LeftBrace) THEN modifiers := Flags() ELSE modifiers := NIL END;
  1984. moduleName := Identifier(position);
  1985. module := SyntaxTree.NewModule( scanner.source^, position, moduleName, moduleScope, scanner.case );
  1986. CommentSymbol(module);
  1987. (*
  1988. module.SetComment(c);
  1989. SetNextInComment(c, module);
  1990. *)
  1991. IF isCellNet THEN module.SetCellNet(TRUE); module.SetModifiers(modifiers); END;
  1992. module.SetType(SyntaxTree.moduleType);
  1993. IF Optional(Scanner.In) THEN
  1994. position := symbol.position;
  1995. context := Identifier(position);
  1996. module.SetContext(context);
  1997. END;
  1998. IF Lax THEN Ignore(Scanner.Semicolon) ELSE Check(Scanner.Semicolon) END;
  1999. IF ~Peek(Scanner.EndOfText) THEN
  2000. module.SetClosingComment(recentComment);
  2001. SetNextInComment(recentComment, module);
  2002. recentComment := NIL;
  2003. END;
  2004. IF Optional(Scanner.Import) THEN ImportList(moduleScope) END;
  2005. DeclarationSequence( moduleScope);
  2006. IF Peek(Scanner.Begin) OR Peek(Scanner.Code) THEN
  2007. moduleScope.SetBodyProcedure(BodyProcedure(moduleScope)); (* insert empty body procedure if necessary *)
  2008. END;
  2009. Check(Scanner.End);
  2010. IF ExpectThisIdentifier( moduleName ) THEN
  2011. IF Token() # Scanner.Period THEN
  2012. Error( symbol.position, Scanner.Period, "" )
  2013. ELSIF ~error & ~scanner.error THEN (* read ahead to read comments and to check for next module *)
  2014. scanner.ResetCase;
  2015. scannerDiagnostics := NIL;
  2016. scanner.ResetErrorDiagnostics(scannerDiagnostics);
  2017. NextSymbol;
  2018. scanner.ResetErrorDiagnostics(scannerDiagnostics);
  2019. END;
  2020. (*
  2021. (* do not use Check for not reading after end of module *)
  2022. IF ~Peek(Scanner.Module) & ~Peek(Scanner.CellNet) THEN
  2023. SetNextInComment(recentComment,module);
  2024. module.SetClosingComment(recentComment);
  2025. recentComment := NIL;
  2026. END;
  2027. *)
  2028. END;
  2029. END;
  2030. IF Trace THEN E( "Module" ) END;
  2031. RETURN module
  2032. END Module;
  2033. (** check if another module declaration is available after recent module parsing -> for parsing and compiling multiple modules within a single file **)
  2034. PROCEDURE NextModule*(): BOOLEAN;
  2035. BEGIN
  2036. RETURN Peek(Scanner.Module) OR Peek(Scanner.CellNet);
  2037. END NextModule;
  2038. END Parser;
  2039. (* utilities *)
  2040. PROCEDURE AppendModifier(VAR list: SyntaxTree.Modifier; modifier: SyntaxTree.Modifier);
  2041. VAR this, next: SyntaxTree.Modifier;
  2042. BEGIN
  2043. IF list = NIL THEN list := modifier
  2044. ELSE
  2045. this := list;
  2046. next := list.nextModifier;
  2047. WHILE next # NIL DO
  2048. this := next;
  2049. next := this.nextModifier;
  2050. END;
  2051. this.SetNext(modifier);
  2052. END;
  2053. END AppendModifier;
  2054. (** parser retrieval **)
  2055. PROCEDURE NewParser*( scanner: Scanner.Scanner; diagnostics: Diagnostics.Diagnostics): Parser;
  2056. VAR parser: Parser;
  2057. BEGIN
  2058. NEW( parser, scanner, diagnostics ); RETURN parser;
  2059. END NewParser;
  2060. END FoxParser.