lexer.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919
  1. let source, pos, end,
  2. openTokenDepth,
  3. lastTokenPos,
  4. openTokenPosStack,
  5. openClassPosStack,
  6. curDynamicImport,
  7. templateStackDepth,
  8. facade,
  9. lastSlashWasDivision,
  10. nextBraceIsClass,
  11. templateDepth,
  12. templateStack,
  13. imports,
  14. exports,
  15. name;
  16. function addImport (ss, s, e, d) {
  17. const impt = { ss, se: d === -2 ? e : d === -1 ? e + 1 : 0, s, e, d, a: -1, n: undefined };
  18. imports.push(impt);
  19. return impt;
  20. }
  21. function addExport (s, e, ls, le) {
  22. exports.push({
  23. s,
  24. e,
  25. ls,
  26. le,
  27. n: s[0] === '"' ? readString(s, '"') : s[0] === "'" ? readString(s, "'") : source.slice(s, e),
  28. ln: ls[0] === '"' ? readString(ls, '"') : ls[0] === "'" ? readString(ls, "'") : source.slice(ls, le)
  29. });
  30. }
  31. function readName (impt) {
  32. let { d, s } = impt;
  33. if (d !== -1)
  34. s++;
  35. impt.n = readString(s, source.charCodeAt(s - 1));
  36. }
  37. // Note: parsing is based on the _assumption_ that the source is already valid
  38. export function parse (_source, _name) {
  39. openTokenDepth = 0;
  40. curDynamicImport = null;
  41. templateDepth = -1;
  42. lastTokenPos = -1;
  43. lastSlashWasDivision = false;
  44. templateStack = Array(1024);
  45. templateStackDepth = 0;
  46. openTokenPosStack = Array(1024);
  47. openClassPosStack = Array(1024);
  48. nextBraceIsClass = false;
  49. facade = true;
  50. name = _name || '@';
  51. imports = [];
  52. exports = [];
  53. source = _source;
  54. pos = -1;
  55. end = source.length - 1;
  56. let ch = 0;
  57. // start with a pure "module-only" parser
  58. m: while (pos++ < end) {
  59. ch = source.charCodeAt(pos);
  60. if (ch === 32 || ch < 14 && ch > 8)
  61. continue;
  62. switch (ch) {
  63. case 101/*e*/:
  64. if (openTokenDepth === 0 && keywordStart(pos) && source.startsWith('xport', pos + 1)) {
  65. tryParseExportStatement();
  66. // export might have been a non-pure declaration
  67. if (!facade) {
  68. lastTokenPos = pos;
  69. break m;
  70. }
  71. }
  72. break;
  73. case 105/*i*/:
  74. if (keywordStart(pos) && source.startsWith('mport', pos + 1))
  75. tryParseImportStatement();
  76. break;
  77. case 59/*;*/:
  78. break;
  79. case 47/*/*/: {
  80. const next_ch = source.charCodeAt(pos + 1);
  81. if (next_ch === 47/*/*/) {
  82. lineComment();
  83. // dont update lastToken
  84. continue;
  85. }
  86. else if (next_ch === 42/***/) {
  87. blockComment(true);
  88. // dont update lastToken
  89. continue;
  90. }
  91. // fallthrough
  92. }
  93. default:
  94. // as soon as we hit a non-module token, we go to main parser
  95. facade = false;
  96. pos--;
  97. break m;
  98. }
  99. lastTokenPos = pos;
  100. }
  101. while (pos++ < end) {
  102. ch = source.charCodeAt(pos);
  103. if (ch === 32 || ch < 14 && ch > 8)
  104. continue;
  105. switch (ch) {
  106. case 101/*e*/:
  107. if (openTokenDepth === 0 && keywordStart(pos) && source.startsWith('xport', pos + 1))
  108. tryParseExportStatement();
  109. break;
  110. case 105/*i*/:
  111. if (keywordStart(pos) && source.startsWith('mport', pos + 1))
  112. tryParseImportStatement();
  113. break;
  114. case 99/*c*/:
  115. if (keywordStart(pos) && source.startsWith('lass', pos + 1) && isBrOrWs(source.charCodeAt(pos + 5)))
  116. nextBraceIsClass = true;
  117. break;
  118. case 40/*(*/:
  119. openTokenPosStack[openTokenDepth++] = lastTokenPos;
  120. break;
  121. case 41/*)*/:
  122. if (openTokenDepth === 0)
  123. syntaxError();
  124. openTokenDepth--;
  125. if (curDynamicImport && curDynamicImport.d === openTokenPosStack[openTokenDepth]) {
  126. if (curDynamicImport.e === 0)
  127. curDynamicImport.e = pos;
  128. curDynamicImport.se = pos;
  129. curDynamicImport = null;
  130. }
  131. break;
  132. case 123/*{*/:
  133. // dynamic import followed by { is not a dynamic import (so remove)
  134. // this is a sneaky way to get around { import () {} } v { import () }
  135. // block / object ambiguity without a parser (assuming source is valid)
  136. if (source.charCodeAt(lastTokenPos) === 41/*)*/ && imports.length && imports[imports.length - 1].e === lastTokenPos) {
  137. imports.pop();
  138. }
  139. openClassPosStack[openTokenDepth] = nextBraceIsClass;
  140. nextBraceIsClass = false;
  141. openTokenPosStack[openTokenDepth++] = lastTokenPos;
  142. break;
  143. case 125/*}*/:
  144. if (openTokenDepth === 0)
  145. syntaxError();
  146. if (openTokenDepth-- === templateDepth) {
  147. templateDepth = templateStack[--templateStackDepth];
  148. templateString();
  149. }
  150. else {
  151. if (templateDepth !== -1 && openTokenDepth < templateDepth)
  152. syntaxError();
  153. }
  154. break;
  155. case 39/*'*/:
  156. case 34/*"*/:
  157. stringLiteral(ch);
  158. break;
  159. case 47/*/*/: {
  160. const next_ch = source.charCodeAt(pos + 1);
  161. if (next_ch === 47/*/*/) {
  162. lineComment();
  163. // dont update lastToken
  164. continue;
  165. }
  166. else if (next_ch === 42/***/) {
  167. blockComment(true);
  168. // dont update lastToken
  169. continue;
  170. }
  171. else {
  172. // Division / regex ambiguity handling based on checking backtrack analysis of:
  173. // - what token came previously (lastToken)
  174. // - if a closing brace or paren, what token came before the corresponding
  175. // opening brace or paren (lastOpenTokenIndex)
  176. const lastToken = source.charCodeAt(lastTokenPos);
  177. if (isExpressionPunctuator(lastToken) &&
  178. !(lastToken === 46/*.*/ && (source.charCodeAt(lastTokenPos - 1) >= 48/*0*/ && source.charCodeAt(lastTokenPos - 1) <= 57/*9*/)) &&
  179. !(lastToken === 43/*+*/ && source.charCodeAt(lastTokenPos - 1) === 43/*+*/) && !(lastToken === 45/*-*/ && source.charCodeAt(lastTokenPos - 1) === 45/*-*/) ||
  180. lastToken === 41/*)*/ && isParenKeyword(openTokenPosStack[openTokenDepth]) ||
  181. lastToken === 125/*}*/ && (isExpressionTerminator(openTokenPosStack[openTokenDepth]) || openClassPosStack[openTokenDepth]) ||
  182. lastToken === 47/*/*/ && lastSlashWasDivision ||
  183. isExpressionKeyword(lastTokenPos) ||
  184. !lastToken) {
  185. regularExpression();
  186. lastSlashWasDivision = false;
  187. }
  188. else {
  189. lastSlashWasDivision = true;
  190. }
  191. }
  192. break;
  193. }
  194. case 96/*`*/:
  195. templateString();
  196. break;
  197. }
  198. lastTokenPos = pos;
  199. }
  200. if (templateDepth !== -1 || openTokenDepth)
  201. syntaxError();
  202. return [imports, exports, facade];
  203. }
  204. function tryParseImportStatement () {
  205. const startPos = pos;
  206. pos += 6;
  207. let ch = commentWhitespace(true);
  208. switch (ch) {
  209. // dynamic import
  210. case 40/*(*/:
  211. openTokenPosStack[openTokenDepth++] = startPos;
  212. if (source.charCodeAt(lastTokenPos) === 46/*.*/)
  213. return;
  214. // dynamic import indicated by positive d
  215. const impt = addImport(startPos, pos + 1, 0, startPos);
  216. curDynamicImport = impt;
  217. // try parse a string, to record a safe dynamic import string
  218. pos++;
  219. ch = commentWhitespace(true);
  220. if (ch === 39/*'*/ || ch === 34/*"*/) {
  221. stringLiteral(ch);
  222. }
  223. else {
  224. pos--;
  225. return;
  226. }
  227. pos++;
  228. ch = commentWhitespace(true);
  229. if (ch === 44/*,*/) {
  230. impt.e = pos;
  231. pos++;
  232. ch = commentWhitespace(true);
  233. impt.a = pos;
  234. readName(impt);
  235. pos--;
  236. }
  237. else if (ch === 41/*)*/) {
  238. openTokenDepth--;
  239. impt.e = pos;
  240. impt.se = pos;
  241. readName(impt);
  242. }
  243. else {
  244. pos--;
  245. }
  246. return;
  247. // import.meta
  248. case 46/*.*/:
  249. pos++;
  250. ch = commentWhitespace(true);
  251. // import.meta indicated by d === -2
  252. if (ch === 109/*m*/ && source.startsWith('eta', pos + 1) && source.charCodeAt(lastTokenPos) !== 46/*.*/)
  253. addImport(startPos, startPos, pos + 4, -2);
  254. return;
  255. default:
  256. // no space after "import" -> not an import keyword
  257. if (pos === startPos + 6)
  258. break;
  259. case 34/*"*/:
  260. case 39/*'*/:
  261. case 123/*{*/:
  262. case 42/***/:
  263. // import statement only permitted at base-level
  264. if (openTokenDepth !== 0) {
  265. pos--;
  266. return;
  267. }
  268. while (pos < end) {
  269. ch = source.charCodeAt(pos);
  270. if (ch === 39/*'*/ || ch === 34/*"*/) {
  271. readImportString(startPos, ch);
  272. return;
  273. }
  274. pos++;
  275. }
  276. syntaxError();
  277. }
  278. }
  279. function tryParseExportStatement () {
  280. const sStartPos = pos;
  281. const prevExport = exports.length;
  282. pos += 6;
  283. const curPos = pos;
  284. let ch = commentWhitespace(true);
  285. if (pos === curPos && !isPunctuator(ch))
  286. return;
  287. switch (ch) {
  288. // export default ...
  289. case 100/*d*/:
  290. addExport(pos, pos + 7, -1, -1);
  291. return;
  292. // export async? function*? name () {
  293. case 97/*a*/:
  294. pos += 5;
  295. commentWhitespace(true);
  296. // fallthrough
  297. case 102/*f*/:
  298. pos += 8;
  299. ch = commentWhitespace(true);
  300. if (ch === 42/***/) {
  301. pos++;
  302. ch = commentWhitespace(true);
  303. }
  304. const startPos = pos;
  305. ch = readToWsOrPunctuator(ch);
  306. addExport(startPos, pos, startPos, pos);
  307. pos--;
  308. return;
  309. // export class name ...
  310. case 99/*c*/:
  311. if (source.startsWith('lass', pos + 1) && isBrOrWsOrPunctuatorNotDot(source.charCodeAt(pos + 5))) {
  312. pos += 5;
  313. ch = commentWhitespace(true);
  314. const startPos = pos;
  315. ch = readToWsOrPunctuator(ch);
  316. addExport(startPos, pos, startPos, pos);
  317. pos--;
  318. return;
  319. }
  320. pos += 2;
  321. // fallthrough
  322. // export var/let/const name = ...(, name = ...)+
  323. case 118/*v*/:
  324. case 109/*l*/:
  325. // destructured initializations not currently supported (skipped for { or [)
  326. // also, lexing names after variable equals is skipped (export var p = function () { ... }, q = 5 skips "q")
  327. pos += 2;
  328. facade = false;
  329. do {
  330. pos++;
  331. ch = commentWhitespace(true);
  332. const startPos = pos;
  333. ch = readToWsOrPunctuator(ch);
  334. // dont yet handle [ { destructurings
  335. if (ch === 123/*{*/ || ch === 91/*[*/) {
  336. pos--;
  337. return;
  338. }
  339. if (pos === startPos)
  340. return;
  341. addExport(startPos, pos, startPos, pos);
  342. ch = commentWhitespace(true);
  343. if (ch === 61/*=*/) {
  344. pos--;
  345. return;
  346. }
  347. } while (ch === 44/*,*/);
  348. pos--;
  349. return;
  350. // export {...}
  351. case 123/*{*/:
  352. pos++;
  353. ch = commentWhitespace(true);
  354. while (true) {
  355. const startPos = pos;
  356. readToWsOrPunctuator(ch);
  357. const endPos = pos;
  358. commentWhitespace(true);
  359. ch = readExportAs(startPos, endPos);
  360. // ,
  361. if (ch === 44/*,*/) {
  362. pos++;
  363. ch = commentWhitespace(true);
  364. }
  365. if (ch === 125/*}*/)
  366. break;
  367. if (pos === startPos)
  368. return syntaxError();
  369. if (pos > end)
  370. return syntaxError();
  371. }
  372. pos++;
  373. ch = commentWhitespace(true);
  374. break;
  375. // export *
  376. // export * as X
  377. case 42/***/:
  378. pos++;
  379. commentWhitespace(true);
  380. ch = readExportAs(pos, pos);
  381. ch = commentWhitespace(true);
  382. break;
  383. }
  384. // from ...
  385. if (ch === 102/*f*/ && source.startsWith('rom', pos + 1)) {
  386. pos += 4;
  387. readImportString(sStartPos, commentWhitespace(true));
  388. // There were no local names.
  389. for (let i = prevExport; i < exports.length; ++i) {
  390. exports[i].ls = exports[i].le = -1;
  391. exports[i].ln = undefined;
  392. }
  393. }
  394. else {
  395. pos--;
  396. }
  397. }
  398. /*
  399. * Ported from Acorn
  400. *
  401. * MIT License
  402. * Copyright (C) 2012-2020 by various contributors (see AUTHORS)
  403. * Permission is hereby granted, free of charge, to any person obtaining a copy
  404. * of this software and associated documentation files (the "Software"), to deal
  405. * in the Software without restriction, including without limitation the rights
  406. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  407. * copies of the Software, and to permit persons to whom the Software is
  408. * furnished to do so, subject to the following conditions:
  409. * The above copyright notice and this permission notice shall be included in
  410. * all copies or substantial portions of the Software.
  411. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  412. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  413. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  414. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  415. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  416. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  417. * THE SOFTWARE.
  418. */
  419. let acornPos;
  420. function readString (start, quote) {
  421. acornPos = start;
  422. let out = '', chunkStart = acornPos;
  423. for (;;) {
  424. if (acornPos >= source.length) syntaxError();
  425. const ch = source.charCodeAt(acornPos);
  426. if (ch === quote) break;
  427. if (ch === 92) { // '\'
  428. out += source.slice(chunkStart, acornPos);
  429. out += readEscapedChar();
  430. chunkStart = acornPos;
  431. }
  432. else if (ch === 0x2028 || ch === 0x2029) {
  433. ++acornPos;
  434. }
  435. else {
  436. if (isBr(ch)) syntaxError();
  437. ++acornPos;
  438. }
  439. }
  440. out += source.slice(chunkStart, acornPos++);
  441. return out;
  442. }
  443. // Used to read escaped characters
  444. function readEscapedChar () {
  445. let ch = source.charCodeAt(++acornPos);
  446. ++acornPos;
  447. switch (ch) {
  448. case 110: return '\n'; // 'n' -> '\n'
  449. case 114: return '\r'; // 'r' -> '\r'
  450. case 120: return String.fromCharCode(readHexChar(2)); // 'x'
  451. case 117: return readCodePointToString(); // 'u'
  452. case 116: return '\t'; // 't' -> '\t'
  453. case 98: return '\b'; // 'b' -> '\b'
  454. case 118: return '\u000b'; // 'v' -> '\u000b'
  455. case 102: return '\f'; // 'f' -> '\f'
  456. case 13: if (source.charCodeAt(acornPos) === 10) ++acornPos; // '\r\n'
  457. case 10: // ' \n'
  458. return '';
  459. case 56:
  460. case 57:
  461. syntaxError();
  462. default:
  463. if (ch >= 48 && ch <= 55) {
  464. let octalStr = source.substr(acornPos - 1, 3).match(/^[0-7]+/)[0];
  465. let octal = parseInt(octalStr, 8);
  466. if (octal > 255) {
  467. octalStr = octalStr.slice(0, -1);
  468. octal = parseInt(octalStr, 8);
  469. }
  470. acornPos += octalStr.length - 1;
  471. ch = source.charCodeAt(acornPos);
  472. if (octalStr !== '0' || ch === 56 || ch === 57)
  473. syntaxError();
  474. return String.fromCharCode(octal);
  475. }
  476. if (isBr(ch)) {
  477. // Unicode new line characters after \ get removed from output in both
  478. // template literals and strings
  479. return '';
  480. }
  481. return String.fromCharCode(ch);
  482. }
  483. }
  484. // Used to read character escape sequences ('\x', '\u', '\U').
  485. function readHexChar (len) {
  486. const start = acornPos;
  487. let total = 0, lastCode = 0;
  488. for (let i = 0; i < len; ++i, ++acornPos) {
  489. let code = source.charCodeAt(acornPos), val;
  490. if (code === 95) {
  491. if (lastCode === 95 || i === 0) syntaxError();
  492. lastCode = code;
  493. continue;
  494. }
  495. if (code >= 97) val = code - 97 + 10; // a
  496. else if (code >= 65) val = code - 65 + 10; // A
  497. else if (code >= 48 && code <= 57) val = code - 48; // 0-9
  498. else break;
  499. if (val >= 16) break;
  500. lastCode = code;
  501. total = total * 16 + val;
  502. }
  503. if (lastCode === 95 || acornPos - start !== len) syntaxError();
  504. return total;
  505. }
  506. // Read a string value, interpreting backslash-escapes.
  507. function readCodePointToString () {
  508. const ch = source.charCodeAt(acornPos);
  509. let code;
  510. if (ch === 123) { // '{'
  511. ++acornPos;
  512. code = readHexChar(source.indexOf('}', acornPos) - acornPos);
  513. ++acornPos;
  514. if (code > 0x10FFFF) syntaxError();
  515. } else {
  516. code = readHexChar(4);
  517. }
  518. // UTF-16 Decoding
  519. if (code <= 0xFFFF) return String.fromCharCode(code);
  520. code -= 0x10000;
  521. return String.fromCharCode((code >> 10) + 0xD800, (code & 1023) + 0xDC00);
  522. }
  523. /*
  524. * </ Acorn Port>
  525. */
  526. function readExportAs (startPos, endPos) {
  527. let ch = source.charCodeAt(pos);
  528. let ls = startPos, le = endPos;
  529. if (ch === 97 /*a*/) {
  530. pos += 2;
  531. ch = commentWhitespace(true);
  532. startPos = pos;
  533. readToWsOrPunctuator(ch);
  534. endPos = pos;
  535. ch = commentWhitespace(true);
  536. }
  537. if (pos !== startPos)
  538. addExport(startPos, endPos, ls, le);
  539. return ch;
  540. }
  541. function readImportString (ss, ch) {
  542. const startPos = pos + 1;
  543. if (ch === 39/*'*/ || ch === 34/*"*/) {
  544. stringLiteral(ch);
  545. }
  546. else {
  547. syntaxError();
  548. return;
  549. }
  550. const impt = addImport(ss, startPos, pos, -1);
  551. readName(impt);
  552. pos++;
  553. ch = commentWhitespace(false);
  554. if (ch !== 97/*a*/ || !source.startsWith('ssert', pos + 1)) {
  555. pos--;
  556. return;
  557. }
  558. const assertIndex = pos;
  559. pos += 6;
  560. ch = commentWhitespace(true);
  561. if (ch !== 123/*{*/) {
  562. pos = assertIndex;
  563. return;
  564. }
  565. const assertStart = pos;
  566. do {
  567. pos++;
  568. ch = commentWhitespace(true);
  569. if (ch === 39/*'*/ || ch === 34/*"*/) {
  570. stringLiteral(ch);
  571. pos++;
  572. ch = commentWhitespace(true);
  573. }
  574. else {
  575. ch = readToWsOrPunctuator(ch);
  576. }
  577. if (ch !== 58/*:*/) {
  578. pos = assertIndex;
  579. return;
  580. }
  581. pos++;
  582. ch = commentWhitespace(true);
  583. if (ch === 39/*'*/ || ch === 34/*"*/) {
  584. stringLiteral(ch);
  585. }
  586. else {
  587. pos = assertIndex;
  588. return;
  589. }
  590. pos++;
  591. ch = commentWhitespace(true);
  592. if (ch === 44/*,*/) {
  593. pos++;
  594. ch = commentWhitespace(true);
  595. if (ch === 125/*}*/)
  596. break;
  597. continue;
  598. }
  599. if (ch === 125/*}*/)
  600. break;
  601. pos = assertIndex;
  602. return;
  603. } while (true);
  604. impt.a = assertStart;
  605. impt.se = pos + 1;
  606. }
  607. function commentWhitespace (br) {
  608. let ch;
  609. do {
  610. ch = source.charCodeAt(pos);
  611. if (ch === 47/*/*/) {
  612. const next_ch = source.charCodeAt(pos + 1);
  613. if (next_ch === 47/*/*/)
  614. lineComment();
  615. else if (next_ch === 42/***/)
  616. blockComment(br);
  617. else
  618. return ch;
  619. }
  620. else if (br ? !isBrOrWs(ch): !isWsNotBr(ch)) {
  621. return ch;
  622. }
  623. } while (pos++ < end);
  624. return ch;
  625. }
  626. function templateString () {
  627. while (pos++ < end) {
  628. const ch = source.charCodeAt(pos);
  629. if (ch === 36/*$*/ && source.charCodeAt(pos + 1) === 123/*{*/) {
  630. pos++;
  631. templateStack[templateStackDepth++] = templateDepth;
  632. templateDepth = ++openTokenDepth;
  633. return;
  634. }
  635. if (ch === 96/*`*/)
  636. return;
  637. if (ch === 92/*\*/)
  638. pos++;
  639. }
  640. syntaxError();
  641. }
  642. function blockComment (br) {
  643. pos++;
  644. while (pos++ < end) {
  645. const ch = source.charCodeAt(pos);
  646. if (!br && isBr(ch))
  647. return;
  648. if (ch === 42/***/ && source.charCodeAt(pos + 1) === 47/*/*/) {
  649. pos++;
  650. return;
  651. }
  652. }
  653. }
  654. function lineComment () {
  655. while (pos++ < end) {
  656. const ch = source.charCodeAt(pos);
  657. if (ch === 10/*\n*/ || ch === 13/*\r*/)
  658. return;
  659. }
  660. }
  661. function stringLiteral (quote) {
  662. while (pos++ < end) {
  663. let ch = source.charCodeAt(pos);
  664. if (ch === quote)
  665. return;
  666. if (ch === 92/*\*/) {
  667. ch = source.charCodeAt(++pos);
  668. if (ch === 13/*\r*/ && source.charCodeAt(pos + 1) === 10/*\n*/)
  669. pos++;
  670. }
  671. else if (isBr(ch))
  672. break;
  673. }
  674. syntaxError();
  675. }
  676. function regexCharacterClass () {
  677. while (pos++ < end) {
  678. let ch = source.charCodeAt(pos);
  679. if (ch === 93/*]*/)
  680. return ch;
  681. if (ch === 92/*\*/)
  682. pos++;
  683. else if (ch === 10/*\n*/ || ch === 13/*\r*/)
  684. break;
  685. }
  686. syntaxError();
  687. }
  688. function regularExpression () {
  689. while (pos++ < end) {
  690. let ch = source.charCodeAt(pos);
  691. if (ch === 47/*/*/)
  692. return;
  693. if (ch === 91/*[*/)
  694. ch = regexCharacterClass();
  695. else if (ch === 92/*\*/)
  696. pos++;
  697. else if (ch === 10/*\n*/ || ch === 13/*\r*/)
  698. break;
  699. }
  700. syntaxError();
  701. }
  702. function readToWsOrPunctuator (ch) {
  703. do {
  704. if (isBrOrWs(ch) || isPunctuator(ch))
  705. return ch;
  706. } while (ch = source.charCodeAt(++pos));
  707. return ch;
  708. }
  709. // Note: non-asii BR and whitespace checks omitted for perf / footprint
  710. // if there is a significant user need this can be reconsidered
  711. function isBr (c) {
  712. return c === 13/*\r*/ || c === 10/*\n*/;
  713. }
  714. function isWsNotBr (c) {
  715. return c === 9 || c === 11 || c === 12 || c === 32 || c === 160;
  716. }
  717. function isBrOrWs (c) {
  718. return c > 8 && c < 14 || c === 32 || c === 160;
  719. }
  720. function isBrOrWsOrPunctuatorNotDot (c) {
  721. return c > 8 && c < 14 || c === 32 || c === 160 || isPunctuator(c) && c !== 46/*.*/;
  722. }
  723. function keywordStart (pos) {
  724. return pos === 0 || isBrOrWsOrPunctuatorNotDot(source.charCodeAt(pos - 1));
  725. }
  726. function readPrecedingKeyword (pos, match) {
  727. if (pos < match.length - 1)
  728. return false;
  729. return source.startsWith(match, pos - match.length + 1) && (pos === 0 || isBrOrWsOrPunctuatorNotDot(source.charCodeAt(pos - match.length)));
  730. }
  731. function readPrecedingKeyword1 (pos, ch) {
  732. return source.charCodeAt(pos) === ch && (pos === 0 || isBrOrWsOrPunctuatorNotDot(source.charCodeAt(pos - 1)));
  733. }
  734. // Detects one of case, debugger, delete, do, else, in, instanceof, new,
  735. // return, throw, typeof, void, yield, await
  736. function isExpressionKeyword (pos) {
  737. switch (source.charCodeAt(pos)) {
  738. case 100/*d*/:
  739. switch (source.charCodeAt(pos - 1)) {
  740. case 105/*i*/:
  741. // void
  742. return readPrecedingKeyword(pos - 2, 'vo');
  743. case 108/*l*/:
  744. // yield
  745. return readPrecedingKeyword(pos - 2, 'yie');
  746. default:
  747. return false;
  748. }
  749. case 101/*e*/:
  750. switch (source.charCodeAt(pos - 1)) {
  751. case 115/*s*/:
  752. switch (source.charCodeAt(pos - 2)) {
  753. case 108/*l*/:
  754. // else
  755. return readPrecedingKeyword1(pos - 3, 101/*e*/);
  756. case 97/*a*/:
  757. // case
  758. return readPrecedingKeyword1(pos - 3, 99/*c*/);
  759. default:
  760. return false;
  761. }
  762. case 116/*t*/:
  763. // delete
  764. return readPrecedingKeyword(pos - 2, 'dele');
  765. default:
  766. return false;
  767. }
  768. case 102/*f*/:
  769. if (source.charCodeAt(pos - 1) !== 111/*o*/ || source.charCodeAt(pos - 2) !== 101/*e*/)
  770. return false;
  771. switch (source.charCodeAt(pos - 3)) {
  772. case 99/*c*/:
  773. // instanceof
  774. return readPrecedingKeyword(pos - 4, 'instan');
  775. case 112/*p*/:
  776. // typeof
  777. return readPrecedingKeyword(pos - 4, 'ty');
  778. default:
  779. return false;
  780. }
  781. case 110/*n*/:
  782. // in, return
  783. return readPrecedingKeyword1(pos - 1, 105/*i*/) || readPrecedingKeyword(pos - 1, 'retur');
  784. case 111/*o*/:
  785. // do
  786. return readPrecedingKeyword1(pos - 1, 100/*d*/);
  787. case 114/*r*/:
  788. // debugger
  789. return readPrecedingKeyword(pos - 1, 'debugge');
  790. case 116/*t*/:
  791. // await
  792. return readPrecedingKeyword(pos - 1, 'awai');
  793. case 119/*w*/:
  794. switch (source.charCodeAt(pos - 1)) {
  795. case 101/*e*/:
  796. // new
  797. return readPrecedingKeyword1(pos - 2, 110/*n*/);
  798. case 111/*o*/:
  799. // throw
  800. return readPrecedingKeyword(pos - 2, 'thr');
  801. default:
  802. return false;
  803. }
  804. }
  805. return false;
  806. }
  807. function isParenKeyword (curPos) {
  808. return source.charCodeAt(curPos) === 101/*e*/ && source.startsWith('whil', curPos - 4) ||
  809. source.charCodeAt(curPos) === 114/*r*/ && source.startsWith('fo', curPos - 2) ||
  810. source.charCodeAt(curPos - 1) === 105/*i*/ && source.charCodeAt(curPos) === 102/*f*/;
  811. }
  812. function isPunctuator (ch) {
  813. // 23 possible punctuator endings: !%&()*+,-./:;<=>?[]^{}|~
  814. return ch === 33/*!*/ || ch === 37/*%*/ || ch === 38/*&*/ ||
  815. ch > 39 && ch < 48 || ch > 57 && ch < 64 ||
  816. ch === 91/*[*/ || ch === 93/*]*/ || ch === 94/*^*/ ||
  817. ch > 122 && ch < 127;
  818. }
  819. function isExpressionPunctuator (ch) {
  820. // 20 possible expression endings: !%&(*+,-.:;<=>?[^{|~
  821. return ch === 33/*!*/ || ch === 37/*%*/ || ch === 38/*&*/ ||
  822. ch > 39 && ch < 47 && ch !== 41 || ch > 57 && ch < 64 ||
  823. ch === 91/*[*/ || ch === 94/*^*/ || ch > 122 && ch < 127 && ch !== 125/*}*/;
  824. }
  825. function isExpressionTerminator (curPos) {
  826. // detects:
  827. // => ; ) finally catch else
  828. // as all of these followed by a { will indicate a statement brace
  829. switch (source.charCodeAt(curPos)) {
  830. case 62/*>*/:
  831. return source.charCodeAt(curPos - 1) === 61/*=*/;
  832. case 59/*;*/:
  833. case 41/*)*/:
  834. return true;
  835. case 104/*h*/:
  836. return source.startsWith('catc', curPos - 4);
  837. case 121/*y*/:
  838. return source.startsWith('finall', curPos - 6);
  839. case 101/*e*/:
  840. return source.startsWith('els', curPos - 3);
  841. }
  842. return false;
  843. }
  844. function syntaxError () {
  845. throw Object.assign(new Error(`Parse error ${name}:${source.slice(0, pos).split('\n').length}:${pos - source.lastIndexOf('\n', pos - 1)}`), { idx: pos });
  846. }