FoxScanner.Mod 52 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595
  1. MODULE FoxScanner; (** AUTHOR "fof & fn"; PURPOSE "Oberon Compiler: Scanner"; **)
  2. (* (c) fof ETH Zürich, 2009 *)
  3. IMPORT Streams, Strings, Diagnostics, Basic := FoxBasic, D := Debugging, Commands, StringPool;
  4. CONST
  5. Trace = FALSE; (* debugging output *)
  6. (* overal scanner limitation *)
  7. MaxIdentifierLength* = 128;
  8. (* parametrization of numeric scanner: *)
  9. MaxHexDigits* = 8; (* maximal hexadecimal longint length *)
  10. MaxHugeHexDigits* = 16; (* maximal hexadecimal hugeint length *)
  11. MaxRealExponent* = 38; (* maximal real exponent *)
  12. MaxLongrealExponent* = 308; (* maximal longreal exponent *)
  13. (* scanner constants *)
  14. EOT* = 0X; LF* = 0AX; CR* = 0DX; TAB* = 09X; ESC* = 1BX;
  15. TYPE
  16. StringType* = Strings.String;
  17. IdentifierType *= StringPool.Index;
  18. IdentifierString*= ARRAY MaxIdentifierLength+1 OF CHAR;
  19. CONST
  20. (** tokens *)
  21. (*
  22. note: order of tokens is important for the parser, do not modify without looking it up
  23. FoxProgTools.Enum --export --linefeed=6
  24. None
  25. (* RelationOps: Equal ... Is *)
  26. Equal DotEqual Unequal DotUnequal
  27. Less DotLess LessEqual DotLessEqual Greater DotGreater GreaterEqual DotGreaterEqual
  28. LessLessQ GreaterGreaterQ Questionmarks ExclamationMarks
  29. In Is
  30. (* MulOps: Times ... And *)
  31. Times TimesTimes DotTimes PlusTimes Slash Backslash DotSlash Div Mod And
  32. (* AddOps: Or ... Minus *)
  33. Or Plus Minus
  34. (* Prefix Unary Operators Plus ... Not *)
  35. Not
  36. (* expressions may start with Plus ... Identifier *)
  37. LeftParenthesis LeftBracket LeftBrace Number Character String Nil Imag True False Self Result New Identifier
  38. (* statementy may start with Self ... Begin *)
  39. If Case While Repeat For Loop With Exit Await Return Begin
  40. (* symbols, expressions and statements cannot start with *)
  41. Semicolon Transpose RightBrace RightBracket RightParenthesis
  42. Questionmark ExclamationMark
  43. LessLess GreaterGreater
  44. Upto Arrow Period Comma Colon Of Then Do To By Becomes Bar End Else Elsif Until Finally
  45. (* declaration elements *)
  46. Code Const Type Var Out Procedure Operator Import Definition Module Cell CellNet Extern
  47. (* composite type symbols *)
  48. Array Object Record Pointer Enum Port Address Size Alias
  49. (* assembler constants *)
  50. Ln PC PCOffset
  51. (* number types *)
  52. Shortint Integer Longint Hugeint Real Longreal
  53. Comment EndOfText Escape
  54. ~
  55. *)
  56. None*= 0;
  57. (* RelationOps: Equal ... Is *)
  58. Equal*= 1; DotEqual*= 2; Unequal*= 3; DotUnequal*= 4; Less*= 5; DotLess*= 6;
  59. LessEqual*= 7; DotLessEqual*= 8; Greater*= 9; DotGreater*= 10; GreaterEqual*= 11; DotGreaterEqual*= 12;
  60. LessLessQ*= 13; GreaterGreaterQ*= 14; Questionmarks*= 15; ExclamationMarks*= 16; In*= 17; Is*= 18;
  61. (* MulOps: Times ... And *)
  62. Times*= 19; TimesTimes*= 20; DotTimes*= 21; PlusTimes*= 22; Slash*= 23; Backslash*= 24;
  63. DotSlash*= 25; Div*= 26; Mod*= 27; And*= 28;
  64. (* AddOps: Or ... Minus *)
  65. Or*= 29; Plus*= 30; Minus*= 31;
  66. (* Prefix Unary Operators Plus ... Not *)
  67. Not*= 32;
  68. (* expressions may start with Plus ... Identifier *)
  69. LeftParenthesis*= 33; LeftBracket*= 34; LeftBrace*= 35; Number*= 36; Character*= 37; String*= 38;
  70. Nil*= 39; Imag*= 40; True*= 41; False*= 42; Self*= 43; Result*= 44;
  71. New*= 45; Identifier*= 46;
  72. (* statementy may start with Self ... Begin *)
  73. If*= 47; Case*= 48; While*= 49; Repeat*= 50; For*= 51; Loop*= 52;
  74. With*= 53; Exit*= 54; Await*= 55; Return*= 56; Begin*= 57;
  75. (* symbols, expressions and statements cannot start with *)
  76. Semicolon*= 58; Transpose*= 59; RightBrace*= 60; RightBracket*= 61; RightParenthesis*= 62; Questionmark*= 63;
  77. ExclamationMark*= 64; LessLess*= 65; GreaterGreater*= 66; Upto*= 67; Arrow*= 68; Period*= 69;
  78. Comma*= 70; Colon*= 71; Of*= 72; Then*= 73; Do*= 74; To*= 75;
  79. By*= 76; Becomes*= 77; Bar*= 78; End*= 79; Else*= 80; Elsif*= 81;
  80. Until*= 82; Finally*= 83;
  81. (* declaration elements *)
  82. Code*= 84; Const*= 85; Type*= 86; Var*= 87; Out*= 88; Procedure*= 89;
  83. Operator*= 90; Import*= 91; Definition*= 92; Module*= 93; Cell*= 94; CellNet*= 95;
  84. Extern*= 96;
  85. (* composite type symbols *)
  86. Array*= 97; Object*= 98; Record*= 99; Pointer*= 100; Enum*= 101; Port*= 102;
  87. Address*= 103; Size*= 104; Alias*= 105;
  88. (* assembler constants *)
  89. Ln*= 106; PC*= 107; PCOffset*= 108;
  90. (* number types *)
  91. Shortint*= 109; Integer*= 110; Longint*= 111; Hugeint*= 112; Real*= 113; Longreal*= 114;
  92. Comment*= 115; EndOfText*= 116; Escape*= 117;
  93. SingleQuote = 27X; DoubleQuote* = 22X;
  94. Ellipsis = 7FX; (* used in Scanner.GetNumber to return with ".." when reading an interval like 3..5 *)
  95. Uppercase*=0;
  96. Lowercase*=1;
  97. Unknown*=2;
  98. TYPE
  99. (* keywords book keeping *)
  100. Keyword* = ARRAY 32 OF CHAR;
  101. KeywordTable* = OBJECT(Basic.HashTableInt); (* string -> index *)
  102. VAR table: POINTER TO ARRAY OF LONGINT;
  103. PROCEDURE &InitTable*(size: LONGINT);
  104. VAR i: LONGINT;
  105. BEGIN
  106. Init(size); NEW(table,size); FOR i := 0 TO size-1 DO table[i] := -1; END;
  107. END InitTable;
  108. PROCEDURE IndexByIdentifier*(identifier: IdentifierType): LONGINT;
  109. VAR stringPoolIndex: LONGINT;
  110. BEGIN
  111. IF Has(identifier) THEN
  112. RETURN GetInt(identifier)
  113. ELSE (* do not modify index *)
  114. RETURN -1
  115. END;
  116. END IndexByIdentifier;
  117. PROCEDURE IndexByString*(CONST name: ARRAY OF CHAR): LONGINT;
  118. VAR stringPoolIndex: LONGINT;
  119. BEGIN
  120. StringPool.GetIndex(name,stringPoolIndex);
  121. IF Has(stringPoolIndex) THEN
  122. RETURN GetInt(stringPoolIndex)
  123. ELSE (* do not modify index *)
  124. RETURN -1
  125. END;
  126. END IndexByString;
  127. PROCEDURE IdentifierByIndex*(index: LONGINT; VAR identifier: IdentifierType);
  128. BEGIN
  129. identifier := table[index]
  130. END IdentifierByIndex;
  131. PROCEDURE StringByIndex*(index: LONGINT; VAR name: ARRAY OF CHAR);
  132. VAR stringPoolIndex: LONGINT;
  133. BEGIN
  134. stringPoolIndex := table[index];
  135. IF stringPoolIndex < 0 THEN
  136. name := ""
  137. ELSE
  138. StringPool.GetString(stringPoolIndex,name);
  139. END;
  140. END StringByIndex;
  141. PROCEDURE PutString*(CONST name: ARRAY OF CHAR; index: LONGINT);
  142. VAR stringPoolIndex: LONGINT;
  143. BEGIN
  144. StringPool.GetIndex(name,stringPoolIndex);
  145. table[index] := stringPoolIndex;
  146. PutInt(stringPoolIndex,index);
  147. END PutString;
  148. END KeywordTable;
  149. TYPE
  150. Token*=LONGINT;
  151. Position*= Basic.Position;
  152. (**
  153. symbol: data structure for the data transfer of the last read input from the scanner to the parser
  154. **)
  155. Symbol*= RECORD
  156. position*: Position;
  157. token*: Token; (* token of symbol *)
  158. identifier*: IdentifierType; (* identifier *)
  159. identifierString*: IdentifierString; (* cache of identifier's string *)
  160. string*: StringType; (* string or identifier *)
  161. stringLength*: LONGINT; (* length of string, if stringLength = 2 then this may be interpreted as character and integer = ORD(ch) *)
  162. numberType*: LONGINT; (* Integer, HugeInteger, Real or Longreal *)
  163. integer*: LONGINT;
  164. hugeint*: HUGEINT; (*! unify longint and hugeint *)
  165. character*: CHAR;
  166. real*: LONGREAL;
  167. END;
  168. StringMaker* = OBJECT (* taken from TF's scanner *)
  169. VAR length : LONGINT;
  170. data : StringType;
  171. PROCEDURE &Init*(initialSize : LONGINT);
  172. BEGIN
  173. IF initialSize < 256 THEN initialSize := 256 END;
  174. NEW(data, initialSize); length := 0;
  175. END Init;
  176. PROCEDURE Add*(CONST buf: ARRAY OF CHAR; ofs, len: LONGINT; propagate: BOOLEAN; VAR res: LONGINT);
  177. VAR i : LONGINT; n: StringType;
  178. BEGIN
  179. IF length + len + 1 >= LEN(data) THEN
  180. NEW(n, LEN(data) + len + 1); FOR i := 0 TO length - 1 DO n[i] := data[i] END;
  181. data := n
  182. END;
  183. WHILE len > 0 DO
  184. data[length] := buf[ofs];
  185. INC(ofs); INC(length); DEC(len)
  186. END;
  187. data[length] := 0X;
  188. END Add;
  189. (* remove last n characters *)
  190. PROCEDURE Shorten*(n : LONGINT);
  191. BEGIN
  192. DEC(length, n);
  193. IF length < 0 THEN length := 0 END;
  194. IF length > 0 THEN data[length - 1] := 0X ELSE data[length] := 0X END
  195. END Shorten;
  196. PROCEDURE Clear*;
  197. BEGIN
  198. data[0] := 0X;
  199. length := 0
  200. END Clear;
  201. PROCEDURE GetWriter*() : Streams.Writer;
  202. VAR w : Streams.Writer;
  203. BEGIN
  204. NEW(w, SELF.Add, 256);
  205. RETURN w
  206. END GetWriter;
  207. PROCEDURE GetReader*(): Streams.Reader;
  208. VAR r: Streams.StringReader;
  209. BEGIN
  210. NEW(r, 256);
  211. r.Set(data^);
  212. RETURN r
  213. END GetReader;
  214. PROCEDURE GetString*(VAR len: LONGINT) : StringType;
  215. BEGIN
  216. len := length;
  217. RETURN data
  218. END GetString;
  219. PROCEDURE GetStringCopy*(VAR len: LONGINT): StringType;
  220. VAR new: StringType;
  221. BEGIN
  222. len := length;
  223. NEW(new,len+1);
  224. COPY(data^,new^);
  225. RETURN new
  226. END GetStringCopy;
  227. END StringMaker;
  228. (** scanner reflects the following EBNF
  229. Symbol = String | Token | Number | Keyword | Identifier.
  230. Token = | '#' | '&' | '(' ['*' any '*' ')'] | ')' | '*'['*'] | '+'['*'] | ',' | '-' | '.' [ '.' | '*' | '/' | '=' | '#' | '>'['='] | '<' ['=']
  231. | '/' | ':' ['='] | ';' | '<' ['=' | '<' ['?'] ] | '=' | '>' [ '=' | '>' ['?']]
  232. | '[' | ']' | '^' | '{' | '|' | '}' | '~' | '\' | '`' | '?' ['?'] | '!' ['!']
  233. Identifier = Letter {Letter | Digit | '_'}.
  234. Letter = 'A' | 'B' | .. | 'Z' | 'a' | 'b' | .. | 'z'.
  235. Digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9' .
  236. String = '"' {Character} '"' | "'" {Character} "'".
  237. Character = Digit [HexDigit] 'X'.
  238. Number = Integer | Real.
  239. Integer = Digit {Digit} | Digit {HexDigit} 'H' | '0x' {HexDigit}.
  240. Real = Digit {Digit} '.' {Digit} [ScaleFactor].
  241. ScaleFactor = ('E' | 'D') ['+' | '-'] digit {digit}.
  242. HexDigit = Digit | 'A' | 'B' | 'C' | 'D' | 'E' | 'F'.
  243. **)
  244. Scanner* = OBJECT
  245. VAR
  246. (* helper state information *)
  247. source-: StringType;
  248. reader-: Streams.Reader; (* source *)
  249. diagnostics: Diagnostics.Diagnostics; (* error logging *)
  250. ch-: CHAR; (* look-ahead character *)
  251. position-: Position;
  252. (*
  253. position-: LONGINT; (* current position *)
  254. line-: LONGINT;
  255. *)
  256. error-: BOOLEAN; (* if error occured during scanning *)
  257. firstIdentifier: BOOLEAN; (* support of lower vs. upper case keywords *)
  258. case-: LONGINT;
  259. stringWriter: Streams.Writer;
  260. stringMaker: StringMaker;
  261. useLineNumbers*: BOOLEAN;
  262. (*
  263. source: name of the source code for reference in error outputs
  264. reader: input stream
  265. position: reference position (offset) of the input stream , for error output
  266. diagnostics: error output object
  267. *)
  268. PROCEDURE & InitializeScanner*( CONST source: ARRAY OF CHAR; reader: Streams.Reader; position: LONGINT; diagnostics: Diagnostics.Diagnostics );
  269. BEGIN
  270. NEW(stringMaker,1024);
  271. stringWriter := stringMaker.GetWriter();
  272. error := FALSE;
  273. NEW(SELF.source, Strings.Length(source)+1);
  274. COPY (source, SELF.source^);
  275. SELF.reader := reader;
  276. SELF.diagnostics := diagnostics;
  277. ch := " ";
  278. case := Unknown;
  279. firstIdentifier := TRUE;
  280. IF reader = NIL THEN ch := EOT ELSE GetNextCharacter END;
  281. IF Trace THEN D.Str( "New scanner " ); D.Ln; END;
  282. SELF.position.start := position;
  283. SELF.position.line := 1;
  284. SELF.position.linepos := 0;
  285. useLineNumbers := FALSE;
  286. END InitializeScanner;
  287. PROCEDURE ResetCase*; (*! needs a better naming ! *)
  288. BEGIN
  289. firstIdentifier := TRUE; case := Unknown;
  290. END ResetCase;
  291. PROCEDURE SetCase*(c: LONGINT);
  292. BEGIN
  293. case := c;
  294. END SetCase;
  295. (** report an error occured during scanning **)
  296. PROCEDURE ErrorS(CONST msg: ARRAY OF CHAR);
  297. BEGIN
  298. Basic.Error(diagnostics, source^, position, msg);
  299. error := TRUE;
  300. END ErrorS;
  301. (** report an error occured during scanning **)
  302. PROCEDURE Error( code: INTEGER );
  303. BEGIN
  304. Basic.ErrorC(diagnostics, source^, position, code, "");
  305. error := TRUE;
  306. END Error;
  307. (** get next character, end of text results in ch = EOT **)
  308. PROCEDURE GetNextCharacter*;
  309. BEGIN
  310. reader.Char(ch); INC(position.start);
  311. IF ch = LF THEN INC(position.line); position.linepos := position.start END;
  312. (*
  313. (* not necessary, as Streams returns 0X if reading failed, but in case Streams.Reader.Char is modified ... *)
  314. IF reader.res # Streams.Ok THEN ch := EOT END;
  315. *)
  316. END GetNextCharacter;
  317. (*
  318. The following is an implementation of the KMP algorithm used in order to traverse strings until some pattern occurs.
  319. It is not necessary for our implementation of string escape sequences, because the first character of the pattern does not occur in the pattern elsewhere
  320. I found the code useful and keep it here for the time being....
  321. (* generate a table to be able to quickly search for string containing overlaps - KMP algorithm *)
  322. PROCEDURE MakeOverlapTable*(CONST pattern: ARRAY OF CHAR; VAR table: ARRAY OF LONGINT);
  323. VAR i, cnd: LONGINT;
  324. BEGIN
  325. ASSERT(pattern[0] # 0X);
  326. (* if first character did not match: reset search *)
  327. table[0] := -1;
  328. (* if second character did not match: compare to first *)
  329. IF pattern[1] # 0X THEN
  330. table[1] := 0;
  331. END;
  332. (* for all other characters: switch back to previous overlay in pattern *)
  333. i := 2; cnd := 0;
  334. WHILE(pattern[i] # 0X) DO
  335. (* do patterns [i-cnd, i-1] match with pattern[0.. cnd] ? *)
  336. IF pattern[i-1] = pattern[cnd] THEN
  337. INC(cnd); table[i] := cnd; INC(i);
  338. (* no, switch back to last overlap, if possible *)
  339. ELSIF cnd > 0 THEN cnd := table[cnd]
  340. (* not possible: restart at beginning *)
  341. ELSE table[i] := 0; INC(i)
  342. END;
  343. END;
  344. END MakeOverlapTable;
  345. (* using KMP substring search algorithm consume and reproduce all characters of a string until endString *)
  346. PROCEDURE GetString(CONST endString: ARRAY OF CHAR);
  347. VAR escapePos: LONGINT; ech: CHAR; i: LONGINT; table: ARRAY 16 OF LONGINT;
  348. next: LONGINT;
  349. PROCEDURE Append(ch :CHAR);
  350. BEGIN
  351. IF ch = 0X THEN
  352. ErrorS("Unexpected end of text in string"); error := TRUE
  353. ELSE
  354. stringWriter.Char(ch)
  355. END;
  356. END Append;
  357. BEGIN
  358. MakeOverlapTable(endString, table);
  359. (* traverse *)
  360. escapePos := 0; ech := endString[0];
  361. GetNextCharacter;
  362. REPEAT
  363. IF ch = ech THEN
  364. INC(escapePos); ech := endString[escapePos];
  365. GetNextCharacter;
  366. ELSIF escapePos = 0 THEN (* frequent case *)
  367. Append(ch); GetNextCharacter;
  368. ELSE
  369. (* overlaps ? *)
  370. next := table[escapePos];
  371. IF next < 0 THEN next := 0 END;
  372. (* account for "forgotten" characters *)
  373. FOR i := 0 TO escapePos-1-next DO
  374. Append(endString[i]);
  375. END;
  376. (* to next overlapping ? *)
  377. escapePos := table[escapePos];
  378. (* no overlapping *)
  379. IF escapePos < 0 THEN
  380. Append(ch);
  381. escapePos := 0;
  382. GetNextCharacter;
  383. END;
  384. ech := endString[escapePos];
  385. END;
  386. UNTIL (ch = EOT) OR (ech = 0X);
  387. END GetString;
  388. *)
  389. (* simple case can be utilized when endString does not contain first character, which is the case for our string convention *)
  390. PROCEDURE ConsumeStringUntil(CONST endString: ARRAY OF CHAR; useControl: BOOLEAN);
  391. VAR escapePos: LONGINT; ech: CHAR; i: LONGINT; startPosition: LONGINT;
  392. CONST
  393. Control = '\';
  394. Delimiter = '"';
  395. PROCEDURE Append(ch :CHAR);
  396. BEGIN
  397. IF ch = 0X THEN
  398. ErrorS("Unexpected end of text in string"); error := TRUE;
  399. ELSE
  400. stringWriter.Char(ch)
  401. END;
  402. END Append;
  403. BEGIN
  404. (* traverse *)
  405. escapePos := 0; ech := endString[0]; startPosition := position.start;
  406. GetNextCharacter;
  407. REPEAT
  408. IF ch = ech THEN
  409. INC(escapePos); ech := endString[escapePos];
  410. GetNextCharacter;
  411. ELSIF useControl & (ch = Control) THEN
  412. GetNextCharacter;
  413. IF (ch = Control) OR (ch = Delimiter) THEN
  414. Append(ch)
  415. ELSIF ch = 'n' THEN
  416. Append(CR); Append(LF);
  417. ELSIF ch = 't' THEN
  418. Append(TAB)
  419. ELSE
  420. ErrorS("Unknown control sequence")
  421. END;
  422. GetNextCharacter
  423. ELSIF escapePos = 0 THEN (* frequent case *)
  424. Append(ch); GetNextCharacter;
  425. ELSE
  426. (* account for "forgotten" characters *)
  427. FOR i := 0 TO escapePos-1 DO
  428. Append(endString[i]);
  429. END;
  430. (* restart *)
  431. ech := endString[0]; escapePos := 0;
  432. END;
  433. UNTIL (ch = EOT) OR (ech = 0X) OR error;
  434. IF ch = EOT THEN position.start := startPosition; ErrorS("Unexpected end of text in string") END;
  435. END ConsumeStringUntil;
  436. PROCEDURE GetEscapedString(VAR symbol: Symbol);
  437. VAR endString: ARRAY 4 OF CHAR; escape: CHAR;
  438. BEGIN
  439. (* backslash already consumed *)
  440. stringMaker.Clear;
  441. IF ch = '"' THEN
  442. escape := 0X;
  443. ELSE
  444. escape := ch; GetNextCharacter;
  445. END;
  446. ASSERT((ch = '"') OR (ch = "'"));
  447. REPEAT
  448. IF escape # 0X THEN
  449. endString[0] := ch;
  450. endString[1] := escape;
  451. endString[2] := '\';
  452. endString[3] := 0X;
  453. ELSE
  454. endString[0] := ch;
  455. endString[1] := '\';
  456. endString[2] := 0X;
  457. END;
  458. ConsumeStringUntil(endString, escape = 0X);
  459. UNTIL TRUE;
  460. stringWriter.Char(0X);
  461. stringWriter.Update;
  462. symbol.string := stringMaker.GetStringCopy(symbol.stringLength);
  463. END GetEscapedString;
  464. (** get a string starting at current position
  465. string = {'"' {Character} '"'} | {"'" {Character} "'"}.
  466. **)
  467. (* multiline indicates that a string may occupy more than one lines, either concatenated or via multi-strings " " " "
  468. *)
  469. PROCEDURE GetString(VAR symbol: Symbol; multiLine, multiString, useControl: BOOLEAN);
  470. VAR och: CHAR; error: BOOLEAN; done: BOOLEAN;
  471. CONST control = '\';
  472. PROCEDURE Append(ch :CHAR);
  473. BEGIN
  474. IF ch = 0X THEN
  475. ErrorS("Unexpected end of text in string"); error := TRUE
  476. ELSE
  477. stringWriter.Char(ch)
  478. END;
  479. END Append;
  480. BEGIN
  481. stringMaker.Clear;
  482. och := ch; error := FALSE;
  483. REPEAT
  484. LOOP
  485. IF error THEN EXIT END;
  486. GetNextCharacter;
  487. IF (ch = och) OR (ch = EOT) THEN EXIT END;
  488. IF useControl & (ch = control) THEN
  489. GetNextCharacter;
  490. IF (ch = control) OR (ch = och) THEN
  491. Append(ch)
  492. ELSIF ch = 'n' THEN
  493. Append(CR); Append(LF);
  494. ELSIF ch = 't' THEN
  495. Append(TAB)
  496. ELSE
  497. ErrorS("Unknown control sequence")
  498. END;
  499. ELSE
  500. IF ~multiLine & (ch < " ") THEN Error( Basic.StringIllegalCharacter ); EXIT END;
  501. Append(ch)
  502. END;
  503. END;
  504. IF ch = EOT THEN
  505. ErrorS("Unexpected end of text in string")
  506. ELSE
  507. GetNextCharacter;
  508. IF multiString THEN SkipBlanks END;
  509. END;
  510. UNTIL ~multiString OR (ch # och);
  511. stringWriter.Char(0X);
  512. stringWriter.Update;
  513. symbol.string := stringMaker.GetStringCopy(symbol.stringLength);
  514. END GetString;
  515. (**
  516. Identifier = Letter {Letter | Digit | '_'} .
  517. Letter = 'A' | 'B' | .. | 'Z' | 'a' | 'b' | .. | 'z' .
  518. Digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9'.
  519. '_' is the underscore character
  520. **)
  521. PROCEDURE GetIdentifier( VAR symbol: Symbol );
  522. VAR i: LONGINT;
  523. BEGIN
  524. i := 0;
  525. REPEAT symbol.identifierString[i] := ch; INC( i ); GetNextCharacter UNTIL reservedCharacter[ORD( ch )] OR (i = MaxIdentifierLength);
  526. IF i = MaxIdentifierLength THEN Error( Basic.IdentifierTooLong ); DEC( i ) END;
  527. symbol.identifierString[i] := 0X;
  528. StringPool.GetIndex(symbol.identifierString, symbol.identifier);
  529. END GetIdentifier;
  530. (**
  531. Number = Integer | Real.
  532. Integer = Digit {Digit} | Digit {HexDigit} 'H' | '0x' {HexDigit}.
  533. Real = Digit {Digit} '.' {Digit} [ScaleFactor].
  534. ScaleFactor = ('E' | 'D') ['+' | '-'] digit {digit}.
  535. HexDigit = Digit | 'A' | 'B' | 'C' | 'D' | 'E' | 'F'.
  536. Digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9' .
  537. **)
  538. PROCEDURE GetNumber(VAR symbol: Symbol): Token;
  539. VAR i, nextInt, m, n, d, e, si: LONGINT;
  540. dig: ARRAY 24 OF CHAR;
  541. f: LONGREAL; expCh: CHAR; neg, long: BOOLEAN;
  542. result: Token;
  543. hugeint, tenh, number: HUGEINT;
  544. digits: LONGINT;
  545. (** 10^e **)
  546. PROCEDURE Ten( e: LONGINT ): LONGREAL;
  547. VAR x, p: LONGREAL;
  548. BEGIN
  549. x := 1; p := 10;
  550. WHILE e > 0 DO
  551. IF ODD( e ) THEN x := x * p END;
  552. e := e DIV 2;
  553. IF e > 0 THEN p := p * p END (* prevent overflow *)
  554. END;
  555. RETURN x
  556. END Ten;
  557. (** return decimal number associated to character ch , error if none **)
  558. PROCEDURE Decimal( ch: CHAR ): LONGINT;
  559. BEGIN (* ("0" <= ch) & (ch <= "9") OR ("A" <= ch) & (ch <= "F") *)
  560. IF ch <= "9" THEN RETURN ORD( ch ) - ORD( "0" ) ELSE Error( Basic.NumberIllegalCharacter ); RETURN 0 END
  561. END Decimal;
  562. (** return hexadecimal number associated to character ch, error if none **)
  563. PROCEDURE Hexadecimal( ch: CHAR ): LONGINT;
  564. BEGIN
  565. IF ch <= "9" THEN RETURN ORD( ch ) - ORD( "0" )
  566. ELSIF ch <= "F" THEN RETURN ORD( ch ) - ORD( "A" ) + 10
  567. ELSIF ch <= "f" THEN RETURN ORD( ch ) - ORD( "a" ) + 10
  568. ELSE Error( Basic.NumberIllegalCharacter ); RETURN 0
  569. END
  570. END Hexadecimal;
  571. BEGIN (* ("0" <= ch) & (ch <= "9") *)
  572. result := Number;
  573. i := 0; m := 0; n := 0; d := 0; si := 0; long := FALSE;
  574. IF (ch = "0") & (reader.Peek() = "x") THEN (* hex number *)
  575. digits := 0;
  576. GetNextCharacter; GetNextCharacter;
  577. WHILE (ch >= "0") & (ch <= "9") OR (ch >= "a") & (ch <="f") OR (ch >= "A") & (ch <= "F") DO
  578. number := number * 10H + Hexadecimal(ch);
  579. INC(digits);
  580. GetNextCharacter;
  581. END;
  582. symbol.hugeint := number;
  583. symbol.integer := SHORT(number);
  584. IF digits > MaxHexDigits THEN
  585. symbol.numberType := Hugeint
  586. ELSE
  587. symbol.numberType := Integer
  588. END;
  589. RETURN result;
  590. END;
  591. LOOP (* read mantissa *)
  592. IF ("0" <= ch) & (ch <= "9") OR (d = 0) & ("A" <= ch) & (ch <= "F") THEN
  593. IF (m > 0) OR (ch # "0") THEN (* ignore leading zeros *)
  594. IF n < LEN( dig ) THEN dig[n] := ch; INC( n ) END;
  595. INC( m )
  596. END;
  597. symbol.identifierString[si] := ch; INC( si ); GetNextCharacter; INC( i )
  598. ELSIF ch = "." THEN
  599. symbol.identifierString[si] := ch; INC( si ); GetNextCharacter;
  600. IF ch = "." THEN ch := Ellipsis; EXIT
  601. ELSIF d = 0 THEN (* i > 0 *) d := i
  602. ELSE Error( Basic.NumberIllegalCharacter )
  603. END
  604. ELSE EXIT
  605. END
  606. END; (* 0 <= n <= m <= i, 0 <= d <= i *)
  607. IF d = 0 THEN (* integer *)
  608. IF n = m THEN
  609. symbol.integer := 0; i := 0; symbol.hugeint := 0;
  610. IF ch = "X" THEN (* character *)
  611. symbol.identifierString[si] := ch; INC( si ); GetNextCharacter; result := Character;
  612. IF (n <= 2) THEN
  613. WHILE i < n DO symbol.integer := symbol.integer * 10H + Hexadecimal( dig[i] ); INC( i ) END;
  614. symbol.character := CHR(symbol.integer);
  615. ELSE Error( Basic.NumberTooLarge )
  616. END
  617. ELSIF ch = "H" THEN (* hexadecimal *)
  618. symbol.identifierString[si] := ch; INC( si ); GetNextCharacter;
  619. IF (n < MaxHexDigits) OR (n=MaxHexDigits) & (dig[0] <= "7") THEN (* otherwise the positive (!) number is not in the range of longints *)
  620. symbol.numberType := Integer;
  621. (* IF (n = MaxHexDigits) & (dig[0] > "7") THEN (* prevent overflow *) symbol.integer := -1 END; *)
  622. WHILE i < n DO symbol.integer := symbol.integer * 10H + Hexadecimal( dig[i] ); INC( i ) END;
  623. symbol.hugeint := symbol.integer;
  624. ELSIF n <= MaxHugeHexDigits THEN
  625. symbol.numberType := Hugeint;
  626. IF (n = MaxHugeHexDigits) & (dig[0] > "7") THEN (* prevent overflow *) symbol.hugeint := -1 END;
  627. WHILE i < n DO symbol.hugeint := Hexadecimal( dig[i] ) + symbol.hugeint * 10H; INC( i ) END;
  628. symbol.integer :=SHORT(symbol.hugeint);
  629. ELSE
  630. symbol.numberType := Hugeint; (* to make parser able to go on *)
  631. Error( Basic.NumberTooLarge )
  632. END
  633. ELSE (* decimal *)
  634. symbol.numberType := Integer;
  635. WHILE (i < n) & ~long DO
  636. d := Decimal( dig[i] ); INC( i );
  637. IF symbol.integer >= MAX(LONGINT) DIV 10 THEN (* multiplication overflow *)long := TRUE END;
  638. nextInt := symbol.integer*10+d;
  639. IF nextInt >=0 THEN symbol.integer := nextInt ELSE (* overflow *) long := TRUE END;
  640. END;
  641. IF long THEN
  642. i := 0; (* restart computation , artificial limit because of compiler problems with hugeint *)
  643. hugeint := 0;
  644. tenh := 10; (* compiler does not like constants here ! *)
  645. symbol.numberType := Hugeint;
  646. WHILE i < n DO
  647. d := Decimal( dig[i] ); INC( i );
  648. IF hugeint >= MAX(HUGEINT) DIV 10 THEN Error( Basic.NumberTooLarge) END;
  649. hugeint := hugeint * tenh + d;
  650. IF hugeint < 0 THEN Error( Basic.NumberTooLarge ) END
  651. END;
  652. symbol.hugeint := hugeint;
  653. symbol.integer := SHORT(symbol.hugeint);
  654. ELSE
  655. symbol.hugeint := symbol.integer;
  656. END
  657. END
  658. ELSE
  659. symbol.numberType := Hugeint;
  660. Error( Basic.NumberTooLarge )
  661. END
  662. ELSE (* fraction *)
  663. f := 0; e := 0; expCh := "E";
  664. WHILE n > 0 DO (* 0 <= f < 1 *) DEC( n ); f := (Decimal( dig[n] ) + f) / 10 END;
  665. IF (ch = "E") OR (ch = "D") THEN
  666. expCh := ch; symbol.identifierString[si] := ch; INC( si ); GetNextCharacter; neg := FALSE;
  667. IF ch = "-" THEN neg := TRUE; symbol.identifierString[si] := ch; INC( si ); GetNextCharacter
  668. ELSIF ch = "+" THEN symbol.identifierString[si] := ch; INC( si ); GetNextCharacter
  669. END;
  670. IF ("0" <= ch) & (ch <= "9") THEN
  671. REPEAT
  672. n := Decimal( ch ); symbol.identifierString[si] := ch; INC( si ); GetNextCharacter;
  673. IF e <= (MAX( INTEGER ) - n) DIV 10 THEN e := e * 10 + n ELSE Error( Basic.NumberTooLarge ) END
  674. UNTIL (ch < "0") OR ("9" < ch);
  675. IF neg THEN e := -e END
  676. ELSE Error( Basic.NumberIllegalCharacter )
  677. END
  678. END;
  679. DEC( e, i - d - m ); (* decimal point shift *)
  680. IF expCh = "E" THEN
  681. symbol.numberType := Real;
  682. IF (1 - MaxRealExponent < e) & (e <= MaxRealExponent) THEN
  683. IF e < 0 THEN symbol.real := f / Ten( -e ) ELSE symbol.real := f * Ten( e ) END
  684. ELSE Error( Basic.NumberTooLarge )
  685. END
  686. ELSE
  687. symbol.numberType := Longreal;
  688. IF (1 - MaxLongrealExponent < e) & (e <= MaxLongrealExponent) THEN
  689. IF e < 0 THEN symbol.real := f / Ten( -e ) ELSE symbol.real := f * Ten( e ) END
  690. ELSE Error( Basic.NumberTooLarge )
  691. END
  692. END
  693. END;
  694. symbol.identifierString[si] := 0X;
  695. RETURN result;
  696. END GetNumber;
  697. (** read / skip a comment **)
  698. PROCEDURE ReadComment(VAR symbol: Symbol);
  699. VAR level: LONGINT;
  700. BEGIN
  701. stringMaker.Clear;
  702. level := 1;
  703. WHILE (level > 0) & (ch # EOT) DO
  704. IF ch = "(" THEN
  705. stringWriter.Char(ch);
  706. GetNextCharacter;
  707. IF ch = "*" THEN INC(level); stringWriter.Char(ch); GetNextCharacter; END;
  708. ELSIF ch = "*" THEN
  709. stringWriter.Char(ch);
  710. GetNextCharacter;
  711. IF ch =")" THEN DEC(level); stringWriter.Char(ch); GetNextCharacter; END;
  712. ELSE
  713. stringWriter.Char(ch);
  714. GetNextCharacter;
  715. END;
  716. END;
  717. IF level > 0 THEN
  718. Error(Basic.CommentNotClosed)
  719. END;
  720. stringWriter.Char(0X);
  721. stringWriter.Update;
  722. stringMaker.Shorten(2); (* remove comment closing *)
  723. symbol.token := Comment;
  724. symbol.string := stringMaker.GetString(symbol.stringLength);
  725. END ReadComment;
  726. PROCEDURE SkipToEndOfCode*(VAR startPos,endPos: LONGINT; VAR symbol: Symbol): Token;
  727. VAR s: LONGINT;
  728. BEGIN
  729. ASSERT(case # Unknown);
  730. stringMaker.Clear;
  731. startPos := symbol.position.end;
  732. IF useLineNumbers THEN startPos := position.line END;
  733. s := symbol.token;
  734. WHILE (s # EndOfText) & (s # End) & (s # With) DO
  735. symbol.position := position;
  736. endPos := position.start;
  737. CASE ch OF
  738. 'A' .. 'Z','a'..'z': s := Identifier;
  739. GetIdentifier(symbol);
  740. IF (case=Uppercase) & (symbol.identifierString = "END") OR (case=Lowercase) & (symbol.identifierString = "end") THEN
  741. s := End
  742. ELSIF (case = Uppercase) & (symbol.identifierString = "WITH") OR (case = Lowercase) & (symbol.identifierString = "with") THEN
  743. s := With
  744. ELSE
  745. stringWriter.String(symbol.identifierString);
  746. END;
  747. ELSE
  748. stringWriter.Char(ch);
  749. GetNextCharacter;
  750. END;
  751. symbol.position.end := position.start;
  752. END;
  753. stringWriter.Update;
  754. symbol.string := stringMaker.GetStringCopy(symbol.stringLength);
  755. symbol.token := s;
  756. IF Trace THEN
  757. D.String("skip to end: "); D.Int(startPos,1); D.String(","); D.Int(endPos,1); D.Ln;
  758. OutSymbol(D.Log,symbol); D.Ln;
  759. END;
  760. RETURN s
  761. END SkipToEndOfCode;
  762. PROCEDURE SkipBlanks;
  763. BEGIN
  764. WHILE (ch <= " ") & (ch # ESC) DO (*ignore control characters*)
  765. IF ch = EOT THEN
  766. IF Trace THEN D.String("EOT"); D.Ln; END;
  767. RETURN
  768. ELSE GetNextCharacter
  769. END
  770. END;
  771. END SkipBlanks;
  772. (** get next symbol **)
  773. PROCEDURE GetNextSymbol*(VAR symbol: Symbol ): BOOLEAN;
  774. VAR s,token: LONGINT;
  775. BEGIN
  776. SkipBlanks;
  777. symbol.position := position;
  778. (*
  779. IF useLineNumbers THEN
  780. symbol.position.start := position.line+1;
  781. ELSE
  782. symbol.position.start := position.start
  783. END;
  784. symbol.position.line := position.line;
  785. *)
  786. stringMaker.Clear;
  787. CASE ch OF (* ch > " " *)
  788. EOT: s := EndOfText
  789. |ESC: s := Escape;; GetNextCharacter
  790. | DoubleQuote:
  791. s := String; GetString(symbol,TRUE, TRUE, FALSE);
  792. | SingleQuote:
  793. s := String; GetString(symbol,FALSE, FALSE,FALSE);
  794. (* to be replaced by:
  795. s := Character; GetString(symbol);
  796. IF symbol.stringLength #2 THEN (* stringlength = 1 for empty string '' *)
  797. Error(Basic.IllegalCharacterValue)
  798. END;
  799. *)
  800. | '#': s := Unequal; GetNextCharacter
  801. | '&': s := And; GetNextCharacter
  802. | '(': GetNextCharacter;
  803. IF ch = '*' THEN GetNextCharacter; ReadComment(symbol); s := Comment; ELSE s := LeftParenthesis END
  804. | ')': s := RightParenthesis; GetNextCharacter
  805. | '*': GetNextCharacter; IF ch = '*' THEN GetNextCharacter; s := TimesTimes ELSE s := Times END
  806. | '+': GetNextCharacter; IF ch = '*' THEN GetNextCharacter; s := PlusTimes ELSE s := Plus END
  807. | ',': s := Comma; GetNextCharacter
  808. | '-': s := Minus; GetNextCharacter
  809. | '.': GetNextCharacter;
  810. IF ch = '.' THEN GetNextCharacter; s := Upto;
  811. ELSIF ch = '*' THEN GetNextCharacter; s := DotTimes;
  812. ELSIF ch = '/' THEN GetNextCharacter; s := DotSlash;
  813. ELSIF ch='=' THEN GetNextCharacter; s := DotEqual;
  814. ELSIF ch='#' THEN GetNextCharacter; s := DotUnequal;
  815. ELSIF ch='>' THEN GetNextCharacter;
  816. IF ch='=' THEN s := DotGreaterEqual; GetNextCharacter
  817. ELSE s := DotGreater;
  818. END
  819. ELSIF ch='<' THEN GetNextCharacter;
  820. IF ch='=' THEN s := DotLessEqual; GetNextCharacter
  821. ELSE s := DotLess;
  822. END
  823. ELSE s := Period END
  824. | '/': s := Slash; GetNextCharacter
  825. | '0'..'9': s := GetNumber(symbol);
  826. | ':': GetNextCharacter;
  827. IF ch = '=' THEN GetNextCharacter; s := Becomes ELSE s := Colon END
  828. | ';': s := Semicolon; GetNextCharacter
  829. | '<': GetNextCharacter;
  830. IF ch = '=' THEN GetNextCharacter; s := LessEqual
  831. ELSIF ch ='<' THEN GetNextCharacter;
  832. IF ch ='?' THEN GetNextCharacter; s := LessLessQ
  833. ELSE s := LessLess
  834. END;
  835. ELSE s := Less;
  836. END
  837. | '=': s := Equal; GetNextCharacter
  838. | '>': GetNextCharacter;
  839. IF ch = '=' THEN GetNextCharacter; s := GreaterEqual
  840. ELSIF ch ='>' THEN GetNextCharacter;
  841. IF ch ='?' THEN GetNextCharacter; s := GreaterGreaterQ
  842. ELSE s := GreaterGreater
  843. END;
  844. ELSE s := Greater; END
  845. | '[': s := LeftBracket; GetNextCharacter
  846. | ']': s := RightBracket; GetNextCharacter
  847. | '^': s := Arrow; GetNextCharacter
  848. | '{': s := LeftBrace; GetNextCharacter
  849. | '|': s := Bar; GetNextCharacter
  850. | '}': s := RightBrace; GetNextCharacter
  851. | '~': s := Not; GetNextCharacter
  852. | '\': s := Backslash; GetNextCharacter;
  853. IF ch = DoubleQuote THEN
  854. s := String;
  855. GetEscapedString(symbol);
  856. (*
  857. GetString(symbol, TRUE, TRUE, TRUE)
  858. *)
  859. ELSIF (ch > " ") & (reader.Peek() = DoubleQuote) THEN
  860. s := String;
  861. GetEscapedString(symbol);
  862. END;
  863. | '`': s := Transpose; GetNextCharacter
  864. | '?': s := Questionmark; GetNextCharacter; IF ch = '?' THEN s := Questionmarks; GetNextCharacter END;
  865. | '!': s := ExclamationMark; GetNextCharacter; IF ch = '!' THEN s := ExclamationMarks; GetNextCharacter END;
  866. | Ellipsis:
  867. s := Upto; GetNextCharacter
  868. | 'A'..'Z': s := Identifier; GetIdentifier( symbol );
  869. IF (case=Uppercase) OR (case=Unknown) THEN
  870. token := keywordsUpper.IndexByIdentifier(symbol.identifier);
  871. IF (token >= 0) THEN s := token END;
  872. IF (s = Module) OR (s=CellNet) THEN case := Uppercase END;
  873. END;
  874. | 'a'..'z': s := Identifier; GetIdentifier( symbol);
  875. IF (case = Lowercase) OR (case=Unknown) THEN
  876. token := keywordsLower.IndexByIdentifier(symbol.identifier);
  877. IF (token >= 0) THEN s := token END;
  878. IF (s = Module) OR (s=CellNet) THEN case := Lowercase END;
  879. END;
  880. IF firstIdentifier & (s # Module) & (s # CellNet) & (case = Unknown) THEN case := Uppercase; s := Identifier END;
  881. ELSE s := Identifier; GetIdentifier( symbol );
  882. END;
  883. firstIdentifier := FALSE;
  884. symbol.token := s;
  885. symbol.position.end := position.start;
  886. IF Trace THEN OutSymbol(D.Log,symbol); D.Ln; END;
  887. RETURN ~error
  888. END GetNextSymbol;
  889. PROCEDURE ResetError*();
  890. BEGIN error := FALSE
  891. END ResetError;
  892. (** set the diagnostics mode of the scanner (diagnostics = NIL ==> no report) and reset the error state
  893. intended for silent symbol peeeking after the end of a module *)
  894. PROCEDURE ResetErrorDiagnostics*(VAR diagnostics: Diagnostics.Diagnostics);
  895. VAR b: BOOLEAN; d: Diagnostics.Diagnostics;
  896. BEGIN
  897. error := FALSE;
  898. d := SELF.diagnostics; SELF.diagnostics := diagnostics; diagnostics := d;
  899. END ResetErrorDiagnostics;
  900. END Scanner;
  901. Context*=RECORD
  902. position: Position;
  903. readerPosition : LONGINT;
  904. ch: CHAR;
  905. END;
  906. (** assembler scanner reflects the following EBNF
  907. Symbol = String | Token | Number | Identifier.
  908. Token = '\' | '#' | '(' ['*' any '*' ')'] | ')' | CR [LF] | LF | '*' | '+' | ',' | '-' | '~' | '.' | '/' | '%' | ':' | ';' | '=' | '[' | ']' | '{' | '}' | '!' | '^' | '$'['$'].
  909. String = '"' {Character} '"' | "'" {Character} "'".
  910. Identifier = '@' | Letter {'@' | '.' | Letter | Digit | '_'} .
  911. Letter = 'A' | 'B' | .. | 'Z' | 'a' | 'b' | .. | 'z' .
  912. Digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9'.
  913. Number = Integer | Real.
  914. Character = Digit [HexDigit] 'X'.
  915. Integer = Digit {Digit} | Digit {HexDigit} 'H' | '0x' {HexDigit}.
  916. Real = Digit {Digit} '.' {Digit} [ScaleFactor].
  917. ScaleFactor = ('E' | 'D') ['+' | '-'] digit {digit}.
  918. HexDigit = Digit | 'A' | 'B' | 'C' | 'D' | 'E' | 'F'.
  919. **)
  920. AssemblerScanner* = OBJECT (Scanner) (*! move to different module? unify with compiler scanner? *)
  921. VAR
  922. startContext-: Context;
  923. PROCEDURE &InitAssemblerScanner*( CONST source: ARRAY OF CHAR; reader: Streams.Reader; position: LONGINT; diagnostics: Diagnostics.Diagnostics );
  924. BEGIN
  925. InitializeScanner(source,reader,position,diagnostics);
  926. GetContext(startContext);
  927. END InitAssemblerScanner;
  928. PROCEDURE GetContext*(VAR context: Context);
  929. BEGIN
  930. context.ch := ch;
  931. context.position := position;
  932. context.readerPosition := reader.Pos();
  933. END GetContext;
  934. PROCEDURE SetContext*(CONST context: Context);
  935. BEGIN
  936. reader.SetPos(context.readerPosition);
  937. ch := context.ch;
  938. position := context.position;
  939. END SetContext;
  940. PROCEDURE SkipToEndOfLine*;
  941. BEGIN
  942. WHILE (ch # EOT) & (ch # CR) & (ch # LF) DO
  943. GetNextCharacter
  944. END;
  945. END SkipToEndOfLine;
  946. (**
  947. note: in contrast to a regular identifier, an assembler scanner identifier may also contain periods and the '@'-symbol
  948. Identifier = '@' | Letter {'@' | '.' | Letter | Digit | '_'} .
  949. Letter = 'A' | 'B' | .. | 'Z' | 'a' | 'b' | .. | 'z' .
  950. Digit = '0' | '1' | '2' | '3' | '4' | '5' | '6' | '7' | '8' | '9'.
  951. '_' is the underscore character
  952. **)
  953. PROCEDURE GetIdentifier( VAR symbol: Symbol );
  954. VAR
  955. i: LONGINT;
  956. PROCEDURE CharacterIsAllowed(character: CHAR): BOOLEAN;
  957. BEGIN
  958. CASE character OF
  959. | 'a' .. 'z', 'A' .. 'Z', '0' .. '9', '@', '.', '_': RETURN TRUE
  960. ELSE RETURN FALSE
  961. END;
  962. END CharacterIsAllowed;
  963. BEGIN
  964. i := 0;
  965. REPEAT
  966. symbol.identifierString[i] := ch; INC( i ); GetNextCharacter
  967. UNTIL ~CharacterIsAllowed(ch) OR (i = MaxIdentifierLength);
  968. IF i = MaxIdentifierLength THEN Error( Basic.IdentifierTooLong ); DEC( i ) END;
  969. symbol.identifierString[i] := 0X;
  970. END GetIdentifier;
  971. PROCEDURE GetNumber(VAR symbol: Symbol): Token;
  972. VAR number: HUGEINT; result: Token; digits: LONGINT;
  973. (** return hexadecimal number associated to character ch, error if none **)
  974. PROCEDURE Hexadecimal( ch: CHAR ): LONGINT;
  975. BEGIN
  976. IF (ch >= "0") & (ch <= "9") THEN RETURN ORD( ch ) - ORD( "0" )
  977. ELSIF (ch >= "a") & (ch <= "f") THEN RETURN ORD( ch ) - ORD( "a" ) + 10
  978. ELSE Error( Basic.NumberIllegalCharacter ); RETURN 0
  979. END
  980. END Hexadecimal;
  981. BEGIN
  982. result := Number;
  983. IF (ch = "0") THEN
  984. IF reader.Peek() = "x" THEN (* hex number *)
  985. digits := 0;
  986. GetNextCharacter; GetNextCharacter;
  987. WHILE (ch >= "0") & (ch <= "9") OR (ch >= "a") & (ch <="f") DO
  988. number := number * 10H + Hexadecimal(ch);
  989. INC(digits);
  990. GetNextCharacter;
  991. END;
  992. symbol.hugeint := number;
  993. symbol.integer := SHORT(number);
  994. IF digits > MaxHexDigits THEN
  995. symbol.numberType := Hugeint
  996. ELSE
  997. symbol.numberType := Integer
  998. END;
  999. ELSIF reader.Peek() = "b" THEN (* binary number *)
  1000. digits := 0;
  1001. GetNextCharacter; GetNextCharacter;
  1002. WHILE (ch >= "0") & (ch <= "1") DO
  1003. number := number * 2;
  1004. INC(digits);
  1005. IF ch = "1" THEN INC(number) END;
  1006. GetNextCharacter;
  1007. END;
  1008. symbol.hugeint := number;
  1009. symbol.integer := SHORT(number);
  1010. IF digits > 32 THEN
  1011. symbol.numberType := Hugeint
  1012. ELSE
  1013. symbol.numberType := Integer
  1014. END;
  1015. ELSE RETURN GetNumber^(symbol)
  1016. END;
  1017. ELSE RETURN GetNumber^(symbol)
  1018. END;
  1019. RETURN result
  1020. END GetNumber;
  1021. (** get next symbol **)
  1022. PROCEDURE GetNextSymbol*(VAR symbol: Symbol ): BOOLEAN;
  1023. VAR s: LONGINT;
  1024. PROCEDURE SkipBlanks;
  1025. BEGIN
  1026. WHILE (ch <= ' ') & (ch # CR) & (ch # LF) & (ch # EOT) DO (* ignore control characters except line feeds *)
  1027. GetNextCharacter
  1028. END;
  1029. END SkipBlanks;
  1030. BEGIN
  1031. REPEAT
  1032. SkipBlanks;
  1033. symbol.position := position;
  1034. (*
  1035. IF useLineNumbers THEN
  1036. symbol.position.start := position.line+1;
  1037. ELSE
  1038. symbol.position.start := position.start;
  1039. END;
  1040. symbol.position.line := position.line;
  1041. *)
  1042. CASE ch OF (* ch > ' ' *)
  1043. | EOT: s := EndOfText;
  1044. | DoubleQuote:
  1045. s := String; GetString(symbol, TRUE, FALSE, TRUE);
  1046. | SingleQuote:
  1047. s := Character; GetString(symbol, FALSE, FALSE, FALSE); symbol.character := symbol.string[0];
  1048. IF symbol.stringLength #2 THEN (* stringlength = 1 for empty string '' *)
  1049. Error(Basic.IllegalCharacterValue)
  1050. END;
  1051. | '\': s := Backslash; GetNextCharacter;
  1052. IF ch = DoubleQuote THEN s := String; GetString(symbol, FALSE, FALSE, TRUE) END;
  1053. | '#': s := Unequal; GetNextCharacter; (* for the ARM assembler *)
  1054. | '(': GetNextCharacter;
  1055. IF ch = '*' THEN GetNextCharacter; ReadComment(symbol); s := Comment; ELSE s := LeftParenthesis END
  1056. | ')': s := RightParenthesis; GetNextCharacter
  1057. | CR: GetNextCharacter; s := Ln;IF ch = LF THEN GetNextCharacter END;
  1058. | LF: GetNextCharacter; s := Ln; IF ch = CR THEN GetNextCharacter END;
  1059. | '*': s := Times; GetNextCharacter;
  1060. | '+': s := Plus ; GetNextCharacter;
  1061. | ',': s := Comma; GetNextCharacter
  1062. | '-': s := Minus; GetNextCharacter
  1063. | '~': s := Not; GetNextCharacter
  1064. | '.': s:= Period; GetNextCharacter
  1065. | '/': s := Div; GetNextCharacter
  1066. | '%': s := Mod; GetNextCharacter
  1067. | '0'..'9': s := GetNumber(symbol);
  1068. | ':': s := Colon; GetNextCharacter;
  1069. | ';': s := Comment; SkipToEndOfLine;
  1070. | '=': s := Equal; GetNextCharacter
  1071. | '[': s := LeftBracket; GetNextCharacter
  1072. | ']': s := RightBracket; GetNextCharacter
  1073. | '{': s := LeftBrace; GetNextCharacter
  1074. | '}': s := RightBrace; GetNextCharacter
  1075. | '!': s := ExclamationMark; GetNextCharacter;
  1076. | '^': s := Arrow; GetNextCharacter;
  1077. | 'A'..'Z': s := Identifier; GetIdentifier( symbol );
  1078. | 'a'..'z': s := Identifier; GetIdentifier( symbol);
  1079. | '@': s := Identifier; GetIdentifier( symbol); (* the '@'-symbol initiates an assembly scanner identifier *)
  1080. | '$': GetNextCharacter;
  1081. IF ch = '$' THEN s := PCOffset; GetNextCharacter ELSE s := PC; END
  1082. ELSE s := None; GetNextCharacter;
  1083. END;
  1084. symbol.position.end := position.start;
  1085. UNTIL s # Comment;
  1086. symbol.token := s;
  1087. IF Trace THEN D.Ln; D.Str( "Scan at " ); D.Int( symbol.position.start,1 ); D.Str( ": " ); OutSymbol(D.Log,symbol); D.Update; END;
  1088. RETURN ~error
  1089. END GetNextSymbol;
  1090. END AssemblerScanner;
  1091. VAR
  1092. reservedCharacter: ARRAY 256 OF BOOLEAN;
  1093. tokens-: ARRAY EndOfText+1 OF Keyword;
  1094. keywordsLower, keywordsUpper: KeywordTable;
  1095. (** return a new scanner on a stream, error output via diagnostics **)
  1096. PROCEDURE NewScanner*( CONST source: ARRAY OF CHAR; reader: Streams.Reader; position: LONGINT; diagnostics: Diagnostics.Diagnostics ): Scanner;
  1097. VAR s: Scanner;
  1098. BEGIN
  1099. NEW( s, source, reader, position, diagnostics ); RETURN s;
  1100. END NewScanner;
  1101. PROCEDURE NewAssemblerScanner*( CONST source: ARRAY OF CHAR; reader: Streams.Reader; position: LONGINT; diagnostics: Diagnostics.Diagnostics ): AssemblerScanner;
  1102. VAR s: AssemblerScanner;
  1103. BEGIN
  1104. NEW( s, source, reader, position, diagnostics ); RETURN s;
  1105. END NewAssemblerScanner;
  1106. PROCEDURE SymbolToString*(CONST symbol: Symbol; case: LONGINT; VAR str: ARRAY OF CHAR);
  1107. VAR id: StringPool.Index;
  1108. BEGIN
  1109. CASE symbol.token OF
  1110. Identifier, Number: COPY(symbol.identifierString, str)
  1111. | String, Comment: ASSERT(LEN(str) >= LEN(symbol.string^)); COPY(symbol.string^, str);
  1112. ELSE
  1113. GetKeyword(case, symbol.token, id);
  1114. IF id < 0 THEN str := "" ELSE StringPool.GetString(id, str) END;
  1115. END;
  1116. END SymbolToString;
  1117. (** debugging output **)
  1118. PROCEDURE OutSymbol*(w: Streams.Writer; CONST symbol: Symbol);
  1119. VAR str: ARRAY 256 OF CHAR;
  1120. BEGIN
  1121. w.Int(symbol.position.start,1); w.String("-");w.Int(symbol.position.end,1); w.String(":");
  1122. w.String(tokens[symbol.token]);
  1123. IF symbol.token= Number THEN
  1124. CASE symbol.numberType OF
  1125. Integer: w.String("(integer)")
  1126. |Hugeint: w.String("(hugeint)")
  1127. |Real: w.String("(real)")
  1128. |Longreal: w.String("(longreal)")
  1129. END;
  1130. END;
  1131. IF symbol.token = String THEN
  1132. w.String(":"); w.Char('"'); w.String(symbol.string^); w.Char('"');
  1133. ELSIF symbol.token = Comment THEN
  1134. w.String("(*"); w.String(symbol.string^); w.String("*)");
  1135. ELSE
  1136. SymbolToString(symbol, Uppercase, str); w.String(": "); w.String(str);
  1137. END
  1138. END OutSymbol;
  1139. (** reserved characters are the characters that may not occur within an identifier **)
  1140. PROCEDURE InitReservedCharacters;
  1141. VAR i: LONGINT;
  1142. BEGIN
  1143. FOR i := 0 TO LEN( reservedCharacter ) - 1 DO
  1144. CASE CHR(i) OF
  1145. | 'a' .. 'z', 'A' .. 'Z': reservedCharacter[i] := FALSE;
  1146. | '0'..'9': reservedCharacter[i] := FALSE;
  1147. | '_': reservedCharacter[i] := FALSE
  1148. ELSE
  1149. reservedCharacter[i] := TRUE
  1150. END;
  1151. END;
  1152. END InitReservedCharacters;
  1153. (* get keyword by token *)
  1154. PROCEDURE GetKeyword*(case:LONGINT; token: LONGINT; VAR identifier: IdentifierType);
  1155. BEGIN
  1156. IF case = Uppercase THEN
  1157. keywordsUpper.IdentifierByIndex(token,identifier);
  1158. ELSE ASSERT(case=Lowercase);
  1159. keywordsLower.IdentifierByIndex(token,identifier);
  1160. END;
  1161. END GetKeyword;
  1162. PROCEDURE InitTokens;
  1163. VAR i: LONGINT;
  1164. BEGIN
  1165. tokens[None] := "None";
  1166. tokens[Equal] := "Equal";
  1167. tokens[DotEqual] := "DotEqual";
  1168. tokens[Unequal] := "Unequal";
  1169. tokens[DotUnequal] := "DotUnequal";
  1170. tokens[Less] := "Less";
  1171. tokens[DotLess] := "DotLess";
  1172. tokens[LessEqual] := "LessEqual";
  1173. tokens[DotLessEqual] := "DotLessEqual";
  1174. tokens[Greater] := "Greater";
  1175. tokens[DotGreater] := "DotGreater";
  1176. tokens[GreaterEqual] := "GreaterEqual";
  1177. tokens[DotGreaterEqual] := "DotGreaterEqual";
  1178. tokens[LessLessQ] := "LessLessQ";
  1179. tokens[GreaterGreaterQ] := "GreaterGreaterQ";
  1180. tokens[In] := "In";
  1181. tokens[Is] := "Is";
  1182. tokens[Times] := "Times";
  1183. tokens[TimesTimes] := "TimesTimes";
  1184. tokens[DotTimes] := "DotTimes";
  1185. tokens[PlusTimes] := "PlusTimes";
  1186. tokens[Slash] := "Slash";
  1187. tokens[Backslash] := "Backslash";
  1188. tokens[DotSlash] := "DotSlash";
  1189. tokens[Div] := "Div";
  1190. tokens[Mod] := "Mod";
  1191. tokens[And] := "And";
  1192. tokens[Or] := "Or";
  1193. tokens[Plus] := "Plus";
  1194. tokens[Minus] := "Minus";
  1195. tokens[Not] := "Not";
  1196. tokens[LeftParenthesis] := "LeftParenthesis";
  1197. tokens[LeftBracket] := "LeftBracket";
  1198. tokens[LeftBrace] := "LeftBrace";
  1199. tokens[Number] := "Number";
  1200. tokens[Character] := "Character";
  1201. tokens[String] := "String";
  1202. tokens[Nil] := "Nil";
  1203. tokens[Imag] := "Imag";
  1204. tokens[True] := "True";
  1205. tokens[False] := "False";
  1206. tokens[Self] := "Self";
  1207. tokens[New] := "New";
  1208. tokens[Result] := "Result";
  1209. tokens[Identifier] := "Identifier";
  1210. tokens[If] := "If";
  1211. tokens[Case] := "Case";
  1212. tokens[While] := "While";
  1213. tokens[Repeat] := "Repeat";
  1214. tokens[For] := "For";
  1215. tokens[Loop] := "Loop";
  1216. tokens[With] := "With";
  1217. tokens[Exit] := "Exit";
  1218. tokens[Await] := "Await";
  1219. tokens[Return] := "Return";
  1220. tokens[Begin] := "Begin";
  1221. tokens[Semicolon] := "Semicolon";
  1222. tokens[Transpose] := "Transpose";
  1223. tokens[RightBrace] := "RightBrace";
  1224. tokens[RightBracket] := "RightBracket";
  1225. tokens[RightParenthesis] := "RightParenthesis";
  1226. tokens[Questionmark] := "Questionmark";
  1227. tokens[ExclamationMark] := "ExclamationMark";
  1228. tokens[Questionmarks] := "Questionmarks";
  1229. tokens[ExclamationMarks] := "ExclamationMarks";
  1230. tokens[LessLess] := "LessLess";
  1231. tokens[GreaterGreater] := "GreaterGreater";
  1232. tokens[Upto] := "Upto";
  1233. tokens[Arrow] := "Arrow";
  1234. tokens[Period] := "Period";
  1235. tokens[Comma] := "Comma";
  1236. tokens[Colon] := "Colon";
  1237. tokens[Of] := "Of";
  1238. tokens[Then] := "Then";
  1239. tokens[Do] := "Do";
  1240. tokens[To] := "To";
  1241. tokens[By] := "By";
  1242. tokens[Becomes] := "Becomes";
  1243. tokens[Bar] := "Bar";
  1244. tokens[End] := "End";
  1245. tokens[Else] := "Else";
  1246. tokens[Elsif] := "Elsif";
  1247. tokens[Extern] := "Extern";
  1248. tokens[Until] := "Until";
  1249. tokens[Finally] := "Finally";
  1250. tokens[Code] := "Code";
  1251. tokens[Const] := "Const";
  1252. tokens[Type] := "Type";
  1253. tokens[Var] := "Var";
  1254. tokens[Out] := "Out";
  1255. tokens[Procedure] := "Procedure";
  1256. tokens[Operator] := "Operator";
  1257. tokens[Import] := "Import";
  1258. tokens[Definition] := "Definition";
  1259. tokens[Module] := "Module";
  1260. tokens[Cell] := "Cell";
  1261. tokens[CellNet] := "CellNet";
  1262. tokens[Array] := "Array";
  1263. tokens[Object] := "Object";
  1264. tokens[Record] := "Record";
  1265. tokens[Pointer] := "Pointer";
  1266. tokens[Enum] := "Enum";
  1267. tokens[Port] := "Port";
  1268. tokens[Address] := "Address";
  1269. tokens[Alias] := "Alias";
  1270. tokens[Size] := "Size";
  1271. tokens[Ln] := "Ln";
  1272. tokens[PC] := "PC";
  1273. tokens[PCOffset] := "PCOffset";
  1274. tokens[Shortint] := "Shortint";
  1275. tokens[Integer] := "Integer";
  1276. tokens[Longint] := "Longint";
  1277. tokens[Hugeint] := "Hugeint";
  1278. tokens[Real] := "Real";
  1279. tokens[Longreal] := "Longreal";
  1280. tokens[Comment] := "Comment";
  1281. tokens[EndOfText] := "EndOfText";
  1282. FOR i := 0 TO EndOfText DO ASSERT(tokens[i] # "") END;
  1283. END InitTokens;
  1284. (** enter keywords in the list of keywords (both upper- and lowercase) **)
  1285. PROCEDURE InitKeywords;
  1286. PROCEDURE Upper(CONST source: ARRAY OF CHAR; VAR dest: ARRAY OF CHAR);
  1287. VAR c: CHAR; i: LONGINT;
  1288. BEGIN
  1289. i := 0;
  1290. REPEAT
  1291. c := source[i];
  1292. IF (c >= 'a') & (c<= 'z') THEN c := CHR(ORD(c)-ORD('a')+ORD('A')) END;
  1293. dest[i] := c; INC(i);
  1294. UNTIL c = 0X;
  1295. END Upper;
  1296. PROCEDURE Enter1(CONST name: ARRAY OF CHAR; token: LONGINT; case: SET);
  1297. BEGIN
  1298. IF Lowercase IN case THEN keywordsLower.PutString(name,token) END;
  1299. IF Uppercase IN case THEN keywordsUpper.PutString(name,token) END;
  1300. Basic.SetErrorExpected(token,name);
  1301. END Enter1;
  1302. PROCEDURE Enter(CONST name: ARRAY OF CHAR; token: LONGINT);
  1303. VAR upper: Keyword;
  1304. BEGIN
  1305. Enter1(name,token,{Lowercase});
  1306. Upper(name,upper);
  1307. Enter1(upper,token,{Uppercase});
  1308. END Enter;
  1309. PROCEDURE EnterSymbol(CONST name: ARRAY OF CHAR; token: LONGINT);
  1310. BEGIN
  1311. Enter1(name,token,{Lowercase,Uppercase});
  1312. END EnterSymbol;
  1313. BEGIN
  1314. NEW(keywordsUpper,EndOfText+1);
  1315. NEW(keywordsLower,EndOfText+1);
  1316. (* constructs and statements *)
  1317. Enter( "cell", Cell );
  1318. Enter( "cellnet", CellNet);
  1319. Enter( "await" , Await);
  1320. Enter( "begin" , Begin);
  1321. Enter( "by" , By);
  1322. Enter( "const" , Const);
  1323. Enter( "case" , Case);
  1324. Enter( "code" , Code);
  1325. Enter( "definition", Definition);
  1326. Enter( "do" , Do);
  1327. Enter( "div" , Div);
  1328. Enter( "end" , End);
  1329. Enter( "enum", Enum);
  1330. Enter( "else" , Else);
  1331. Enter( "elsif" , Elsif);
  1332. Enter( "exit" , Exit);
  1333. Enter( "extern" , Extern);
  1334. Enter( "false" , False);
  1335. Enter( "for" , For);
  1336. Enter( "finally" , Finally);
  1337. Enter( "if" , If);
  1338. Enter( "imag" , Imag);
  1339. Enter( "in" , In);
  1340. Enter( "is" , Is);
  1341. Enter( "import" , Import);
  1342. Enter( "loop" , Loop);
  1343. Enter( "module", Module);
  1344. Enter( "mod" , Mod);
  1345. Enter( "nil" , Nil );
  1346. Enter( "of" , Of);
  1347. Enter( "or" , Or);
  1348. Enter( "out", Out);
  1349. Enter( "operator" , Operator);
  1350. Enter( "procedure" , Procedure);
  1351. Enter( "port", Port);
  1352. Enter( "repeat" , Repeat);
  1353. Enter( "return" , Return);
  1354. Enter( "self", Self);
  1355. Enter( "new", New);
  1356. Enter( "result", Result);
  1357. Enter( "then" , Then);
  1358. Enter( "true" , True);
  1359. Enter( "to" , To);
  1360. Enter( "type" , Type);
  1361. Enter( "until" , Until );
  1362. Enter( "var" , Var );
  1363. Enter( "while" , While);
  1364. Enter( "with" , With);
  1365. (* types *)
  1366. Enter( "array" , Array );
  1367. Enter( "object" , Object);
  1368. Enter( "pointer" , Pointer);
  1369. Enter( "record" , Record);
  1370. Enter( "address" , Address);
  1371. Enter( "size" , Size);
  1372. Enter( "alias" , Alias);
  1373. (* symbols *)
  1374. EnterSymbol( "#", Unequal);
  1375. EnterSymbol( "&", And);
  1376. EnterSymbol( "(", LeftParenthesis);
  1377. EnterSymbol( ")", RightParenthesis);
  1378. EnterSymbol( "*", Times);
  1379. EnterSymbol( "**",TimesTimes);
  1380. EnterSymbol( "+", Plus);
  1381. EnterSymbol( "+*", PlusTimes);
  1382. EnterSymbol( ",", Comma);
  1383. EnterSymbol( "-", Minus);
  1384. EnterSymbol(".",Period );
  1385. EnterSymbol("..",Upto );
  1386. EnterSymbol(".*",DotTimes );
  1387. EnterSymbol("./",DotSlash );
  1388. EnterSymbol(".=",DotEqual );
  1389. EnterSymbol(".#",DotUnequal );
  1390. EnterSymbol(".>",DotGreater );
  1391. EnterSymbol(".>=",DotGreaterEqual );
  1392. EnterSymbol(".<", DotLess);
  1393. EnterSymbol(".<=",DotLessEqual );
  1394. EnterSymbol( "/", Slash);
  1395. EnterSymbol( ":", Colon);
  1396. EnterSymbol( ":=",Becomes);
  1397. EnterSymbol( ";", Semicolon);
  1398. EnterSymbol( "<", Less);
  1399. EnterSymbol( "<=", LessEqual);
  1400. EnterSymbol( "=", Equal);
  1401. EnterSymbol( ">", Greater);
  1402. EnterSymbol( ">=", GreaterEqual);
  1403. EnterSymbol( "[", LeftBracket);
  1404. EnterSymbol( "]", RightBracket);
  1405. EnterSymbol( "^", Arrow);
  1406. EnterSymbol( "{", LeftBrace);
  1407. EnterSymbol( "|",Bar);
  1408. EnterSymbol( "}", RightBrace);
  1409. EnterSymbol( "~", Not);
  1410. EnterSymbol( "\", Backslash);
  1411. EnterSymbol( "`", Transpose);
  1412. EnterSymbol( "?",Questionmark);
  1413. EnterSymbol( "??",Questionmarks);
  1414. EnterSymbol( "!",ExclamationMark);
  1415. EnterSymbol( "!!",ExclamationMarks);
  1416. EnterSymbol( "<<",LessLess);
  1417. EnterSymbol( "<<?",LessLessQ);
  1418. EnterSymbol( ">>",GreaterGreater);
  1419. EnterSymbol( ">>?",GreaterGreaterQ);
  1420. Basic.SetErrorMessage(Number,"missing number");
  1421. Basic.SetErrorMessage(String,"missing string");
  1422. Basic.SetErrorMessage(Character,"missing character");
  1423. Basic.SetErrorMessage(Identifier,"missing identifier");
  1424. Basic.SetErrorMessage(EndOfText,"unexpected symbol before end");
  1425. END InitKeywords;
  1426. (** debugging / reporting **)
  1427. PROCEDURE ReportKeywords*(context: Commands.Context);
  1428. VAR i: LONGINT; name: Keyword;
  1429. BEGIN
  1430. FOR i := 0 TO EndOfText DO
  1431. context.out.Int(i,1); context.out.String(": ");
  1432. context.out.Char('"');
  1433. keywordsLower.StringByIndex(i,name);
  1434. context.out.String(name);
  1435. context.out.Char('"');
  1436. context.out.String(", ");
  1437. context.out.Char('"');
  1438. keywordsUpper.StringByIndex(i,name);
  1439. context.out.String(name);
  1440. context.out.Char('"');
  1441. context.out.Ln;
  1442. END;
  1443. END ReportKeywords;
  1444. (*
  1445. PROCEDURE TestScanner*(context: Commands.Context);
  1446. VAR filename: ARRAY 256 OF CHAR; reader: Streams.Reader; scanner: Scanner;sym: Symbol;
  1447. BEGIN
  1448. context.arg.SkipWhitespace; context.arg.String(filename);
  1449. reader := TextUtilities.GetTextReader(filename);
  1450. scanner := NewScanner(filename,reader,0,NIL);
  1451. REPEAT
  1452. IF scanner.GetNextSymbol(sym) THEN
  1453. OutSymbol(context.out,sym);context.out.Ln;
  1454. END;
  1455. UNTIL scanner.error OR (sym.token=EndOfText)
  1456. END TestScanner;
  1457. *)
  1458. BEGIN
  1459. InitReservedCharacters; InitTokens; InitKeywords
  1460. END FoxScanner.
  1461. FoxScanner.ReportKeywords
  1462. FoxScanner.TestScanner Test.Mod ~