LLex.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995
  1. /*
  2. #########
  3. ############
  4. #############
  5. ## ###########
  6. ### ###### #####
  7. ### ####### ####
  8. ### ########## ####
  9. #### ########### ####
  10. #### ########### #####
  11. ##### ### ######## #####
  12. ##### ### ######## ######
  13. ###### ### ########### ######
  14. ###### #### ############## ######
  15. ####### ##################### ######
  16. ####### ###################### ######
  17. ####### ###### ################# ######
  18. ####### ###### ###### ######### ######
  19. ####### ## ###### ###### ######
  20. ####### ###### ##### #####
  21. ###### ##### ##### ####
  22. ##### #### ##### ###
  23. ##### ### ### #
  24. ### ### ###
  25. ## ### ###
  26. __________#_______####_______####______________
  27. 我们的未来没有BUG
  28. * ==============================================================================
  29. * Filename: LLex
  30. * Created: 2018/7/2 11:36:16
  31. * Author: エル・プサイ・コングリィ
  32. * Purpose:
  33. * ==============================================================================
  34. */
  35. #if UNITY_EDITOR || USE_LUA_PROFILER
  36. using System;
  37. using System.Collections.Generic;
  38. using System.Text;
  39. using NumberStyles = System.Globalization.NumberStyles;
  40. namespace MikuLuaProfiler
  41. {
  42. public enum TK
  43. {
  44. // reserved words
  45. AND = 257,
  46. BREAK,
  47. CONTINUE,
  48. DO,
  49. ELSE,
  50. ELSEIF,
  51. END,
  52. FALSE,
  53. FOR,
  54. FUNCTION,
  55. GOTO,
  56. IF,
  57. IN,
  58. LOCAL,
  59. NIL,
  60. NOT,
  61. OR,
  62. REPEAT,
  63. RETURN,
  64. THEN,
  65. TRUE,
  66. UNTIL,
  67. WHILE,
  68. // other terminal symbols
  69. CONCAT,
  70. DOTS,
  71. EQ,
  72. GE,
  73. LE,
  74. NE,
  75. DBCOLON,
  76. NUMBER,
  77. STRING,
  78. NAME,
  79. EOS,
  80. }
  81. public class StringLoadInfo
  82. {
  83. public StringLoadInfo(string s)
  84. {
  85. Str = new StringBuilder(s);
  86. Str.Append(' ');
  87. Pos = 0;
  88. }
  89. public int ReadByte()
  90. {
  91. if (Pos >= Str.Length)
  92. return -1;
  93. else
  94. return Str[Pos++];
  95. }
  96. public int PosChar
  97. {
  98. get
  99. {
  100. if (Pos >= Str.Length)
  101. return (int)TK.EOS;
  102. else
  103. return Str[Pos];
  104. }
  105. }
  106. public int Length
  107. {
  108. get
  109. {
  110. return Str.Length;
  111. }
  112. }
  113. public int PeekByte()
  114. {
  115. if (Pos >= Str.Length)
  116. return -1;
  117. else
  118. return Str[Pos];
  119. }
  120. public void Replace(int start, int len, string value)
  121. {
  122. Str = Str.Remove(start, len);
  123. Str = Str.Insert(start, value);
  124. if ((start + len) <= Pos)
  125. {
  126. Pos = Pos - (len - value.Length);
  127. }
  128. }
  129. public string ReadString(int startPos, int len)
  130. {
  131. char[] chars = new char[len];
  132. for (int i = 0, imax = len; i < imax; i++)
  133. {
  134. chars[i] = Str[i + startPos];
  135. }
  136. return new string(chars);
  137. }
  138. public void InsertString(int startPos, string value)
  139. {
  140. Str = Str.Insert(startPos, value);
  141. Pos += value.Length;
  142. }
  143. public string code
  144. {
  145. get
  146. {
  147. return Str.ToString();
  148. }
  149. }
  150. private StringBuilder Str;
  151. public int Pos;
  152. }
  153. public class BytesLoadInfo
  154. {
  155. public BytesLoadInfo(byte[] bytes)
  156. {
  157. Bytes = bytes;
  158. Pos = 0;
  159. }
  160. public int ReadByte()
  161. {
  162. if (Pos >= Bytes.Length)
  163. return -1;
  164. else
  165. return Bytes[Pos++];
  166. }
  167. public int PeekByte()
  168. {
  169. if (Pos >= Bytes.Length)
  170. return -1;
  171. else
  172. return Bytes[Pos];
  173. }
  174. private byte[] Bytes;
  175. private int Pos;
  176. }
  177. public abstract class Token
  178. {
  179. public abstract int TokenType { get; }
  180. public bool EqualsToToken(Token other)
  181. {
  182. return TokenType == other.TokenType;
  183. }
  184. public bool EqualsToToken(int other)
  185. {
  186. return TokenType == other;
  187. }
  188. public bool EqualsToToken(TK other)
  189. {
  190. return TokenType == (int)other;
  191. }
  192. }
  193. public class JumpToken : Token
  194. {
  195. public JumpToken(int pos)
  196. {
  197. Pos = pos;
  198. }
  199. public int Pos;
  200. public override int TokenType
  201. {
  202. get { return -1; }
  203. }
  204. }
  205. public class LiteralToken : Token
  206. {
  207. private int _Literal;
  208. public LiteralToken(int literal)
  209. {
  210. _Literal = literal;
  211. }
  212. public override int TokenType
  213. {
  214. get { return _Literal; }
  215. }
  216. public override string ToString()
  217. {
  218. return string.Format("LiteralToken: {0}", (char)_Literal);
  219. }
  220. }
  221. public class TypedToken : Token
  222. {
  223. private TK _Type;
  224. public TypedToken(TK type)
  225. {
  226. _Type = type;
  227. }
  228. public override int TokenType
  229. {
  230. get { return (int)_Type; }
  231. }
  232. public override string ToString()
  233. {
  234. return string.Format("TypedToken: {0}", _Type);
  235. }
  236. }
  237. public class StringToken : TypedToken
  238. {
  239. public string SemInfo;
  240. public StringToken(string seminfo) : base(TK.STRING)
  241. {
  242. SemInfo = seminfo;
  243. }
  244. public override string ToString()
  245. {
  246. return string.Format("StringToken: {0}", SemInfo);
  247. }
  248. }
  249. public class NameToken : TypedToken
  250. {
  251. public string SemInfo;
  252. public NameToken(string seminfo) : base(TK.NAME)
  253. {
  254. SemInfo = seminfo;
  255. }
  256. public override string ToString()
  257. {
  258. return string.Format("NameToken: {0}", SemInfo);
  259. }
  260. }
  261. public class NumberToken : TypedToken
  262. {
  263. public double SemInfo;
  264. public NumberToken(double seminfo) : base(TK.NUMBER)
  265. {
  266. SemInfo = seminfo;
  267. }
  268. public override string ToString()
  269. {
  270. return string.Format("NumberToken: {0}", SemInfo);
  271. }
  272. }
  273. public class LLex
  274. {
  275. public const char EOZ = Char.MaxValue;
  276. public int pos
  277. {
  278. get
  279. {
  280. return LoadInfo.Pos;
  281. }
  282. }
  283. private int Current;
  284. public int LineNumber;
  285. public int LastLine;
  286. private StringLoadInfo LoadInfo;
  287. public string Source;
  288. public Token Token;
  289. private Token LookAhead;
  290. private StringBuilder _Saved;
  291. private StringBuilder Saved
  292. {
  293. get
  294. {
  295. if (_Saved == null) { _Saved = new StringBuilder(); }
  296. return _Saved;
  297. }
  298. }
  299. private static Dictionary<string, TK> ReservedWordDict;
  300. static LLex()
  301. {
  302. ReservedWordDict = new Dictionary<string, TK>();
  303. ReservedWordDict.Add("and", TK.AND);
  304. ReservedWordDict.Add("break", TK.BREAK);
  305. ReservedWordDict.Add("continue", TK.CONTINUE);
  306. ReservedWordDict.Add("do", TK.DO);
  307. ReservedWordDict.Add("else", TK.ELSE);
  308. ReservedWordDict.Add("elseif", TK.ELSEIF);
  309. ReservedWordDict.Add("end", TK.END);
  310. ReservedWordDict.Add("false", TK.FALSE);
  311. ReservedWordDict.Add("for", TK.FOR);
  312. ReservedWordDict.Add("function", TK.FUNCTION);
  313. ReservedWordDict.Add("goto", TK.GOTO);
  314. ReservedWordDict.Add("if", TK.IF);
  315. ReservedWordDict.Add("in", TK.IN);
  316. ReservedWordDict.Add("local", TK.LOCAL);
  317. ReservedWordDict.Add("nil", TK.NIL);
  318. ReservedWordDict.Add("not", TK.NOT);
  319. ReservedWordDict.Add("or", TK.OR);
  320. ReservedWordDict.Add("repeat", TK.REPEAT);
  321. ReservedWordDict.Add("return", TK.RETURN);
  322. ReservedWordDict.Add("then", TK.THEN);
  323. ReservedWordDict.Add("true", TK.TRUE);
  324. ReservedWordDict.Add("until", TK.UNTIL);
  325. ReservedWordDict.Add("while", TK.WHILE);
  326. }
  327. public LLex(StringLoadInfo loadinfo, string name)
  328. {
  329. LoadInfo = loadinfo;
  330. LineNumber = 1;
  331. LastLine = 1;
  332. Token = null;
  333. LookAhead = null;
  334. _Saved = null;
  335. Source = name;
  336. _Next();
  337. }
  338. public string code
  339. {
  340. get
  341. {
  342. return LoadInfo.code;
  343. }
  344. }
  345. public void InsertString(int startPos, string value)
  346. {
  347. LoadInfo.InsertString(startPos, value);
  348. }
  349. public void Next()
  350. {
  351. LastLine = LineNumber;
  352. if (LookAhead != null)
  353. {
  354. Token = LookAhead;
  355. LookAhead = null;
  356. }
  357. else
  358. {
  359. Token = _Lex();
  360. }
  361. }
  362. public Token GetLookAhead()
  363. {
  364. Utl.Assert(LookAhead == null);
  365. LookAhead = _Lex();
  366. return LookAhead;
  367. }
  368. public int Length
  369. {
  370. get
  371. {
  372. return LoadInfo.Length;
  373. }
  374. }
  375. public void Replace(int start, int end, string value)
  376. {
  377. LoadInfo.Replace(start, end + 1 - start, value);
  378. }
  379. public string ReadString(int start, int end)
  380. {
  381. int len = end - start + 1;
  382. return LoadInfo.ReadString(start, len);
  383. }
  384. private void _Next()
  385. {
  386. var c = LoadInfo.ReadByte();
  387. Current = (c == -1) ? EOZ : c;
  388. }
  389. private void _SaveAndNext()
  390. {
  391. Saved.Append((char)Current);
  392. _Next();
  393. }
  394. private void _Save(char c)
  395. {
  396. Saved.Append(c);
  397. }
  398. private string _GetSavedString()
  399. {
  400. return Saved.ToString();
  401. }
  402. private void _ClearSaved()
  403. {
  404. _Saved = null;
  405. }
  406. private bool _CurrentIsNewLine()
  407. {
  408. return Current == '\n' || Current == '\r';
  409. }
  410. private bool _CurrentIsDigit()
  411. {
  412. return Char.IsDigit((char)Current);
  413. }
  414. private bool _CurrentIsXDigit()
  415. {
  416. return _CurrentIsDigit() ||
  417. ('A' <= Current && Current <= 'F') ||
  418. ('a' <= Current && Current <= 'f');
  419. }
  420. private bool _CurrentIsSpace()
  421. {
  422. return Char.IsWhiteSpace((char)Current);
  423. }
  424. private bool _CurrentIsAlpha()
  425. {
  426. return Char.IsLetter((char)Current);
  427. }
  428. private bool _IsReserved(string identifier, out TK type)
  429. {
  430. return ReservedWordDict.TryGetValue(identifier, out type);
  431. }
  432. public bool IsReservedWord(string name)
  433. {
  434. return ReservedWordDict.ContainsKey(name);
  435. }
  436. private void _IncLineNumber()
  437. {
  438. var old = Current;
  439. _Next();
  440. if (_CurrentIsNewLine() && Current != old)
  441. _Next();
  442. if (++LineNumber >= Int32.MaxValue)
  443. _Error("chunk has too many lines");
  444. }
  445. private string _ReadLongString(int sep, bool isComment)
  446. {
  447. _SaveAndNext();
  448. if (_CurrentIsNewLine())
  449. _IncLineNumber();
  450. while (true)
  451. {
  452. switch (Current)
  453. {
  454. case EOZ:
  455. _LexError(_GetSavedString(),
  456. "unfinished long string/comment",
  457. (int)TK.EOS);
  458. break;
  459. case '[':
  460. {
  461. if (_SkipSep() == sep)
  462. {
  463. _SaveAndNext();
  464. if (sep == 0 && !isComment)
  465. {
  466. _LexError(_GetSavedString(),
  467. "nesting of [[...]] is deprecated",
  468. (int)TK.EOS);
  469. }
  470. }
  471. break;
  472. }
  473. case ']':
  474. {
  475. if (_SkipSep() == sep)
  476. {
  477. _SaveAndNext();
  478. goto endloop;
  479. }
  480. break;
  481. }
  482. case '\n':
  483. case '\r':
  484. {
  485. _Save('\n');
  486. _IncLineNumber();
  487. break;
  488. }
  489. default:
  490. {
  491. _SaveAndNext();
  492. break;
  493. }
  494. }
  495. }
  496. endloop:
  497. var r = _GetSavedString();
  498. int len = r.Length - 2 * (2 + sep);
  499. if (len <= 0)
  500. {
  501. return string.Empty;
  502. }
  503. else
  504. {
  505. return r.Substring(2 + sep, len);
  506. }
  507. }
  508. private void _EscapeError(string info, string msg)
  509. {
  510. _LexError("\\" + info, msg, (int)TK.STRING);
  511. }
  512. private byte _ReadHexEscape()
  513. {
  514. int r = 0;
  515. var c = new char[3] { 'x', (char)0, (char)0 };
  516. // read two hex digits
  517. for (int i = 1; i < 3; ++i)
  518. {
  519. _Next();
  520. c[i] = (char)Current;
  521. if (!_CurrentIsXDigit())
  522. {
  523. _EscapeError(new String(c, 0, i + 1),
  524. "hexadecimal digit expected");
  525. // error
  526. }
  527. r = (r << 4) + Int32.Parse(Current.ToString(),
  528. NumberStyles.HexNumber);
  529. }
  530. return (byte)r;
  531. }
  532. private byte _ReadDecEscape()
  533. {
  534. int r = 0;
  535. var c = new char[3];
  536. // read up to 3 digits
  537. int i = 0;
  538. for (i = 0; i < 3 && _CurrentIsDigit(); ++i)
  539. {
  540. c[i] = (char)Current;
  541. r = r * 10 + Current - '0';
  542. _Next();
  543. }
  544. if (r > Byte.MaxValue)
  545. _EscapeError(new String(c, 0, i),
  546. "decimal escape too large");
  547. return (byte)r;
  548. }
  549. private string _ReadString()
  550. {
  551. var del = Current;
  552. _Next();
  553. while (Current != del)
  554. {
  555. switch (Current)
  556. {
  557. case EOZ:
  558. _Error("unfinished string");
  559. continue;
  560. case '\n':
  561. case '\r':
  562. _Error("unfinished string");
  563. continue;
  564. case '\\':
  565. {
  566. byte c;
  567. _Next();
  568. switch (Current)
  569. {
  570. case 'a': c = (byte)'\a'; break;
  571. case 'b': c = (byte)'\b'; break;
  572. case 'f': c = (byte)'\f'; break;
  573. case 'n': c = (byte)'\n'; break;
  574. case 'r': c = (byte)'\r'; break;
  575. case 't': c = (byte)'\t'; break;
  576. case 'v': c = (byte)'\v'; break;
  577. case 'x': c = _ReadHexEscape(); break;
  578. case '\n':
  579. case '\r': _Save('\n'); _IncLineNumber(); continue;
  580. case 'u' :
  581. case '\\':
  582. case '\"':
  583. case '\'': c = (byte)Current; break;
  584. case EOZ: continue;
  585. // zap following span of spaces
  586. case 'z':
  587. {
  588. _Next(); // skip `z'
  589. while (_CurrentIsSpace())
  590. {
  591. if (_CurrentIsNewLine())
  592. _IncLineNumber();
  593. else
  594. _Next();
  595. }
  596. continue;
  597. }
  598. default:
  599. {
  600. if (!_CurrentIsDigit())
  601. _EscapeError(Current.ToString(),
  602. "invalid escape sequence");
  603. // digital escape \ddd
  604. c = _ReadDecEscape();
  605. _Save((char)c);
  606. continue;
  607. // {
  608. // c = (char)0;
  609. // for(int i=0; i<3 && _CurrentIsDigit(); ++i)
  610. // {
  611. // c = (char)(c*10 + Current - '0');
  612. // _Next();
  613. // }
  614. // _Save( c );
  615. // }
  616. // continue;
  617. }
  618. }
  619. _Save((char)c);
  620. _Next();
  621. continue;
  622. }
  623. default:
  624. _SaveAndNext();
  625. continue;
  626. }
  627. }
  628. _Next();
  629. return _GetSavedString();
  630. }
  631. private double _ReadNumber()
  632. {
  633. var expo = new char[] { 'E', 'e' };
  634. Utl.Assert(_CurrentIsDigit());
  635. var first = Current;
  636. _SaveAndNext();
  637. if (first == '0' && (Current == 'X' || Current == 'x'))
  638. {
  639. expo = new char[] { 'P', 'p' };
  640. _SaveAndNext();
  641. }
  642. for (; ; )
  643. {
  644. if (Current == expo[0] || Current == expo[1])
  645. {
  646. _SaveAndNext();
  647. if (Current == '+' || Current == '-')
  648. _SaveAndNext();
  649. }
  650. if (_CurrentIsXDigit() || Current == '.')
  651. _SaveAndNext();
  652. else
  653. break;
  654. }
  655. double ret;
  656. var str = _GetSavedString();
  657. if (O_Str2Decimal(str, out ret))
  658. {
  659. return ret;
  660. }
  661. else
  662. {
  663. _Error("malformed number: " + str);
  664. return 0.0;
  665. }
  666. }
  667. public static bool O_Str2Decimal(string s, out double result)
  668. {
  669. result = 0.0;
  670. if (s.Contains("n") || s.Contains("N")) // reject `inf' and `nan'
  671. return false;
  672. int pos = 0;
  673. if (s.Contains("x") || s.Contains("X"))
  674. result = Utl.StrX2Number(s, ref pos);
  675. else
  676. result = Utl.Str2Number(s, ref pos);
  677. if (pos == 0)
  678. return false; // nothing recognized
  679. while (pos < s.Length && Char.IsWhiteSpace(s[pos])) ++pos;
  680. return pos == s.Length; // OK if no trailing characters
  681. }
  682. // private float _ReadNumber()
  683. // {
  684. // do
  685. // {
  686. // _SaveAndNext();
  687. // } while( _CurrentIsDigit() || Current == '.' );
  688. // if( Current == 'E' || Current == 'e' )
  689. // {
  690. // _SaveAndNext();
  691. // if( Current == '+' || Current == '-' )
  692. // _SaveAndNext();
  693. // }
  694. // while( _CurrentIsAlpha() || _CurrentIsDigit() || Current == '_' )
  695. // _SaveAndNext();
  696. // float ret;
  697. // if( !Single.TryParse( _GetSavedString(), out ret ) )
  698. // _Error( "malformed number" );
  699. // return ret;
  700. // }
  701. private void _Error(string error)
  702. {
  703. throw new Exception(string.Format("{0}:{1}: {2}", Source, LineNumber, error));
  704. }
  705. private void _LexError(string info, string msg, int tokenType)
  706. {
  707. // TODO
  708. _Error(msg + ":" + info);
  709. }
  710. public void SyntaxError(string msg)
  711. {
  712. // TODO
  713. _Error(msg);
  714. }
  715. private int _SkipSep()
  716. {
  717. int count = 0;
  718. var boundary = Current;
  719. _SaveAndNext();
  720. while (Current == '=')
  721. {
  722. _SaveAndNext();
  723. count++;
  724. }
  725. return (Current == boundary ? count : (-count) - 1);
  726. }
  727. private Token _Lex()
  728. {
  729. _ClearSaved();
  730. while (true)
  731. {
  732. switch (Current)
  733. {
  734. case '\n':
  735. case '\r':
  736. {
  737. var token = new JumpToken(pos);
  738. _IncLineNumber();
  739. return token;
  740. }
  741. case '-':
  742. {
  743. _Next();
  744. if (Current != '-') return new LiteralToken('-');
  745. // else is a long comment
  746. _Next();
  747. if (Current == '[')
  748. {
  749. int sep = _SkipSep();
  750. _ClearSaved();
  751. if (sep >= 0)
  752. {
  753. _ReadLongString(sep, true);
  754. _ClearSaved();
  755. return new JumpToken(pos);
  756. }
  757. }
  758. // else is a short comment
  759. while (!_CurrentIsNewLine() && Current != EOZ)
  760. _Next();
  761. return new JumpToken(pos);
  762. }
  763. case '[':
  764. {
  765. int sep = _SkipSep();
  766. if (sep >= 0)
  767. {
  768. string seminfo = _ReadLongString(sep, false);
  769. return new StringToken(seminfo);
  770. }
  771. else if (sep == -1) return new LiteralToken('[');
  772. else _Error("invalid long string delimiter");
  773. continue;
  774. }
  775. case '=':
  776. {
  777. _Next();
  778. if (Current != '=') return new LiteralToken('=');
  779. _Next();
  780. return new TypedToken(TK.EQ);
  781. }
  782. case '<':
  783. {
  784. _Next();
  785. if (Current != '=') return new LiteralToken('<');
  786. _Next();
  787. return new TypedToken(TK.LE);
  788. }
  789. case '>':
  790. {
  791. _Next();
  792. if (Current != '=') return new LiteralToken('>');
  793. _Next();
  794. return new TypedToken(TK.GE);
  795. }
  796. case '~':
  797. {
  798. _Next();
  799. if (Current != '=') return new LiteralToken('~');
  800. _Next();
  801. return new TypedToken(TK.NE);
  802. }
  803. case ':':
  804. {
  805. _Next();
  806. if (Current != ':') return new LiteralToken(':');
  807. _Next();
  808. return new TypedToken(TK.DBCOLON); // new in 5.2 ?
  809. }
  810. case '"':
  811. case '\'':
  812. {
  813. return new StringToken(_ReadString());
  814. }
  815. case '.':
  816. {
  817. _SaveAndNext();
  818. if (Current == '.')
  819. {
  820. _SaveAndNext();
  821. if (Current == '.')
  822. {
  823. _SaveAndNext();
  824. return new TypedToken(TK.DOTS);
  825. }
  826. else
  827. {
  828. return new TypedToken(TK.CONCAT);
  829. }
  830. }
  831. else if (!_CurrentIsDigit())
  832. return new LiteralToken('.');
  833. else
  834. return new NumberToken(_ReadNumber());
  835. }
  836. case EOZ:
  837. {
  838. return new TypedToken(TK.EOS);
  839. }
  840. default:
  841. {
  842. if (_CurrentIsSpace())
  843. {
  844. var token = new JumpToken(pos);
  845. _Next();
  846. return token;
  847. //continue;
  848. }
  849. else if (Current == ';')
  850. {
  851. var token = new JumpToken(pos);
  852. _Next();
  853. return token;
  854. }
  855. else if (_CurrentIsDigit())
  856. {
  857. return new NumberToken(_ReadNumber());
  858. }
  859. else if (_CurrentIsAlpha() || Current == '_')
  860. {
  861. do
  862. {
  863. _SaveAndNext();
  864. } while (_CurrentIsAlpha() ||
  865. _CurrentIsDigit() ||
  866. Current == '_');
  867. string identifier = _GetSavedString();
  868. TK type;
  869. if (_IsReserved(identifier, out type))
  870. {
  871. return new TypedToken(type);
  872. }
  873. else
  874. {
  875. return new NameToken(identifier);
  876. }
  877. }
  878. else
  879. {
  880. var c = Current;
  881. _Next();
  882. return new LiteralToken(c);
  883. }
  884. }
  885. }
  886. }
  887. }
  888. }
  889. }
  890. #endif